aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/savebangs.c258
1 files changed, 186 insertions, 72 deletions
diff --git a/src/savebangs.c b/src/savebangs.c
index af8b7fa..df16711 100644
--- a/src/savebangs.c
+++ b/src/savebangs.c
@@ -31,20 +31,138 @@
* TODO: how does this behave in sub-patches?
* -> BUG: the depth should _really_ refer to the abstraction-depth
* else we get weird duplicates (most likely due to the "$0" trick
- *
- * TODO: make [savebangs] do something on top-level
- * that is: if the patch the [savebangs] is in gets saved, [savebangs] will fire
- * think (a little) about how the args to savebang have to look like to make it compatible with the [canvas*] stiff
- *
- * TODO: maintain our own list of [savebangs] to be called per abstraction rather than using the $0-trick
*/
#include "m_pd.h"
#include "g_canvas.h"
-/* ------------------------- help methods ---------------------------- */
+/* ------------------------- helper methods for callbacks ---------------------------- */
+typedef struct _savebangs_objlist {
+ const t_pd*obj;
+ struct _savebangs_objlist*next;
+} t_savebangs_objlist;
+
+typedef struct _savebangs_canvaslist {
+ const t_pd*parent;
+ t_savebangs_objlist*obj;
+
+ struct _savebangs_canvaslist*next;
+} t_savebangs_canvaslist;
+
+static t_savebangs_canvaslist*s_canvaslist=0;
+
+
+static t_savebangs_canvaslist*findCanvas(const t_pd*parent) {
+ t_savebangs_canvaslist*list=s_canvaslist;
+ if(0==parent || 0==list)
+ return 0;
+
+ for(list=s_canvaslist; list; list=list->next) {
+ if(parent == list->parent) {
+ return list;
+ }
+ }
+ return 0;
+}
+
+static t_savebangs_canvaslist*addCanvas(const t_pd*parent)
+{
+ t_savebangs_canvaslist*list=findCanvas(parent);
+ if(!list) {
+ list=(t_savebangs_canvaslist*)getbytes(sizeof(t_savebangs_canvaslist));
+ list->parent=parent;
+ list->obj=0;
+ list->next=0;
+
+ if(0==s_canvaslist) {
+ /* new list */
+ s_canvaslist=list;
+ } else {
+ /* add to the end of existing list */
+ t_savebangs_canvaslist*dummy=s_canvaslist;
+ while(dummy->next)
+ dummy=dummy->next;
+ dummy->next = list;
+ }
+ }
+ return list;
+}
+
+static t_savebangs_objlist*objectsInCanvas(const t_pd*parent) {
+ t_savebangs_canvaslist*list=findCanvas(parent);
+ if(list)
+ return list->obj;
+
+ return 0;
+}
+
+static void addObjectToCanvas(const t_pd*parent, const t_pd*obj) {
+ t_savebangs_canvaslist*p=addCanvas(parent);
+ t_savebangs_objlist*list=0;
+ t_savebangs_objlist*entry=0;
+ if(!p || !obj)
+ return;
+ list=p->obj;
+
+ if(list&&obj==list->obj)
+ return;
+
+ while(list && list->next) {
+ if(obj==list->obj) /* obj already in list */
+ return;
+ list=list->next;
+ }
+
+ /* we are at the end of the list that does not contain obj yet, so add it */
+ entry=(t_savebangs_objlist*)getbytes(sizeof(t_savebangs_objlist));
+ entry->obj=obj;
+ entry->next=0;
+ if(list) {
+ list->next=entry;
+ } else {
+ p->obj=entry;
+ }
+}
+
+static void removeObjectFromCanvas(const t_pd*parent, const t_pd*obj) {
+ t_savebangs_canvaslist*p=findCanvas(parent);
+ t_savebangs_objlist*list=0, *last=0, *next=0;
+ if(!p || !obj)return;
+ list=p->obj;
+ if(!list)
+ return;
+
+ while(list && obj!=list->obj) {
+ last=list;
+ list=list->next;
+ }
+
+ if(!list) /* couldn't find this object */
+ return;
+
+ next=list->next;
+
+ if(last)
+ last->next=next;
+ else
+ p->obj=next;
+
+ freebytes((void*)list, sizeof(t_savebangs_objlist));
+ list=0;
+}
+
+static void removeObjectFromCanvases(const t_pd*obj) {
+ t_savebangs_canvaslist*parents=s_canvaslist;
+
+ while(parents) {
+ removeObjectFromCanvas(parents->parent, obj);
+ parents=parents->next;
+ }
+}
+/* ------------------------- helper methods for savefunctions ---------------------------- */
+
typedef struct _savefuns {
t_class*class;
t_savefn savefn;
@@ -89,7 +207,27 @@ static void add_savefn(t_class*class)
}
}
+/* ------------------------- savefunctions ---------------------------- */
+
+static void orig_savefn(t_gobj*z, t_binbuf*b)
+{
+ t_class*class=z->g_pd;
+ t_savefn savefn=find_savefn(class);
+ if(savefn) {
+ savefn(z, b);
+ }
+}
+static void savebangs_bangem(t_savebangs_objlist*objs, int pst);
+static void savebangs_savefn(t_gobj*z, t_binbuf*b) {
+ /* z is the parent abstraction;
+ * we maintain a list of all [savebangs] within such each parent, in order to call all of them
+ */
+ t_savebangs_objlist*obj=objectsInCanvas((t_pd*)z);
+ savebangs_bangem(obj, 0);
+ orig_savefn(z, b);
+ savebangs_bangem(obj, 1);
+}
/* ------------------------- savebangs ---------------------------- */
@@ -98,66 +236,33 @@ static t_class *savebangs_class;
typedef struct _savebangs
{
t_object x_obj;
-
- t_symbol *x_d0;
t_outlet *x_pre, *x_post;
-
- t_savefn x_parentsavefn;
+ t_canvas *x_parent;
} t_savebangs;
-static void savebangs_free(t_savebangs *x)
+static void savebangs_bangs(t_savebangs*x, int pst)
{
- /* unpatch the parent canvas */
- if(x->x_d0) {
- pd_unbind(&x->x_obj.ob_pd, x->x_d0);
- }
-}
-
-static void orig_savefn(t_gobj*z, t_binbuf*b)
-{
- t_class*class=z->g_pd;
- t_savefn savefn=find_savefn(class);
- if(savefn) {
- savefn(z, b);
- }
+ if(!pst)
+ outlet_bang(x->x_pre);
+ else
+ outlet_bang(x->x_post);
}
-static void savebangs_savefn(t_gobj*z, t_binbuf*b) {
- /* argh: z is the abstraction! but we need to access ourselfs!
- * we handle this by binding to a special symbol. e.g. "$0 savebangs"
- * (we use the space between in order to make it hard for the ordinary user
- * to use this symbol for other things...
- */
-
- /* alternatively we could just search the abstraction for all instances of savebangs_class
- * and bang these;
- * but using the pd_bind-trick is simpler for now
- * though not as sweet, as somebody could use our bind-symbol for other things...
- */
-
- t_symbol*s_d0=canvas_realizedollar((t_canvas*)z, gensym("$0 savebangs"));
- t_atom ap[2];
- SETPOINTER(ap+0, (t_gpointer*)z);
- SETPOINTER(ap+1, (t_gpointer*)b);
-
- if(s_d0->s_thing) {
- pd_list(s_d0->s_thing, &s_list, 2, ap);
- } else {
- orig_savefn(z, b);
+static void savebangs_bangem(t_savebangs_objlist*objs, int pst) {
+ while(objs) {
+ t_savebangs*x=(t_savebangs*)objs->obj;
+ savebangs_bangs(x, pst);
+ objs=objs->next;
}
}
-static void savebangs_list(t_savebangs *x, t_symbol*s, int argc, t_atom*argv)
-{
- if(argv[0].a_type == A_POINTER && argv[1].a_type == A_POINTER) {
- t_gobj *z =(t_gobj*) argv[0].a_w.w_gpointer;
- t_binbuf*b =(t_binbuf*)argv[1].a_w.w_gpointer;
-
- outlet_bang(x->x_pre);
- orig_savefn(z, b);
- outlet_bang(x->x_post);
- }
+static void savebangs_mysavefn(t_gobj*z, t_binbuf*b) {
+ t_savebangs*x=(t_savebangs*)z;
+ int doit=(!x->x_parent);
+ if(doit)savebangs_bangs(x, 0);
+ orig_savefn(z, b);
+ if(doit)savebangs_bangs(x, 1);
}
static void *savebangs_new(t_floatarg f)
@@ -167,35 +272,44 @@ static void *savebangs_new(t_floatarg f)
t_canvas *canvas=(t_canvas*)glist_getcanvas(glist);
t_class *class = 0;
-
int depth=(int)f;
if(depth<0)depth=0;
- while(depth && canvas) {
- canvas=canvas->gl_owner;
+ if(depth) {
depth--;
- }
-
- if(canvas) {
- class=((t_gobj*)canvas)->g_pd;
- x->x_d0=canvas_realizedollar(canvas, gensym("$0 savebangs"));
- pd_bind(&x->x_obj.ob_pd, x->x_d0);
+ while(depth && canvas) {
+ canvas=canvas->gl_owner;
+ depth--;
+ }
- add_savefn(class);
- class_setsavefn(class, savebangs_savefn);
- } else {
- x->x_d0=0;
+ if(canvas) {
+ class=((t_gobj*)canvas)->g_pd;
+ add_savefn(class);
+ class_setsavefn(class, savebangs_savefn);
+
+ x->x_parent=canvas;
+ } else {
+ x->x_parent=0;
+ }
+
+ addObjectToCanvas((t_pd*)canvas, (t_pd*)x);
}
- x->x_pre=outlet_new(&x->x_obj, &s_bang);
x->x_post=outlet_new(&x->x_obj, &s_bang);
+ x->x_pre=outlet_new(&x->x_obj, &s_bang);
return (x);
}
+static void savebangs_free(t_savebangs *x)
+{
+ /* unpatch the parent canvas */
+ removeObjectFromCanvases((t_pd*)x);
+}
void savebangs_setup(void)
{
savebangs_class = class_new(gensym("savebangs"), (t_newmethod)savebangs_new,
(t_method)savebangs_free, sizeof(t_savebangs), CLASS_NOINLET, A_DEFFLOAT, 0);
- class_addlist(savebangs_class, savebangs_list);
+ add_savefn(savebangs_class);
+ class_setsavefn(savebangs_class, savebangs_mysavefn);
}