/* -*- Mode: C; c-basic-offset: 4 -*-
* pygtk- Python bindings for the GTK toolkit.
* Copyright (C) 1998-2003 James Henstridge
*
* pygboxed.c: wrapper for GBoxed
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
* USA
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include "pygobject-private.h"
static void
pyg_boxed_dealloc(PyGBoxed *self)
{
if (self->free_on_dealloc && self->boxed) {
PyGILState_STATE state = pyg_gil_state_ensure();
g_boxed_free(self->gtype, self->boxed);
pyg_gil_state_release(state);
}
self->ob_type->tp_free((PyObject *)self);
}
static int
pyg_boxed_compare(PyGBoxed *self, PyGBoxed *v)
{
if (self->boxed == v->boxed) return 0;
if (self->boxed > v->boxed) return -1;
return 1;
}
static long
pyg_boxed_hash(PyGBoxed *self)
{
return (long)self->boxed;
}
static PyObject *
pyg_boxed_repr(PyGBoxed *self)
{
gchar buf[128];
g_snprintf(buf, sizeof(buf), "<%s at 0x%lx>", g_type_name(self->gtype),
(long)self->boxed);
return PyString_FromString(buf);
}
static int
pyg_boxed_init(PyGBoxed *self, PyObject *args, PyObject *kwargs)
{
gchar buf[512];
if (!PyArg_ParseTuple(args, ":GBoxed.__init__"))
return -1;
self->boxed = NULL;
self->gtype = 0;
self->free_on_dealloc = FALSE;
g_snprintf(buf, sizeof(buf), "%s can not be constructed", self->ob_type->tp_name);
PyErr_SetString(PyExc_NotImplementedError, buf);
return -1;
}
static void
pyg_boxed_free(PyObject *op)
{
PyObject_FREE(op);
}
static PyObject *
pyg_boxed_copy(PyGBoxed *self)
{
return pyg_boxed_new (self->gtype, self->boxed, TRUE, TRUE);
}
static PyMethodDef pygboxed_methods[] = {
{ "copy", (PyCFunction) pyg_boxed_copy, METH_NOARGS },
{ NULL, NULL, 0 }
};
PyTypeObject PyGBoxed_Type = {
PyObject_HEAD_INIT(NULL)
0, /* ob_size */
"gobject.GBoxed", /* tp_name */
sizeof(PyGBoxed), /* tp_basicsize */
0, /* tp_itemsize */
/* methods */
(destructor)pyg_boxed_dealloc, /* tp_dealloc */
(printfunc)0, /* tp_print */
(getattrfunc)0, /* tp_getattr */
(setattrfunc)0, /* tp_setattr */
(cmpfunc)pyg_boxed_compare, /* tp_compare */
(reprfunc)pyg_boxed_repr, /* tp_repr */
0, /* tp_as_number */
0, /* tp_as_sequence */
0, /* tp_as_mapping */
(hashfunc)pyg_boxed_hash, /* tp_hash */
(ternaryfunc)0, /* tp_call */
(reprfunc)0, /* tp_str */
(getattrofunc)0, /* tp_getattro */
(setattrofunc)0, /* tp_setattro */
0, /* tp_as_buffer */
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /* tp_flags */
NULL, /* Documentation string */
(traverseproc)0, /* tp_traverse */
(inquiry)0, /* tp_clear */
(richcmpfunc)0, /* tp_richcompare */
0, /* tp_weaklistoffset */
(getiterfunc)0, /* tp_iter */
(iternextfunc)0, /* tp_iternext */
pygboxed_methods, /* tp_methods */
0, /* tp_members */
0, /* tp_getset */
(PyTypeObject *)0, /* tp_base */
(PyObject *)0, /* tp_dict */
0, /* tp_descr_get */
0, /* tp_descr_set */
0, /* tp_dictoffset */
(initproc)pyg_boxed_init, /* tp_init */
(allocfunc)0, /* tp_alloc */
(newfunc)0, /* tp_new */
(freefunc)pyg_boxed_free, /* tp_free */
(inquiry)0, /* tp_is_gc */
(PyObject *)0, /* tp_bases */
};
/**
* pyg_register_boxed:
* @dict: the module dictionary to store the wrapper class.
* @class_name: the Python name for the wrapper class.
* @boxed_type: the GType of the boxed type being wrapped.
* @type: the wrapper class.
*
* Registers a wrapper for a boxed type. The wrapper class will be a
* subclass of gobject.GBoxed, and a reference to the wrapper class
* will be stored in the provided module dictionary.
*/
void
pyg_register_boxed(PyObject *dict, const gchar *class_name,
GType boxed_type, PyTypeObject *type)
{
PyObject *o;
g_return_if_fail(dict != NULL);
g_return_if_fail(class_name != NULL);
g_return_if_fail(boxed_type != 0);
if (!type->tp_dealloc) type->tp_dealloc = (destructor)pyg_boxed_dealloc;
type->ob_type = &PyType_Type;
type->tp_base = &PyGBoxed_Type;
if (PyType_Ready(type) < 0) {
g_warning("could not get type `%s' ready", type->tp_name);
return;
}
PyDict_SetItemString(type->tp_dict, "__gtype__",
o=pyg_type_wrapper_new(boxed_type));
Py_DECREF(o);
g_type_set_qdata(boxed_type, pygboxed_type_key, type);
PyDict_SetItemString(dict, (char *)class_name, (PyObject *)type);
}
/**
* pyg_boxed_new:
* @boxed_type: the GType of the boxed value.
* @boxed: the boxed value.
* @copy_boxed: whether the new boxed wrapper should hold a copy of the value.
* @own_ref: whether the boxed wrapper should own the boxed value.
*
* Creates a wrapper for a boxed value. If @copy_boxed is set to
* True, the wrapper will hold a copy of the value, instead of the
* value itself. If @own_ref is True, then the value held by the
* wrapper will be freed when the wrapper is deallocated. If
* @copy_boxed is True, then @own_ref must also be True.
*
* Returns: the boxed wrapper.
*/
PyObject *
pyg_boxed_new(GType boxed_type, gpointer boxed, gboolean copy_boxed,
gboolean own_ref)
{
PyGILState_STATE state;
PyGBoxed *self;
PyTypeObject *tp;
g_return_val_if_fail(boxed_type != 0, NULL);
g_return_val_if_fail(!copy_boxed || (copy_boxed && own_ref), NULL);
state = pyg_gil_state_ensure();
if (!boxed) {
Py_INCREF(Py_None);
pyg_gil_state_release(state);
return Py_None;
}
tp = g_type_get_qdata(boxed_type, pygboxed_type_key);
if (!tp)
tp = (PyTypeObject *)&PyGBoxed_Type; /* fallback */
self = PyObject_NEW(PyGBoxed, tp);
if (self == NULL) {
pyg_gil_state_release(state);
return NULL;
}
if (copy_boxed)
boxed = g_boxed_copy(boxed_type, boxed);
self->boxed = boxed;
self->gtype = boxed_type;
self->free_on_dealloc = own_ref;
pyg_gil_state_release(state);
return (PyObject *)self;
}
syntax highlighted by Code2HTML, v. 0.9.1