gambas-source-code/main/gbx/gbx_object.c
Benoît Minisini d3501bf140 [DEVELOPMENT ENVIRONMENT]
* NEW: Work continues on integrating the database manager.
* NEW: Some cosmetic changes in the way controls are drawing on the form
  editor.
* NEW: Panels with Border property set to None are now drawn with a light 
  border.
* BUG: Fix the "Show tab" button and menu.

[INTERPRETER]
* NEW: _attach is a new dynamic special method that is called when an 
  object is attached to or detached from its event observer. The first 
  argument of this method is the event observer, and the second argument 
  the event handler prefix.

[COMPILER]
* NEW: An expression can be a NEW instruction now. Beware that it does not 
  work inside braces.

[GB.DB]
* BUG: Fix an error message in the sqlite handler.

[GB.DB.FORM]
* NEW: DataSource.Table can now be any SQL query. The Filter property is 
  ignored in that case.
* BUG: Setting DataSource.Table to NULL correctly resets the DataSource and
  its children.
* NEW: DataView automatically adjusts the height of its rows to the 
  contents.
* NEW: DataSource.CacheSize is a new property to set the number of rows 
  stored in the internal DataSource cache. When this property is set to 
  zero, the cache size takes its default value (64 rows).

[GB.DB.SQLITE2]
* BUG: Fix a crash in datatype mapping.

[GB.DB.SQLITE3]
* BUG: Fix a crash in datatype mapping.

[GB.QT4]
* BUG: Window.AutoResize property works as expected now.
* OPT: Some optimizations in GridView.
* NEW: GridView.Rows[].Visible returns if a specific row is visible.
* NEW: GridView.Rows[].EnsureVisible ensures that a specific row is 
  visible.
* BUG: Draw.Style.Panel draws the same thing as a panel border now.
* BUG: Window.Closed always returns the accurate value now.


git-svn-id: svn://localhost/gambas/trunk@2108 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2009-07-12 21:49:13 +00:00

473 lines
9.6 KiB
C
Raw Blame History

/***************************************************************************
Object.c
Object management routines
(c) 2000-2007 Benoit Minisini <gambas@users.sourceforge.net>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 1, or (at your option)
any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
***************************************************************************/
#define __OBJECT_C
#include "gb_common.h"
#include "gb_alloc.h"
#include "gb_list.h"
#include "gbx_class.h"
#include "gbx_event.h"
#include "gbx_exec.h"
#include "gbx_compare.h"
#include "gbx_c_gambas.h"
#include "gbx_object.h"
#if DEBUG_REF
const char *OBJECT_ref_where = 0;
#endif
static OBJECT *EventObject = NULL;
void OBJECT_new(void **ptr, CLASS *class, const char *name, OBJECT *parent)
{
OBJECT_alloc(ptr, class, class->size);
//#if DEBUG_EVENT
//printf("OBJECT_new: %s %p %d\n", class->name, *ptr, class->off_event);
//#endif
OBJECT_attach(*ptr, parent, name);
}
void OBJECT_alloc(void **ptr, CLASS *class, size_t size)
{
OBJECT *object;
if (size < sizeof(OBJECT))
size = sizeof(OBJECT);
ALLOC_ZERO(&object, size, "OBJECT_alloc");
object->class = class;
#if DEBUG_REF
object->ref = 0;
OBJECT_REF(object, "OBJECT_alloc");
#else
object->ref = 1;
#endif
class->count++;
*ptr = object;
}
#if 0
static void dump_attach(char *title)
{
void *ob;
fprintf(stderr, ">>>> %s: ", title);
for (ob = EventObject; ob; ob = OBJECT_event(ob)->next)
fprintf(stderr, "%p -> ", ob);
fprintf(stderr, "(nil)\n");
}
#endif
void OBJECT_detach(OBJECT *ob)
{
CLASS *class = OBJECT_class(ob);
OBJECT *parent;
OBJECT_EVENT *ev;
bool lock;
ev = (OBJECT_EVENT *)((char *)ob + class->off_event);
//if (!ev->parent)
// return;
if (ev->prev)
OBJECT_event(ev->prev)->next = ev->next;
if (ev->next)
OBJECT_event(ev->next)->prev = ev->prev;
if (ob == EventObject)
EventObject = ev->next;
ev->prev = NULL;
ev->next = NULL;
//dump_attach("OBJECT_detach");
// Do not free the observers there anymore
/* Avoids an infinite recursion, if freeing the parent implies freeing the object */
parent = OBJECT_parent(ob);
if (parent)
{
if (class->special[SPEC_ATTACH] != NO_SYMBOL)
{
STACK_check(2);
SP->_object.class = OBJECT_class(parent);
SP->_object.object = parent;
PUSH();
SP->type = T_NULL;
SP++;
EXEC_special(SPEC_ATTACH, class, ob, 2, TRUE);
}
lock = OBJECT_is_locked(ob);
ev->parent = NULL;
OBJECT_lock(ob, lock);
#if DEBUG_EVENT || DEBUG_REF
fprintf(stderr, "OBJECT_detach : Detach (%s %p) from (%s %p)\n",
ob->class->name, ob, parent->class->name, parent);
#endif
OBJECT_UNREF(parent, "OBJECT_detach");
}
}
static void remove_observers(OBJECT *ob)
{
CLASS *class = OBJECT_class(ob);
OBJECT_EVENT *ev;
COBSERVER *obs, *next;
ev = (OBJECT_EVENT *)((char *)ob + class->off_event);
obs = ev->observer;
while (obs)
{
next = obs->list.next;
#if DEBUG_EVENT
fprintf(stderr, "Remove observer %p\n", obs);
#endif
OBJECT_UNREF(obs, "remove_observers");
obs = next;
}
ev->observer = NULL;
}
void OBJECT_attach(OBJECT *ob, OBJECT *parent, const char *name)
{
CLASS *class = OBJECT_class(ob);
OBJECT_EVENT *ev;
bool lock;
if (EVENT_Name != name)
{
STRING_unref(&EVENT_Name);
if (name)
STRING_new(&EVENT_Name, name, 0);
else
EVENT_Name = NULL;
}
if (!name)
return;
lock = OBJECT_is_locked(ob);
OBJECT_detach(ob);
ev = (OBJECT_EVENT *)((char *)ob + class->off_event);
ev->parent = parent;
OBJECT_lock(ob, lock);
#if DEBUG_EVENT || DEBUG_REF
fprintf(stderr, "OBJECT_attach : Attach (%s %p) to (%s %p) as %s\n",
ob->class->name, ob, parent->class->name, parent, name);
#endif
OBJECT_REF(parent, "OBJECT_attach");
EVENT_search(class, ev->event, name, parent);
ev->next = EventObject;
ev->prev = NULL;
if (EventObject)
OBJECT_event(EventObject)->prev = ob;
EventObject = ob;
if (class->special[SPEC_ATTACH] != NO_SYMBOL)
{
STACK_check(2);
SP->_object.class = OBJECT_class(parent);
SP->_object.object = parent;
PUSH();
SP->type = T_STRING;
SP->_string.addr = EVENT_Name;
SP->_string.start = 0;
SP->_string.len = STRING_length(EVENT_Name);
PUSH();
EXEC_special(SPEC_ATTACH, class, ob, 2, TRUE);
}
//dump_attach("OBJECT_attach");
}
void OBJECT_free(CLASS *class, OBJECT *ob)
{
/* Il faut emp<6D>her EXEC_leave() de relib<69>er l'objet */
/*((OBJECT *)free_ptr)->ref = 1;*/
/*OBJECT_REF(ob, "OBJECT_free");*/
/* Ex<45>ution de la m<>hode _free pour toutes les classes parentes
puis pour la classe de l'objet */
/*if (CLASS_is_native(class))
EXEC_special_inheritance(SPEC_FREE, class, ob, 0, TRUE);*/
OBJECT_detach(ob);
remove_observers(ob);
class->count--;
#if DEBUG_REF
ob->class = (CLASS *)0x23232323;
#endif
FREE(&ob, "OBJECT_free");
}
bool OBJECT_comp_value(VALUE *ob1, VALUE *ob2)
{
if (ob1->type == T_NULL && ob2->type == T_NULL)
return FALSE;
else if (ob1->type == T_NULL)
return ob2->_object.object != NULL;
else if (ob2->type == T_NULL)
return ob1->_object.object != NULL;
else
return COMPARE_object(&ob1->_object.object, &ob2->_object.object);
}
static void release(CLASS *class, OBJECT *ob)
{
int i;
CLASS_VAR *var;
int nelt;
char *data;
if (class->parent != NULL && ob)
release(class->parent, ob);
if (CLASS_is_native(class))
return;
if (ob == NULL)
{
var = class->load->stat;
nelt = class->load->n_stat;
data = class->stat;
}
else
{
var = class->load->dyn;
nelt = class->load->n_dyn;
data = (char *)ob;
}
for (i = 0; i < nelt; i++)
{
#if TRACE_MEMORY
if (var->type.id == T_STRING || var->type.id == T_OBJECT)
{
printf("release: %s %p [%d] trying %p\n", class->name, ob, i, (*(void **)&data[var->pos]));
fflush(NULL);
}
#endif
if (var->type.id == T_STRING)
STRING_unref((char **)&data[var->pos]);
else if (var->type.id == T_OBJECT)
{
OBJECT_UNREF(*((void **)&data[var->pos]), "release");
}
else if (var->type.id == T_VARIANT)
VARIANT_free((VARIANT *)&data[var->pos]);
else if (var->type.id == T_ARRAY)
ARRAY_free_data((void *)&data[var->pos], (ARRAY_DESC *)class->load->array[var->type.value]);
var++;
}
}
void OBJECT_release(CLASS *class, OBJECT *ob)
{
#if TRACE_MEMORY
printf("> OBJECT_release %s %p\n", class->name, ob);
#endif
release(class, ob);
if (ob)
OBJECT_free(class, ob);
#if TRACE_MEMORY
printf("< OBJECT_release %s %p\n", class->name, ob);
#endif
}
void OBJECT_exit(void)
{
#if DEBUG_LOAD
fprintf(stderr, "------------ OBJECT_exit - BEGIN---------\n");
#endif
while (EventObject)
OBJECT_detach(EventObject);
#if DEBUG_LOAD
fprintf(stderr, "------------ OBJECT_exit - END ----------\n");
#endif
}
void OBJECT_create(void **object, CLASS *class, const char *name, void *parent, int nparam)
{
void *ob;
if (class->no_create)
THROW(E_CSTATIC, class->name);
TRY
{
OBJECT_new(&ob, class, name, parent);
*object = ob;
OBJECT_lock(ob, TRUE);
EXEC_special_inheritance(SPEC_NEW, class, ob, nparam, TRUE);
OBJECT_lock(ob, FALSE);
OBJECT_UNREF_KEEP(ob, "OBJECT_create");
}
CATCH
{
*object = NULL;
OBJECT_UNREF_KEEP(ob, "OBJECT_create");
PROPAGATE();
}
END_TRY
}
/* FIXME: The _new are methods called differently from EXEC_special_inheritance */
void OBJECT_create_native(void **object, CLASS *class, VALUE *param)
{
CLASS_DESC *desc;
short index;
OBJECT_new(object, class, NULL, NULL);
for(;;)
{
index = class->special[SPEC_NEW];
if (index != NO_SYMBOL)
{
desc = CLASS_get_desc(class, index);
EXEC_call_native(desc->method.exec, *object, desc->method.type, param);
}
class = class->parent;
if (!class)
break;
}
OBJECT_UNREF_KEEP(*object, "OBJECT_create");
}
void OBJECT_lock(OBJECT *object, bool block)
{
CLASS *class;
if (!object)
return;
class = object->class;
if (class->is_observer)
{
COBSERVER_lock((COBSERVER *)object, block);
return;
}
if (class->n_event == 0)
return;
// fprintf(stderr, "OBJECT_lock: (%s %p) %s\n", class->name, object, block ? "lock" : "unlock");
if (block)
OBJECT_event(object)->parent = (OBJECT *)((intptr_t)OBJECT_event(object)->parent | 1);
else
OBJECT_event(object)->parent = (OBJECT *)((intptr_t)OBJECT_event(object)->parent & ~1);
}
bool OBJECT_is_locked(OBJECT *object)
{
CLASS *class;
if (!object)
return FALSE;
class = object->class;
if (class->is_observer)
return COBSERVER_is_locked((COBSERVER *)object);
if (class->n_event == 0)
return FALSE;
return (((intptr_t)OBJECT_event(object)->parent & 1) != 0);
}
OBJECT *OBJECT_parent(void *object)
{
//if (!OBJECT_has_events(object))
// return NULL;
return ((OBJECT *)((intptr_t)OBJECT_event(object)->parent & ~1));
}
OBJECT *OBJECT_active_parent(void *object)
{
OBJECT *parent = OBJECT_parent(object);
if (!parent || OBJECT_is_locked((OBJECT *)object) || OBJECT_is_locked(parent))
return NULL;
return parent;
}