##// END OF EJS Templates
added initial generated wrappers for Qt 4.4.3...
added initial generated wrappers for Qt 4.4.3 git-svn-id: svn://svn.code.sf.net/p/pythonqt/code/trunk@49 ea8d5007-eb21-0410-b261-ccb3ea6e24a9

File last commit:

r10:21d959e02207
r13:d46b01f7163a
Show More
PythonQtWrapper.cpp
484 lines | 16.4 KiB | text/x-c | CppLexer
/ src / PythonQtWrapper.cpp
/*
*
* Copyright (C) 2006 MeVis Research GmbH All Rights Reserved.
*
* 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.
*
* Further, this software is distributed without any warranty that it is
* free of the rightful claim of any third person regarding infringement
* or the like. Any license provided herein, whether implied or
* otherwise, applies only to this software file. Patent licenses, if
* any, provided herein do not apply to combinations of this program with
* other software, or any other product whatsoever.
*
* 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
*
* Contact information: MeVis Research GmbH, Universitaetsallee 29,
* 28359 Bremen, Germany or:
*
* http://www.mevis.de
*
*/
//----------------------------------------------------------------------------------
/*!
// \file PythonQtWrapper.cpp
// \author Florian Link
// \author Last changed by $Author: florian $
// \date 2006-05
*/
//----------------------------------------------------------------------------------
#include "PythonQtWrapper.h"
#include <QObject>
#include "PythonQt.h"
#include "PythonQtSlot.h"
#include "PythonQtClassInfo.h"
#include "PythonQtConversion.h"
static void PythonQtWrapper_deleteObject(PythonQtWrapper* self) {
if (self->_wrappedPtr) {
//mlabDebugConst("Python","c++ wrapper removed " << self->_wrappedPtr << " " << self->_obj->className() << " " << self->_info->wrappedClassName().latin1());
PythonQt::priv()->removeWrapperPointer(self->_wrappedPtr);
// we own our qobject, so we delete it now:
delete self->_obj;
self->_obj = NULL;
if (self->_ownedByPythonQt) {
int type = self->_info->metaTypeId();
if (self->_useQMetaTypeDestroy && type>=0) {
// use QMetaType to destroy the object
QMetaType::destroy(type, self->_wrappedPtr);
} else {
PythonQtSlotInfo* slot = PythonQt::priv()->getDestructorSlot(self->_info->className());
if (slot) {
void* args[2];
args[0] = NULL;
args[1] = &self->_wrappedPtr;
slot->decorator()->qt_metacall(QMetaObject::InvokeMetaMethod, slot->slotIndex(), args);
self->_wrappedPtr = NULL;
} else {
if (type>=0) {
// use QMetaType to destroy the object
QMetaType::destroy(type, self->_wrappedPtr);
} else {
// TODO: warn about not being able to destroy the object?
}
}
}
}
} else {
//mlabDebugConst("Python","qobject wrapper removed " << self->_obj->className() << " " << self->_info->wrappedClassName().latin1());
if (self->_objPointerCopy) {
PythonQt::priv()->removeWrapperPointer(self->_objPointerCopy);
}
if (self->_obj) {
if (self->_ownedByPythonQt) {
if (!self->_obj->parent()) {
delete self->_obj;
}
} else {
if (self->_obj->parent()==NULL) {
// tell someone who is interested that the qobject is no longer wrapped, if it has no parent
PythonQt::qObjectNoLongerWrappedCB(self->_obj);
}
}
}
}
self->_obj = NULL;
}
static void PythonQtWrapper_dealloc(PythonQtWrapper* self)
{
PythonQtWrapper_deleteObject(self);
self->_obj.~QPointer<QObject>();
self->ob_type->tp_free((PyObject*)self);
}
static PyObject* PythonQtWrapper_new(PyTypeObject *type, PyObject * /*args*/, PyObject * /*kwds*/)
{
PythonQtWrapper *self;
self = (PythonQtWrapper *)type->tp_alloc(type, 0);
if (self != NULL) {
self->_info = NULL;
new (&self->_obj) QPointer<QObject>();
self->_wrappedPtr = NULL;
self->_ownedByPythonQt = false;
self->_useQMetaTypeDestroy = false;
}
return (PyObject *)self;
}
static int PythonQtWrapper_init(PythonQtWrapper * /*self*/, PyObject * /*args*/, PyObject * /*kwds*/)
{
return 0;
}
static PyObject *PythonQtWrapper_classname(PythonQtWrapper* type)
{
return PyString_FromString(type->_info->className());
}
static PyObject *PythonQtWrapper_help(PythonQtWrapper* type)
{
return PythonQt::self()->helpCalled(type->_info);
}
static PyObject *PythonQtWrapper_delete(PythonQtWrapper * self)
{
PythonQtWrapper_deleteObject(self);
Py_INCREF(Py_None);
return Py_None;
}
static PyMethodDef PythonQtWrapper_methods[] = {
{"className", (PyCFunction)PythonQtWrapper_classname, METH_NOARGS,
"Return the classname of the object"
},
{"help", (PyCFunction)PythonQtWrapper_help, METH_NOARGS,
"Shows the help of available methods for this class"
},
{"delete", (PyCFunction)PythonQtWrapper_delete, METH_NOARGS,
"Deletes the C++ object (at your own risk, my friend!)"
},
{NULL, NULL, 0, NULL} /* Sentinel */
};
static PyObject *PythonQtWrapper_getattro(PyObject *obj,PyObject *name)
{
const char *attributeName;
PythonQtWrapper *wt = (PythonQtWrapper *)obj;
if ((attributeName = PyString_AsString(name)) == NULL) {
return NULL;
}
if (!wt->_obj && !wt->_wrappedPtr) {
QString error = QString("Trying to read attribute '") + attributeName + "' from a destroyed " + wt->_info->className() + " object";
PyErr_SetString(PyExc_ValueError, error.toLatin1().data());
return NULL;
}
// mlabDebugConst("Python","get " << attributeName);
// TODO: dynamic properties are missing
PythonQtMemberInfo member = wt->_info->member(attributeName);
switch (member._type) {
case PythonQtMemberInfo::Property:
if (wt->_obj) {
return PythonQtConv::QVariantToPyObject(member._property.read(wt->_obj));
}
break;
case PythonQtMemberInfo::Slot:
return PythonQtSlotFunction_New(member._slot, obj, NULL);
break;
case PythonQtMemberInfo::EnumValue:
return PyInt_FromLong(member._enumValue);
break;
default:
// is an invalid type, go on
break;
}
// look for the interal methods (className(), help())
PyObject* internalMethod = Py_FindMethod( PythonQtWrapper_methods, obj, (char*)attributeName);
if (internalMethod) {
return internalMethod;
}
PyErr_Clear();
if (wt->_obj) {
// look for a child
QObjectList children = wt->_obj->children();
for (int i = 0; i < children.count(); i++) {
QObject *child = children.at(i);
if (child->objectName() == attributeName) {
return PythonQt::self()->priv()->wrapQObject(child);
}
}
}
if (qstrcmp(attributeName, "__dict__")==0) {
QStringList l = wt->_info->memberList(false);
PyObject* dict = PyDict_New();
foreach (QString name, l) {
//PyObject* o = PyObject_GetAttrString(obj, name.toLatin1().data());
PyDict_SetItemString(dict, name.toLatin1().data(), Py_None);
//Py_DECREF(o);
}
// Note: we do not put children into the dict, is would look confusing?!
return dict;
}
QString error = QString(wt->_info->className()) + " has no attribute named '" + QString(attributeName) + "'";
PyErr_SetString(PyExc_AttributeError, error.toLatin1().data());
return NULL;
}
static int PythonQtWrapper_setattro(PyObject *obj,PyObject *name,PyObject *value)
{
QString error;
char *attributeName;
PythonQtWrapper *wt = (PythonQtWrapper *)obj;
if ((attributeName = PyString_AsString(name)) == NULL)
return -1;
if (!wt->_obj) {
error = QString("Trying to set attribute '") + attributeName + "' on a destroyed " + wt->_info->className() + " object";
PyErr_SetString(PyExc_AttributeError, error.toLatin1().data());
return -1;
}
PythonQtMemberInfo member = wt->_info->member(attributeName);
if (member._type == PythonQtMemberInfo::Property) {
QMetaProperty prop = member._property;
if (prop.isWritable()) {
QVariant v;
if (prop.isEnumType()) {
// this will give us either a string or an int, everything else will probably be an error
v = PythonQtConv::PyObjToQVariant(value);
} else {
int t = prop.userType();
v = PythonQtConv::PyObjToQVariant(value, t);
}
bool success = false;
if (v.isValid()) {
success = prop.write(wt->_obj, v);
}
if (success) {
return 0;
} else {
error = QString("Property '") + attributeName + "' of type '" +
prop.typeName() + "' does not accept an object of type "
+ QString(value->ob_type->tp_name) + " (" + PythonQtConv::PyObjGetRepresentation(value) + ")";
}
} else {
error = QString("Property '") + attributeName + "' of " + wt->_info->className() + " object is not writable";
}
} else {
if (member._type == PythonQtMemberInfo::Slot) {
error = QString("Slot '") + attributeName + "' can not be overwritten on " + wt->_info->className() + " object";
} else if (member._type == PythonQtMemberInfo::EnumValue) {
error = QString("EnumValue '") + attributeName + "' can not be overwritten on " + wt->_info->className() + " object";
}
}
PyErr_SetString(PyExc_AttributeError, error.toLatin1().data());
return -1;
}
static PyObject * PythonQtWrapper_str(PyObject * obj)
{
PythonQtWrapper* wt = (PythonQtWrapper*)obj;
QObject *qobj = wt->_obj;
if (wt->_wrappedPtr) {
QString str = PythonQtConv::CPPObjectToString(wt->_info->metaTypeId(), wt->_wrappedPtr);
if (!str.isEmpty()) {
return PyString_FromFormat("%s", str.toLatin1().constData());
} else
if (wt->_obj) {
return PyString_FromFormat("%s (C++ Object %p wrapped by %s %p))", wt->_info->className(), wt->_wrappedPtr, wt->_obj->metaObject()->className(), qobj);
} else {
return PyString_FromFormat("%s (C++ Object %p)", wt->_info->className(), wt->_wrappedPtr);
}
} else {
return PyString_FromFormat("%s (QObject %p)", wt->_info->className(), qobj);
}
}
static PyObject * PythonQtWrapper_repr(PyObject * obj)
{
PythonQtWrapper* wt = (PythonQtWrapper*)obj;
QObject *qobj = wt->_obj;
if (wt->_wrappedPtr) {
QString str = PythonQtConv::CPPObjectToString(wt->_info->metaTypeId(), wt->_wrappedPtr);
if (!str.isEmpty()) {
return PyString_FromFormat("%s(%s, %p)", QMetaType::typeName(wt->_info->metaTypeId()), str.toLatin1().constData(), wt->_wrappedPtr);
} else
if (wt->_obj) {
return PyString_FromFormat("%s (C++ Object %p wrapped by %s %p))", wt->_info->className(), wt->_wrappedPtr, wt->_obj->metaObject()->className(), qobj);
} else {
return PyString_FromFormat("%s (C++ Object %p)", wt->_info->className(), wt->_wrappedPtr);
}
} else {
return PyString_FromFormat("%s (QObject %p)", wt->_info->className(), qobj);
}
}
static int PythonQtWrapper_compare(PyObject * obj1, PyObject * obj2)
{
if (obj1->ob_type == &PythonQtWrapper_Type &&
obj2->ob_type == &PythonQtWrapper_Type) {
PythonQtWrapper* w1 = (PythonQtWrapper*)obj1;
PythonQtWrapper* w2 = (PythonQtWrapper*)obj2;
// check pointers directly first:
if (w1->_wrappedPtr != NULL) {
if (w1->_wrappedPtr == w2->_wrappedPtr) {
return 0;
}
} else if (w1->_obj == w2->_obj) {
return 0;
}
const char* class1 = w1->_info->className();
const char* class2 = w2->_info->className();
if (strcmp(class1, class2) == 0) {
// same class names, so we can try the operator_equal
PythonQtMemberInfo info = w1->_info->member("operator_equal");
if (info._type == PythonQtMemberInfo::Slot) {
bool result = false;
void* obj1 = w1->_wrappedPtr;
if (!obj1) {
obj1 = w1->_obj;
}
if (!obj1) { return -1; }
void* obj2 = w2->_wrappedPtr;
if (!obj2) {
obj2 = w2->_obj;
}
if (!obj2) { return -1; }
if (info._slot->isInstanceDecorator()) {
// call on decorator QObject
void* args[3];
args[0] = &result;
args[1] = &obj1; // this is a pointer, so it needs a pointer to a pointer
args[2] = obj2; // this is a reference, so it needs the direct pointer
info._slot->decorator()->qt_metacall(QMetaObject::InvokeMetaMethod, info._slot->slotIndex(), args);
return result?0:-1;
} else {
// call directly on QObject
if (w1->_obj && w2->_obj) {
void* args[2];
args[0] = &result;
args[2] = obj2; // this is a reference, so it needs the direct pointer
w1->_obj->qt_metacall(QMetaObject::InvokeMetaMethod, info._slot->slotIndex(), args);
}
}
}
}
}
return -1;
}
static int PythonQtWrapper_nonzero(PyObject *obj)
{
PythonQtWrapper* wt = (PythonQtWrapper*)obj;
return (wt->_wrappedPtr == NULL && wt->_obj == NULL)?0:1;
}
static long PythonQtWrapper_hash(PythonQtWrapper *obj)
{
if (obj->_wrappedPtr != NULL) {
return reinterpret_cast<long>(obj->_wrappedPtr);
} else {
QObject* qobj = obj->_obj; // get pointer from QPointer wrapper
return reinterpret_cast<long>(qobj);
}
}
// we override nb_nonzero, so that one can do 'if' expressions to test for a NULL ptr
static PyNumberMethods PythonQtWrapper_as_number = {
0, /* nb_add */
0, /* nb_subtract */
0, /* nb_multiply */
0, /* nb_divide */
0, /* nb_remainder */
0, /* nb_divmod */
0, /* nb_power */
0, /* nb_negative */
0, /* nb_positive */
0, /* nb_absolute */
PythonQtWrapper_nonzero, /* nb_nonzero */
0, /* nb_invert */
0, /* nb_lshift */
0, /* nb_rshift */
0, /* nb_and */
0, /* nb_xor */
0, /* nb_or */
0, /* nb_coerce */
0, /* nb_int */
0, /* nb_long */
0, /* nb_float */
0, /* nb_oct */
0, /* nb_hex */
0, /* nb_inplace_add */
0, /* nb_inplace_subtract */
0, /* nb_inplace_multiply */
0, /* nb_inplace_divide */
0, /* nb_inplace_remainder */
0, /* nb_inplace_power */
0, /* nb_inplace_lshift */
0, /* nb_inplace_rshift */
0, /* nb_inplace_and */
0, /* nb_inplace_xor */
0, /* nb_inplace_or */
0, /* nb_floor_divide */
0, /* nb_true_divide */
0, /* nb_inplace_floor_divide */
0, /* nb_inplace_true_divide */
};
PyTypeObject PythonQtWrapper_Type = {
PyObject_HEAD_INIT(NULL)
0, /*ob_size*/
"PythonQt.PythonQtWrapper", /*tp_name*/
sizeof(PythonQtWrapper), /*tp_basicsize*/
0, /*tp_itemsize*/
(destructor)PythonQtWrapper_dealloc, /*tp_dealloc*/
0, /*tp_print*/
0, /*tp_getattr*/
0, /*tp_setattr*/
PythonQtWrapper_compare, /*tp_compare*/
PythonQtWrapper_repr, /*tp_repr*/
&PythonQtWrapper_as_number, /*tp_as_number*/
0, /*tp_as_sequence*/
0, /*tp_as_mapping*/
(hashfunc)PythonQtWrapper_hash, /*tp_hash */
0, /*tp_call*/
PythonQtWrapper_str, /*tp_str*/
PythonQtWrapper_getattro, /*tp_getattro*/
PythonQtWrapper_setattro, /*tp_setattro*/
0, /*tp_as_buffer*/
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /*tp_flags*/
"PythonQtWrapper object", /* tp_doc */
0, /* tp_traverse */
0, /* tp_clear */
0, /* tp_richcompare */
0, /* tp_weaklistoffset */
0, /* tp_iter */
0, /* tp_iternext */
0, /* tp_methods */
0, /* tp_members */
0, /* tp_getset */
0, /* tp_base */
0, /* tp_dict */
0, /* tp_descr_get */
0, /* tp_descr_set */
0, /* tp_dictoffset */
(initproc)PythonQtWrapper_init, /* tp_init */
0, /* tp_alloc */
PythonQtWrapper_new, /* tp_new */
};
//-------------------------------------------------------