/* 

flext - C++ layer for Max/MSP and pd (pure data) externals

Copyright (c) 2001-2005 Thomas Grill (gr@grrrr.org)
For information on usage and redistribution, and for a DISCLAIMER OF ALL
WARRANTIES, see the file, "license.txt," in this distribution.  

*/

/*! \file flbind.cpp
    \brief Functionality for symbol-bound methods.
*/
 
#include "flext.h"
#include "flinternal.h"


t_class *flext_base::pxbnd_class = NULL;

#if FLEXT_SYS == FLEXT_SYS_MAX
t_object *px_freelist = NULL;
t_messlist px_messlist[3];
#endif

/*! \brief Set up the proxy class for symbol-bound methods
*/
void flext_base::SetupBindProxy()
{
    // already initialized?
    if(!pxbnd_class) {
#if FLEXT_SYS == FLEXT_SYS_PD
        pxbnd_class = class_new(gensym("flext_base bind proxy"),NULL,NULL,sizeof(pxbnd_object),CLASS_PD|CLASS_NOINLET, A_NULL);
        add_anything(pxbnd_class,pxbnd_object::px_method); // for symbol-bound methods
#elif FLEXT_SYS == FLEXT_SYS_MAX
        pxbnd_class = new t_class;

        pxbnd_class->c_sym = const_cast<t_symbol *>(sym__);
        pxbnd_class->c_freelist = &px_freelist;
        pxbnd_class->c_freefun = NULL;
        pxbnd_class->c_size = sizeof(pxbnd_object);
        pxbnd_class->c_tiny = 0;
        pxbnd_class->c_noinlet = 1;
        px_messlist[0].m_sym = (t_symbol *)pxbnd_class;

        px_messlist[1].m_sym = const_cast<t_symbol *>(sym_anything);
        px_messlist[1].m_fun = (method)pxbnd_object::px_method;
        px_messlist[1].m_type[0] = A_GIMME;
        px_messlist[1].m_type[1] = 0;

        px_messlist[2].m_sym = 0;
#else
#pragma warning("Not implemented!")
#endif
    }
}


flext_base::BindItem::BindItem(bool (*f)(flext_base *,t_symbol *s,int,t_atom *,void *data),pxbnd_object *p):
    Item(NULL),fun(f),px(p)
{}

flext_base::BindItem::~BindItem()
{
    if(px) {
        FLEXT_ASSERT(!fun); // check if already unbound
        object_free(&px->obj);
    }
}

void flext_base::BindItem::Unbind(const t_symbol *tag)
{
    if(px) {
        FLEXT_ASSERT(fun);

#if FLEXT_SYS == FLEXT_SYS_PD
        pd_unbind(&px->obj.ob_pd,const_cast<t_symbol *>(tag)); 
#elif FLEXT_SYS == FLEXT_SYS_MAX
        if(tag->s_thing == (t_object *)px) 
            const_cast<t_symbol *>(tag)->s_thing = NULL; 
        else
            error("flext - Binding to symbol %s not found",tag->s_name);
#else
#           pragma warning("Not implemented")
#endif

        fun = NULL;
    }
}

#if FLEXT_SYS == FLEXT_SYS_PD
    //! Bind object to a symbol
    bool flext_base::Bind(const t_symbol *sym) { pd_bind(&thisHdr()->ob_pd,const_cast<t_symbol *>(sym)); return true; }
    //! Unbind object from a symbol
    bool flext_base::Unbind(const t_symbol *sym) { pd_unbind(&thisHdr()->ob_pd,const_cast<t_symbol *>(sym)); return true; }
#elif FLEXT_SYS == FLEXT_SYS_MAX
    //! Bind object to a symbol
    bool flext_base::Bind(const t_symbol *sym) { if(sym->s_thing) return false; else { const_cast<t_symbol *>(sym)->s_thing = (t_object *)thisHdr(); return true; } }
    //! Unbind object from a symbol
    bool flext_base::Unbind(const t_symbol *sym) { if(sym->s_thing != (t_object *)thisHdr()) return false; else { const_cast<t_symbol *>(sym)->s_thing = NULL; return true; } }
#endif

bool flext_base::BindMethod(const t_symbol *sym,bool (*fun)(flext_base *,t_symbol *s,int argc,t_atom *argv,void *data),void *data)
{
    if(!bindhead) 
        bindhead = new ItemCont;
    else {
        // Search for symbol
        for(Item *it = bindhead->FindList(sym); it; it = it->nxt) {
            BindItem *item = (BindItem *)it;

            // go through all items with matching tag
            if(item->fun == fun) {
                // function already registered -> bail out!
                post("%s - Symbol already bound with this method",thisName());
                return false;
            }
        }
    }

    SetupBindProxy(); 

#if FLEXT_SYS == FLEXT_SYS_PD
    pxbnd_object *px = (pxbnd_object *)object_new(pxbnd_class);
#elif FLEXT_SYS == FLEXT_SYS_MAX
    pxbnd_object *px = (pxbnd_object *)newobject(px_messlist);
#else
#pragma warning("Not implemented!")
#endif

    if(px) {
        BindItem *mi = new BindItem(fun,px);
        bindhead->Add(mi,sym);

        px->init(this,mi,data);

#if FLEXT_SYS == FLEXT_SYS_PD
        pd_bind(&px->obj.ob_pd,const_cast<t_symbol *>(sym)); 
#elif FLEXT_SYS == FLEXT_SYS_MAX
        if(!sym->s_thing) 
            const_cast<t_symbol *>(sym)->s_thing = (t_object *)px;
        else
            error("%s - Symbol is already bound",thisName());
#else
#       pragma warning("Not implemented")
#endif
    }
    else 
        error("%s - Symbol proxy could not be created",thisName());

    return true;
}

bool flext_base::UnbindMethod(const t_symbol *sym,bool (*fun)(flext_base *,t_symbol *s,int argc,t_atom *argv,void *data),void **data)
{
    bool ok = false;
    
    if(bindhead && bindhead->Contained(0)) {
        ItemSet &set = bindhead->GetInlet();

/*
        ItemSet::iterator it1,it2;
        if(sym) { 
            // specific tag
            it1 = it2 = set.find(sym); it2++; 
        }
        else { 
            // any tag
            it1 = set.begin(),it2 = set.end(); 
        }

        BindItem *it = NULL;
        for(ItemSet::iterator si = it1; si != it2 && !it; ++si) {
            for(Item *i = si.data(); i; i = i->nxt) {
                BindItem *item = (BindItem *)i;
                if(!fun || item->fun == fun) 
                { 
                    it = item; 
                    if(!sym) sym = si.key();
                    break; 
                }
            }
        }
*/
        BindItem *item = NULL;
        if(sym) {
            // symbol is given
            Item *it = set.find(sym);
            if(fun) {
                // check if function matches
                for(; it && static_cast<BindItem *>(it)->fun != fun; it = it->nxt) {}
            }
            item = static_cast<BindItem *>(it); 
        }
        else {
            // take any entry that matches
            for(ItemSet::iterator si(set); si && !item; ++si) {
                for(Item *i = si.data(); i; i = i->nxt) {
                    BindItem *bit = (BindItem *)i;
                    if(!fun || bit->fun == fun) { 
                        item = bit; 
                        if(!sym) sym = si.key();
                        break; 
                    }
                }
            }
        }

        if(item) {
            if(data) *data = item->px->data;
            ok = bindhead->Remove(item,sym,0,false);
            if(ok) {
                item->Unbind(sym);
                delete item;
            }
        }
    }
    return ok;
}

bool flext_base::GetBoundMethod(const t_symbol *sym,bool (*fun)(flext_base *,t_symbol *s,int argc,t_atom *argv,void *data),void *&data)
{
    if(bindhead) {
        // Search for symbol
        for(Item *it = bindhead->FindList(sym); it; it = it->nxt) {
            BindItem *item = (BindItem *)it;

            // go through all items with matching tag
            if(item->fun == fun) {
                data = item->px->data;
                return true;
            }
        }
    }
    return false;
}

bool flext_base::UnbindAll()
{
    if(bindhead && bindhead->Contained(0)) {
        ItemSet &set = bindhead->GetInlet();
//        for(ItemSet::iterator si = set.begin(); si != set.end(); ++si) {
        for(ItemSet::iterator si(set); si; ++si) {
            Item *lst = si.data();
            while(lst) {
                Item *nxt = lst->nxt;
                BindItem *it = (BindItem *)lst;
                it->Unbind(si.key());
                delete it;
                lst = nxt;
            }
        }
        set.clear();
    }
    return true;
}

void flext_base::pxbnd_object::px_method(pxbnd_object *c,const t_symbol *s,int argc,t_atom *argv)
{
    c->item->fun(c->base,(t_symbol *)s,argc,(t_atom *)argv,c->data);
}