/* * * Copyright (C) 2010 MeVis Medical Solutions AG 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 Medical Solutions AG, Universitaetsallee 29, * 28359 Bremen, Germany or: * * http://www.mevis.de * */ //---------------------------------------------------------------------------------- /*! // \file PythonQtClassWrapper.cpp // \author Florian Link // \author Last changed by $Author: florian $ // \date 2006-05 */ //---------------------------------------------------------------------------------- #include "PythonQtClassWrapper.h" #include #include "PythonQt.h" #include "PythonQtSlot.h" #include "PythonQtSignal.h" #include "PythonQtClassInfo.h" #include "PythonQtConversion.h" #include "PythonQtInstanceWrapper.h" static PyObject* PythonQtInstanceWrapper_invert(PythonQtInstanceWrapper* wrapper) { PyObject* result = nullptr; static QByteArray memberName = "__invert__"; PythonQtMemberInfo opSlot = wrapper->classInfo()->member(memberName); if (opSlot._type == PythonQtMemberInfo::Slot) { result = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, nullptr, nullptr, wrapper->_wrappedPtr); } return result; } static PyObject* PythonQtInstanceWrapper_negative(PythonQtInstanceWrapper* wrapper) { PyObject* result = nullptr; static QByteArray memberName = "__sub__"; PythonQtMemberInfo opSlot = wrapper->classInfo()->member(memberName); if (opSlot._type == PythonQtMemberInfo::Slot) { result = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, nullptr, nullptr, wrapper->_wrappedPtr); } return result; } static PyObject* PythonQtInstanceWrapper_positive(PythonQtInstanceWrapper* wrapper) { PyObject* result = nullptr; static QByteArray memberName = "__add__"; PythonQtMemberInfo opSlot = wrapper->classInfo()->member(memberName); if (opSlot._type == PythonQtMemberInfo::Slot) { result = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, nullptr, nullptr, wrapper->_wrappedPtr); } return result; } static int PythonQtInstanceWrapper_nonzero(PythonQtInstanceWrapper* wrapper) { int result = (wrapper->_wrappedPtr == nullptr && wrapper->_obj == nullptr) ? 0 : 1; if (result) { static QByteArray memberName = "__nonzero__"; PythonQtMemberInfo opSlot = wrapper->classInfo()->member(memberName); if (opSlot._type == PythonQtMemberInfo::Slot) { PyObject* resultObj = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, nullptr, nullptr, wrapper->_wrappedPtr); if (resultObj == Py_False) { result = 0; } Py_XDECREF(resultObj); } } return result; } static Py_ssize_t PythonQtInstanceWrapper_length(PythonQtInstanceWrapper* wrapper) { qint64 result = -1; if (wrapper->_wrappedPtr != nullptr || wrapper->_obj != nullptr) { static QByteArray memberName = "__len__"; PythonQtMemberInfo opSlot = wrapper->classInfo()->member(memberName); if (opSlot._type == PythonQtMemberInfo::Slot) { PyObject* resultObj = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, nullptr, nullptr, wrapper->_wrappedPtr); bool ok; result = PythonQtConv::PyObjGetLongLong(resultObj, false, ok); if (!ok) { result = -1; } Py_XDECREF(resultObj); } } return result; } static int PythonQtInstanceWrapper_setitem(PyObject* self, PyObject* index, PyObject* value) { PythonQtInstanceWrapper* wrapper = (PythonQtInstanceWrapper*)self; bool isSetItem = value; const char* methodName = isSetItem ? "__setitem__" : "__delitem__"; PythonQtMemberInfo opSlot = wrapper->classInfo()->member(methodName); if (opSlot._type == PythonQtMemberInfo::Slot) { PyObject* args = PyTuple_New(isSetItem ? 2 : 1); Py_INCREF(index); PyTuple_SET_ITEM(args, 0, index); if (isSetItem) { Py_INCREF(value); PyTuple_SET_ITEM(args, 1, value); } PyObject* result = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, args, nullptr, wrapper->_wrappedPtr); if (result) { Py_DECREF(result); } Py_DECREF(args); return PyErr_Occurred() ? -1 : 0; } else { QString e = QString("No method '%1' on class '%2'").arg(methodName).arg(QString(wrapper->classInfo()->className())); PyErr_SetString(PyExc_AttributeError, QStringToPythonConstCharPointer(e)); return -1; } } static PyObject* PythonQtInstanceWrapper_binaryfunc(PyObject* self, PyObject* other, const QByteArray& opName, const QByteArray& fallbackOpName = QByteArray()) { // since we disabled type checking, we can receive any object as self, but we currently only support // different objects on the right. Otherwise we would need to generate __radd__ etc. methods. if (!PyObject_TypeCheck(self, &PythonQtInstanceWrapper_Type)) { QString error = "Unsupported operation " + opName + "(" + self->ob_type->tp_name + ", " + other->ob_type->tp_name + ")"; PyErr_SetString(PyExc_ArithmeticError, QStringToPythonCharPointer(error)); return nullptr; } PythonQtInstanceWrapper* wrapper = (PythonQtInstanceWrapper*)self; PyObject* result = nullptr; PythonQtMemberInfo opSlot = wrapper->classInfo()->member(opName); if (opSlot._type == PythonQtMemberInfo::Slot) { PyObject* args = PyTuple_New(1); Py_INCREF(other); PyTuple_SET_ITEM(args, 0, other); result = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, args, nullptr, wrapper->_wrappedPtr); Py_DECREF(args); if (!result && !fallbackOpName.isEmpty()) { // try fallback if we did not get a result result = PythonQtInstanceWrapper_binaryfunc(self, other, fallbackOpName); } } return result; } static PyObject* PythonQtInstanceWrapper_mul(PyObject* self, PyObject* other) { // since we disabled type checking, we can receive any object as self, // in case of multiplication we support this by switching the types: if (!PyObject_TypeCheck(self, &PythonQtInstanceWrapper_Type)) { PyObject* tmp; tmp = self; self = other; other = tmp; } PythonQtInstanceWrapper* wrapper = (PythonQtInstanceWrapper*)self; PyObject* result = nullptr; PythonQtMemberInfo opSlot = wrapper->classInfo()->member("__mul__"); if (opSlot._type == PythonQtMemberInfo::Slot) { PyObject* args = PyTuple_New(1); Py_INCREF(other); PyTuple_SET_ITEM(args, 0, other); result = PythonQtSlotFunction_CallImpl(wrapper->classInfo(), wrapper->_obj, opSlot._slot, args, nullptr, wrapper->_wrappedPtr); Py_DECREF(args); } return result; } #define BINARY_OP(NAME) \ static PyObject* PythonQtInstanceWrapper_##NAME(PyObject* self, PyObject* other) \ { \ static const QByteArray opName("__" #NAME "__"); \ return PythonQtInstanceWrapper_binaryfunc(self, other, opName); \ } #define BINARY_OP_INPLACE(NAME) \ static PyObject* PythonQtInstanceWrapper_i##NAME(PyObject* self, PyObject* other) \ { \ static const QByteArray opName("__i" #NAME "__"); \ static const QByteArray fallbackName("__" #NAME "__"); \ return PythonQtInstanceWrapper_binaryfunc(self, other, opName, fallbackName); \ } BINARY_OP(add) BINARY_OP(sub) BINARY_OP(div) BINARY_OP(and) BINARY_OP(or) BINARY_OP(xor) BINARY_OP(mod) BINARY_OP(lshift) BINARY_OP(rshift) BINARY_OP(getitem) BINARY_OP_INPLACE(add) BINARY_OP_INPLACE(sub) BINARY_OP_INPLACE(mul) BINARY_OP_INPLACE(div) BINARY_OP_INPLACE(and) BINARY_OP_INPLACE(or) BINARY_OP_INPLACE(xor) BINARY_OP_INPLACE(mod) BINARY_OP_INPLACE(lshift) BINARY_OP_INPLACE(rshift) static void initializeSlots(PythonQtClassWrapper* wrap) { int typeSlots = wrap->classInfo()->typeSlots(); if (typeSlots) { if (typeSlots & PythonQt::Type_MappingGetItem) { wrap->_base.as_mapping.mp_subscript = (binaryfunc)PythonQtInstanceWrapper_getitem; } if (typeSlots & PythonQt::Type_MappingSetItem) { wrap->_base.as_mapping.mp_ass_subscript = (objobjargproc)PythonQtInstanceWrapper_setitem; } if (typeSlots & (PythonQt::Type_MappingGetItem | PythonQt::Type_MappingSetItem)) { if (typeSlots & PythonQt::Type_Length) { wrap->_base.as_mapping.mp_length = (lenfunc)PythonQtInstanceWrapper_length; } } if (typeSlots & PythonQt::Type_Add) { wrap->_base.as_number.nb_add = (binaryfunc)PythonQtInstanceWrapper_add; wrap->_base.as_number.nb_positive = (unaryfunc)PythonQtInstanceWrapper_positive; } if (typeSlots & PythonQt::Type_Subtract) { wrap->_base.as_number.nb_subtract = (binaryfunc)PythonQtInstanceWrapper_sub; wrap->_base.as_number.nb_negative = (unaryfunc)PythonQtInstanceWrapper_negative; } if (typeSlots & PythonQt::Type_Multiply) { wrap->_base.as_number.nb_multiply = (binaryfunc)PythonQtInstanceWrapper_mul; } if (typeSlots & PythonQt::Type_Divide) { wrap->_base.as_number.nb_true_divide = (binaryfunc)PythonQtInstanceWrapper_div; } if (typeSlots & PythonQt::Type_And) { wrap->_base.as_number.nb_and = (binaryfunc)PythonQtInstanceWrapper_and; } if (typeSlots & PythonQt::Type_Or) { wrap->_base.as_number.nb_or = (binaryfunc)PythonQtInstanceWrapper_or; } if (typeSlots & PythonQt::Type_Xor) { wrap->_base.as_number.nb_xor = (binaryfunc)PythonQtInstanceWrapper_xor; } if (typeSlots & PythonQt::Type_Mod) { wrap->_base.as_number.nb_remainder = (binaryfunc)PythonQtInstanceWrapper_mod; } if (typeSlots & PythonQt::Type_LShift) { wrap->_base.as_number.nb_lshift = (binaryfunc)PythonQtInstanceWrapper_lshift; } if (typeSlots & PythonQt::Type_RShift) { wrap->_base.as_number.nb_rshift = (binaryfunc)PythonQtInstanceWrapper_rshift; } if (typeSlots & PythonQt::Type_InplaceAdd) { wrap->_base.as_number.nb_inplace_add = (binaryfunc)PythonQtInstanceWrapper_iadd; } if (typeSlots & PythonQt::Type_InplaceSubtract) { wrap->_base.as_number.nb_inplace_subtract = (binaryfunc)PythonQtInstanceWrapper_isub; } if (typeSlots & PythonQt::Type_InplaceMultiply) { wrap->_base.as_number.nb_inplace_multiply = (binaryfunc)PythonQtInstanceWrapper_imul; } if (typeSlots & PythonQt::Type_InplaceDivide) { wrap->_base.as_number.nb_inplace_true_divide = (binaryfunc)PythonQtInstanceWrapper_idiv; } if (typeSlots & PythonQt::Type_InplaceAnd) { wrap->_base.as_number.nb_inplace_and = (binaryfunc)PythonQtInstanceWrapper_iand; } if (typeSlots & PythonQt::Type_InplaceOr) { wrap->_base.as_number.nb_inplace_or = (binaryfunc)PythonQtInstanceWrapper_ior; } if (typeSlots & PythonQt::Type_InplaceXor) { wrap->_base.as_number.nb_inplace_xor = (binaryfunc)PythonQtInstanceWrapper_ixor; } if (typeSlots & PythonQt::Type_InplaceMod) { wrap->_base.as_number.nb_inplace_remainder = (binaryfunc)PythonQtInstanceWrapper_imod; } if (typeSlots & PythonQt::Type_InplaceLShift) { wrap->_base.as_number.nb_inplace_lshift = (binaryfunc)PythonQtInstanceWrapper_ilshift; } if (typeSlots & PythonQt::Type_InplaceRShift) { wrap->_base.as_number.nb_inplace_rshift = (binaryfunc)PythonQtInstanceWrapper_irshift; } if (typeSlots & PythonQt::Type_Invert) { wrap->_base.as_number.nb_invert = (unaryfunc)PythonQtInstanceWrapper_invert; } if (typeSlots & PythonQt::Type_NonZero) { wrap->_base.as_number.nb_bool = (inquiry)PythonQtInstanceWrapper_nonzero; } } } static PyObject* PythonQtClassWrapper_alloc(PyTypeObject* self, Py_ssize_t nitems) { // call the default type alloc PyObject* obj = PyType_Type.tp_alloc(self, nitems); // take current class type, if we are called via newPythonQtClassWrapper() PythonQtClassWrapper* wrap = (PythonQtClassWrapper*)obj; wrap->_classInfo = PythonQt::priv()->currentClassInfoForClassWrapperCreation(); if (wrap->_classInfo) { initializeSlots(wrap); } return obj; } static int PythonQtClassWrapper_init(PythonQtClassWrapper* self, PyObject* args, PyObject* kwds) { // call the default type init if (PyType_Type.tp_init((PyObject*)self, args, kwds) < 0) { return -1; } self->_dynamicClassInfo = nullptr; // if we have no CPP class information, try our base class if (!self->classInfo()) { PyTypeObject* superType = ((PyTypeObject*)self)->tp_base; // recursively search for PythonQtClassWrapper superclass, // this is needed for multiple levels of inheritance in python, // e.g. // class MyWidgetBase(QWidget): // ... // class MyWidget(MyWidgetBase): // ... while (superType && Py_TYPE(superType) != &PythonQtClassWrapper_Type) { superType = superType->tp_base; } if (!superType || (Py_TYPE(superType) != &PythonQtClassWrapper_Type)) { PyErr_Format(PyExc_TypeError, "type %s is not derived from PythonQtClassWrapper", ((PyTypeObject*)self)->tp_name); return -1; } // take the class info from the superType self->_classInfo = ((PythonQtClassWrapper*)superType)->classInfo(); self->_dynamicClassInfo = new PythonQtDynamicClassInfo(); // take the class info from the superType and fill the whole chain PyTypeObject* typeChain = (PyTypeObject*)self; while (typeChain && Py_TYPE(typeChain) != &PythonQtClassWrapper_Type) { ((PythonQtClassWrapper*)typeChain)->_classInfo = ((PythonQtClassWrapper*)superType)->classInfo(); ((PythonQtClassWrapper*)typeChain)->_dynamicClassInfo = new PythonQtDynamicClassInfo(); typeChain = typeChain->tp_base; } } return 0; } static PyObject* PythonQtClassWrapper_classname(PythonQtClassWrapper* type) { return PyUnicode_FromString((QByteArray("Class_") + type->classInfo()->className()).constData()); } static PyObject* PythonQtClassWrapper_help(PythonQtClassWrapper* type) { return PythonQt::self()->helpCalled(type->classInfo()); } PyObject* PythonQtClassWrapper_delete(PythonQtClassWrapper* type, PyObject* args) { Q_UNUSED(type) Py_ssize_t argc = PyTuple_Size(args); if (argc > 0) { PyObject* self = PyTuple_GET_ITEM(args, 0); if (PyObject_TypeCheck(self, &PythonQtInstanceWrapper_Type)) { return PythonQtInstanceWrapper_delete((PythonQtInstanceWrapper*)self); } } return nullptr; } PyObject* PythonQtClassWrapper_inherits(PythonQtClassWrapper* type, PyObject* args) { Q_UNUSED(type) PythonQtInstanceWrapper* wrapper = nullptr; char* name = nullptr; if (!PyArg_ParseTuple(args, "O!s:PythonQtClassWrapper.inherits", &PythonQtInstanceWrapper_Type, &wrapper, &name)) { return nullptr; } return PythonQtConv::GetPyBool(wrapper->classInfo()->inherits(name)); } static PyMethodDef PythonQtClassWrapper_methods[] = { {"className", reinterpret_cast(reinterpret_cast(PythonQtClassWrapper_classname)), METH_NOARGS, "Return the classname of the object"}, {"inherits", reinterpret_cast(reinterpret_cast(PythonQtClassWrapper_inherits)), METH_VARARGS, "Returns if the class inherits or is of given type name"}, {"help", reinterpret_cast(reinterpret_cast(PythonQtClassWrapper_help)), METH_NOARGS, "Shows the help of available methods for this class"}, {"delete", reinterpret_cast(reinterpret_cast(PythonQtClassWrapper_delete)), METH_VARARGS, "Deletes the given C++ object"}, {nullptr, nullptr, 0, nullptr} /* Sentinel */ }; static PyObject* PythonQtClassWrapper_getDummyInstanceForProperty(PythonQtClassWrapper* wrapper, const QString& property) { PythonQtClassInfo* info = wrapper->classInfo()->getClassInfoForProperty(property); if (info) { return (PyObject*)PythonQt::priv()->createNewPythonQtInstanceWrapper(nullptr, info); } Py_RETURN_NONE; } static PyObject* PythonQtClassWrapper_getattro(PyObject* obj, PyObject* name) { const char* attributeName; PythonQtClassWrapper* wrapper = (PythonQtClassWrapper*)obj; if ((attributeName = PyUnicode_AsUTF8(name)) == nullptr) { return nullptr; } if (obj == (PyObject*)&PythonQtInstanceWrapper_Type) { // if we are called as PythonQtInstanceWrapper_Type, we need to get the properties from the type... PyObject* superAttr = PyType_Type.tp_getattro(obj, name); return superAttr; } if (qstrcmp(attributeName, "__dict__") == 0) { PyObject* objectDict = ((PyTypeObject*)wrapper)->tp_dict; if (!wrapper->classInfo()) { Py_INCREF(objectDict); return objectDict; } PyObject* dict = PyDict_New(); auto members = wrapper->classInfo()->memberList(); auto properties = wrapper->classInfo()->propertyList(); #if (QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)) QSet completeSet(members.begin(), members.end()); completeSet.unite(QSet(properties.begin(), properties.end())); #else QSet completeSet = QSet::fromList(members); completeSet.unite(QSet::fromList(properties)); #endif for (const QString& name : qAsConst(completeSet)) { if (name.startsWith("py_")) { // do not expose internal slots continue; } PyObject* o = PyObject_GetAttrString(obj, QStringToPythonConstCharPointer(name)); if (o) { PyDict_SetItemString(dict, QStringToPythonConstCharPointer(name), o); Py_DECREF(o); } else { // we should not get here anymore, but ignore this anyways. PyErr_Clear(); } } if (wrapper->classInfo()->constructors()) { PyObject* initName = PyUnicode_FromString("__init__"); PyObject* func = PyType_Type.tp_getattro(obj, initName); Py_DECREF(initName); PyDict_SetItemString(dict, "__init__", func); Py_DECREF(func); } for (int i = 0; PythonQtClassWrapper_methods[i].ml_name != nullptr; i++) { PyObject* func = PyCFunction_New(&PythonQtClassWrapper_methods[i], obj); PyDict_SetItemString(dict, PythonQtClassWrapper_methods[i].ml_name, func); Py_DECREF(func); } PyDict_Update(dict, objectDict); return dict; } // look in Python to support derived Python classes PyObject* superAttr = PyType_Type.tp_getattro(obj, name); if (superAttr) { return superAttr; } PyErr_Clear(); if ((qstrncmp("py_", attributeName, 3) != 0) && wrapper->classInfo()) { PythonQtMemberInfo member = wrapper->classInfo()->member(attributeName); if (member._type == PythonQtMemberInfo::EnumValue) { PyObject* enumValue = member._enumValue; Py_INCREF(enumValue); return enumValue; } else if (member._type == PythonQtMemberInfo::EnumWrapper || member._type == PythonQtMemberInfo::NestedClass) { PyObject* enumWrapper = member._pythonType; Py_INCREF(enumWrapper); return enumWrapper; } else if (member._type == PythonQtMemberInfo::Slot) { // NOTE: this does an extra lookup and string copy, it might be better to move this to // PythonQtClassInfo and add a new cache for qualified class methods... QByteArray qualifiedMemberName = "py_q_"; qualifiedMemberName += attributeName; PythonQtMemberInfo qualifiedMember = wrapper->classInfo()->member(qualifiedMemberName); if (qualifiedMember._type == PythonQtMemberInfo::Slot) { // return the qualified member, so that virtual calls on classes call the qualified member return PythonQtSlotFunction_New(qualifiedMember._slot, obj, nullptr); } else { // we return all slots, even the instance slots, since they are callable as unbound slots with self argument return PythonQtSlotFunction_New(member._slot, obj, nullptr); } } else if (member._type == PythonQtMemberInfo::Signal) { // we return all signals, even the instance signals, since they are callable as unbound signals with self argument return PythonQtSignalFunction_New(member._slot, obj, nullptr); } else if (member._type == PythonQtMemberInfo::Property) { PyObject* dummy = PythonQtClassWrapper_getDummyInstanceForProperty(wrapper, attributeName); return dummy; } QByteArray pyget = "py_get_"; member = wrapper->classInfo()->member((pyget + attributeName).constData()); if (member._type == PythonQtMemberInfo::Slot) { PyObject* dummy = PythonQtClassWrapper_getDummyInstanceForProperty(wrapper, attributeName); return dummy; } } // look for the internal methods (className(), help()) PyObject* internalMethod = PyObject_GenericGetAttr(obj, name); if (internalMethod) { return internalMethod; } QString error = QString(wrapper->classInfo()->className()) + " has no attribute named '" + QString(attributeName) + "'"; PyErr_SetString(PyExc_AttributeError, QStringToPythonConstCharPointer(error)); return nullptr; } static int PythonQtClassWrapper_setattro(PyObject* obj, PyObject* name, PyObject* value) { return PyType_Type.tp_setattro(obj, name, value); } /* static PyObject * PythonQtClassWrapper_repr(PyObject * obj) { PythonQtClassWrapper* wrapper = (PythonQtClassWrapper*)obj; if (wrapper->classInfo()->isCPPWrapper()) { const QMetaObject* meta = wrapper->classInfo()->metaObject(); if (!meta) { QObject* decorator = wrapper->classInfo()->decorator(); if (decorator) { meta = decorator->metaObject(); } } if (meta) { return PyUnicode_FromFormat("%s Class (C++ wrapped by %s)", wrapper->classInfo()->className(), meta->className()); } else { return PyUnicode_FromFormat("%s Class (C++ unwrapped)", wrapper->classInfo()->className()); } } else { return PyUnicode_FromFormat("%s Class", wrapper->classInfo()->className()); } } */ PyTypeObject PythonQtClassWrapper_Type = { PyVarObject_HEAD_INIT(nullptr, 0) /*tp_base*/ "PythonQt.PythonQtClassWrapper", /*tp_name*/ sizeof(PythonQtClassWrapper), /*tp_basicsize*/ 0, /*tp_itemsize*/ nullptr, /*tp_dealloc*/ 0, /*tp_vectorcall_offset*/ nullptr, /*tp_getattr*/ nullptr, /*tp_setattr*/ nullptr, /*tp_compare*/ nullptr, //PythonQtClassWrapper_repr, /*tp_repr*/ nullptr, /*tp_as_number*/ nullptr, /*tp_as_sequence*/ nullptr, /*tp_as_mapping*/ nullptr, /*tp_hash */ nullptr, /*tp_call*/ nullptr, /*tp_str*/ PythonQtClassWrapper_getattro, /*tp_getattro*/ PythonQtClassWrapper_setattro, /*tp_setattro*/ nullptr, /*tp_as_buffer*/ Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /*tp_flags*/ nullptr, /* tp_doc */ nullptr, /* tp_traverse */ nullptr, /* tp_clear */ nullptr, /* tp_richcompare */ 0, /* tp_weaklistoffset */ nullptr, /* tp_iter */ nullptr, /* tp_iternext */ PythonQtClassWrapper_methods, /* tp_methods */ nullptr, /* tp_members */ nullptr, /* tp_getset */ nullptr, /* tp_base */ nullptr, /* tp_dict */ nullptr, /* tp_descr_get */ nullptr, /* tp_descr_set */ 0, /* tp_dictoffset */ (initproc)PythonQtClassWrapper_init, /* tp_init */ PythonQtClassWrapper_alloc, /* tp_alloc */ nullptr, /* tp_new */ nullptr, /* tp_free */ }; //-------------------------------------------------------