aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--cursor-help.pd17
-rw-r--r--cursor.c286
2 files changed, 303 insertions, 0 deletions
diff --git a/cursor-help.pd b/cursor-help.pd
new file mode 100644
index 0000000..9b04a7a
--- /dev/null
+++ b/cursor-help.pd
@@ -0,0 +1,17 @@
+#N canvas 0 22 454 304 10;
+#X msg 178 124 start;
+#X msg 227 124 stop;
+#X obj 142 123 tgl 15 0 empty empty empty 0 -6 0 8 -262144 -1 -1 0
+1;
+#X obj 157 151 cursor 80 80;
+#X floatatom 99 242 5 0 0 0 - - -;
+#X floatatom 239 242 5 0 0 0 - - -;
+#X obj 131 258 print LEFT;
+#X obj 226 265 print RIGHT;
+#X connect 0 0 3 0;
+#X connect 1 0 3 0;
+#X connect 2 0 3 0;
+#X connect 3 0 4 0;
+#X connect 3 0 6 0;
+#X connect 3 1 5 0;
+#X connect 3 1 7 0;
diff --git a/cursor.c b/cursor.c
new file mode 100644
index 0000000..f75d0eb
--- /dev/null
+++ b/cursor.c
@@ -0,0 +1,286 @@
+/* (C) Guenter Geiger <geiger@xdv.org> */
+
+
+#include <m_pd.h>
+#include "g_canvas.h"
+
+/* ------------------------ cursor ----------------------------- */
+
+
+#define BACKGROUNDCOLOR "grey"
+
+#define DEFAULTSIZE 80
+
+static t_class *cursor_class;
+
+typedef struct _cursor
+{
+ t_object x_obj;
+ t_glist * x_glist;
+ t_outlet* out2;
+ int x_width;
+ int x_height;
+ int x;
+ int y;
+} t_cursor;
+
+/* widget helper functions */
+
+void cursor_drawme(t_cursor *x, t_glist *glist, int firsttime)
+{
+ if (firsttime) {
+ sys_vgui(".x%x.c create rectangle %d %d %d %d -tags %xS -fill %s\n",
+ glist_getcanvas(glist),
+ x->x_obj.te_xpix, x->x_obj.te_ypix,
+ x->x_obj.te_xpix + x->x_width, x->x_obj.te_ypix + x->x_height,
+ x,BACKGROUNDCOLOR);
+ }
+ else {
+ sys_vgui(".x%x.c coords %xS \
+%d %d %d %d\n",
+ glist_getcanvas(glist), x,
+ x->x_obj.te_xpix, x->x_obj.te_ypix,
+ x->x_obj.te_xpix + x->x_width, x->x_obj.te_ypix + x->x_height);
+ }
+
+ {
+ /* outlets */
+ int n = 2;
+ int nplus, i;
+ nplus = (n == 1 ? 1 : n-1);
+ for (i = 0; i < n; i++)
+ {
+ int onset = x->x_obj.te_xpix + (x->x_width - IOWIDTH) * i / nplus;
+ if (firsttime)
+ sys_vgui(".x%x.c create rectangle %d %d %d %d -tags %xo%d\n",
+ glist_getcanvas(glist),
+ onset, x->x_obj.te_ypix + x->x_height - 1,
+ onset + IOWIDTH, x->x_obj.te_ypix + x->x_height,
+ x, i);
+ else
+ sys_vgui(".x%x.c coords %xo%d %d %d %d %d\n",
+ glist_getcanvas(glist), x, i,
+ onset, x->x_obj.te_ypix + x->x_height - 1,
+ onset + IOWIDTH, x->x_obj.te_ypix + x->x_height);
+ }
+ /* inlets */
+ n = 0;
+ nplus = (n == 1 ? 1 : n-1);
+ for (i = 0; i < n; i++)
+ {
+ int onset = x->x_obj.te_xpix + (x->x_width - IOWIDTH) * i / nplus;
+ if (firsttime)
+ sys_vgui(".x%x.c create rectangle %d %d %d %d -tags %xi%d\n",
+ glist_getcanvas(glist),
+ onset, x->x_obj.te_ypix,
+ onset + IOWIDTH, x->x_obj.te_ypix + 1,
+ x, i);
+ else
+ sys_vgui(".x%x.c coords %xi%d %d %d %d %d\n",
+ glist_getcanvas(glist), x, i,
+ onset, x->x_obj.te_ypix,
+ onset + IOWIDTH, x->x_obj.te_ypix + 1);
+
+ }
+ }
+
+}
+
+
+
+
+void cursor_erase(t_cursor* x,t_glist* glist)
+{
+ int n;
+ sys_vgui(".x%x.c delete %xS\n",
+ glist_getcanvas(glist), x);
+ n = 2;
+ while (n--) {
+ sys_vgui(".x%x.c delete %xo%d\n",glist_getcanvas(glist),x,n);
+ }
+}
+
+
+
+/* ------------------------ cursor widgetbehaviour----------------------------- */
+
+
+static void cursor_getrect(t_gobj *z, t_glist *owner,
+ int *xp1, int *yp1, int *xp2, int *yp2)
+{
+ int width, height;
+ t_cursor* s = (t_cursor*)z;
+
+
+ width = s->x_width;
+ height = s->x_height;
+ *xp1 = s->x_obj.te_xpix;
+ *yp1 = s->x_obj.te_ypix;
+ *xp2 = s->x_obj.te_xpix + width;
+ *yp2 = s->x_obj.te_ypix + height;
+}
+
+static void cursor_displace(t_gobj *z, t_glist *glist,
+ int dx, int dy)
+{
+ t_cursor *x = (t_cursor *)z;
+ x->x_obj.te_xpix += dx;
+ x->x_obj.te_ypix += dy;
+ cursor_drawme(x, glist, 0);
+ canvas_fixlinesfor(glist_getcanvas(glist),(t_text*) x);
+}
+
+static void cursor_select(t_gobj *z, t_glist *glist, int state)
+{
+ t_cursor *x = (t_cursor *)z;
+ sys_vgui(".x%x.c itemconfigure %xS -fill %s\n", glist,
+ x, (state? "blue" : BACKGROUNDCOLOR));
+}
+
+static void cursor_motion(t_cursor *x, t_floatarg dx, t_floatarg dy)
+{
+ x->x += dx;
+ x->y += dy;
+ outlet_float(x->out2,x->y);
+ outlet_float(x->x_obj.ob_outlet,x->x);
+}
+
+static void cursor_start(t_cursor *x)
+{
+ glist_grab(x->x_glist, &x->x_obj.te_g, (t_glistmotionfn) cursor_motion,
+ (t_glistkeyfn) NULL, NULL, NULL);
+
+/* x->x = xpos - x->x_obj.te_xpix; */
+/* x->y = ypos - x->x_obj.te_ypix; */
+/* outlet_float(x->out2,x->y); */
+/* outlet_float(x->x_obj.ob_outlet,x->x); */
+}
+
+static void cursor_activate(t_gobj *z, t_glist *glist, int state)
+{
+/* t_text *x = (t_text *)z;
+ t_rtext *y = glist_findrtext(glist, x);
+ if (z->g_pd != gatom_class) rtext_activate(y, state);*/
+ t_cursor *x = (t_cursor *) z;
+ cursor_start(x);
+}
+
+static void cursor_delete(t_gobj *z, t_glist *glist)
+{
+ t_text *x = (t_text *)z;
+ canvas_deletelinesfor(glist_getcanvas(glist), x);
+}
+
+
+static void cursor_vis(t_gobj *z, t_glist *glist, int vis)
+{
+ t_cursor* s = (t_cursor*)z;
+ if (vis)
+ cursor_drawme(s, glist, 1);
+ else
+ cursor_erase(s,glist);
+}
+
+/* can we use the normal text save function ?? */
+
+static void cursor_save(t_gobj *z, t_binbuf *b)
+{
+ t_cursor *x = (t_cursor *)z;
+ binbuf_addv(b, "ssiisii", gensym("#X"),gensym("obj"),
+ (t_int)x->x_obj.te_xpix, (t_int)x->x_obj.te_ypix,
+ gensym("cursor"),x->x_width,x->x_height);
+ binbuf_addv(b, ";");
+}
+
+
+t_widgetbehavior cursor_widgetbehavior;
+
+void cursor_key(t_cursor *x, t_floatarg f)
+{
+ post("key");
+}
+
+
+
+static void cursor_click(t_cursor *x,
+ t_floatarg xpos, t_floatarg ypos, t_floatarg shift, t_floatarg ctrl,
+ t_floatarg alt)
+{
+ glist_grab(x->x_glist, &x->x_obj.te_g, (t_glistmotionfn) cursor_motion,
+ (t_glistkeyfn) NULL, xpos, ypos);
+
+ x->x = xpos - x->x_obj.te_xpix;
+ x->y = ypos - x->x_obj.te_ypix;
+ outlet_float(x->out2,x->y);
+ outlet_float(x->x_obj.ob_outlet,x->x);
+}
+
+static int cursor_newclick(t_gobj *z, struct _glist *glist,
+ int xpix, int ypix, int shift, int alt, int dbl, int doit)
+{
+ if (doit)
+ cursor_click((t_cursor *)z, (t_floatarg)xpix, (t_floatarg)ypix,
+ (t_floatarg)shift, 0, (t_floatarg)alt);
+ return (1);
+}
+
+void cursor_size(t_cursor* x,t_floatarg w,t_floatarg h) {
+ x->x_width = w;
+ x->x_height = h;
+ cursor_drawme(x, x->x_glist, 0);
+}
+
+static void cursor_setwidget(void)
+{
+ cursor_widgetbehavior.w_getrectfn = cursor_getrect;
+ cursor_widgetbehavior.w_displacefn = cursor_displace;
+ cursor_widgetbehavior.w_selectfn = cursor_select;
+ cursor_widgetbehavior.w_activatefn = cursor_activate;
+ cursor_widgetbehavior.w_deletefn = cursor_delete;
+ cursor_widgetbehavior.w_visfn = cursor_vis;
+ cursor_widgetbehavior.w_clickfn = cursor_newclick;
+}
+
+
+static void *cursor_new(t_floatarg h,t_floatarg o)
+{
+ t_cursor *x = (t_cursor *)pd_new(cursor_class);
+
+ x->x_glist = (t_glist*) canvas_getcurrent();
+ if (h) x->x_width = h;
+ else
+ x->x_width = DEFAULTSIZE;
+
+ if (o) x->x_height = o;
+ else
+ x->x_height = DEFAULTSIZE;
+
+ outlet_new(&x->x_obj, &s_float);
+ x->out2 = outlet_new(&x->x_obj, &s_float);
+
+ return (x);
+}
+
+void cursor_setup(void)
+{
+ cursor_class = class_new(gensym("cursor"), (t_newmethod)cursor_new, 0,
+ sizeof(t_cursor),0, A_DEFFLOAT,A_DEFFLOAT,0);
+
+ class_addcreator((t_newmethod)cursor_new,gensym("bng"),A_DEFSYM,A_DEFFLOAT,A_DEFFLOAT,A_GIMME,0);
+
+ class_addmethod(cursor_class, (t_method)cursor_click, gensym("click"),
+ A_FLOAT, A_FLOAT, A_FLOAT, A_FLOAT, A_FLOAT, 0);
+ class_addmethod(cursor_class, (t_method)cursor_size, gensym("size"),
+ A_FLOAT, A_FLOAT, 0);
+
+ class_addmethod(cursor_class,(t_method) cursor_start,gensym("start"),0);
+/* class_addfloat(cursor_class,(t_method) cursor_float); */
+
+ cursor_setwidget();
+ class_setwidget(cursor_class,&cursor_widgetbehavior);
+#if PD_MINOR_VERSION >= 37
+ class_setsavefn(cursor_class,&cursor_save);
+#endif
+}
+
+