Skip to content

Commit

Permalink
Lots of fixed. Draft
Browse files Browse the repository at this point in the history
  • Loading branch information
iakov committed Apr 3, 2021
1 parent f5c7cac commit ce7b3a2
Show file tree
Hide file tree
Showing 5 changed files with 122 additions and 102 deletions.
191 changes: 106 additions & 85 deletions src/PythonQt.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -412,6 +412,16 @@ PythonQt::PythonQt(int flags, const QByteArray& pythonQtModuleName)
PythonQt::~PythonQt() {
delete _p;
_p = NULL;

Py_DECREF(&PythonQtSlotFunction_Type);
Py_DECREF(&PythonQtSignalFunction_Type);
Py_DECREF(&PythonQtSlotDecorator_Type);
Py_DECREF(&PythonQtProperty_Type);
Py_DECREF(&PythonQtBoolResult_Type);
Py_DECREF(&PythonQtClassWrapper_Type);
Py_DECREF(&PythonQtInstanceWrapper_Type);
Py_DECREF(&PythonQtStdOutRedirectType);
Py_DECREF(&PythonQtStdInRedirectType);
}

PythonQtPrivate::~PythonQtPrivate() {
Expand Down Expand Up @@ -439,7 +449,9 @@ void PythonQt::setRedirectStdInCallback(PythonQtInputChangedCB* callback, void *

// Backup original 'sys.stdin' if not yet done
if( !PyObject_HasAttrString(sys.object(), "pythonqt_original_stdin") ) {
PyObject_SetAttrString(sys.object(), "pythonqt_original_stdin", PyObject_GetAttrString(sys.object(), "stdin"));
PyObject *_stdin = PyObject_GetAttrString(sys.object(), "stdin");
PyObject_SetAttrString(sys.object(), "pythonqt_original_stdin", _stdin);
Py_XDECREF(_stdin);
}

in = PythonQtStdInRedirectType.tp_new(&PythonQtStdInRedirectType, NULL, NULL);
Expand All @@ -456,15 +468,16 @@ void PythonQt::setRedirectStdInCallback(PythonQtInputChangedCB* callback, void *
void PythonQt::setRedirectStdInCallbackEnabled(bool enabled)
{
PythonQtObjectPtr sys;
PythonQtObjectPtr _stdin;
sys.setNewRef(PyImport_ImportModule("sys"));

if (enabled) {
if( !PyObject_HasAttrString(sys.object(), "pythonqt_stdin") ) {
PyObject_SetAttrString(sys.object(), "stdin", PyObject_GetAttrString(sys.object(), "pythonqt_stdin"));
if( PyObject_HasAttrString(sys.object(), "pythonqt_stdin") ) {
_stdin.setNewRef(PyObject_GetAttrString(sys.object(), "pythonqt_stdin"));
}
} else {
if( !PyObject_HasAttrString(sys.object(), "pythonqt_original_stdin") ) {
PyObject_SetAttrString(sys.object(), "stdin", PyObject_GetAttrString(sys.object(), "pythonqt_original_stdin"));
if( PyObject_HasAttrString(sys.object(), "pythonqt_original_stdin") ) {
_stdin.setNewRef(PyObject_GetAttrString(sys.object(), "pythonqt_original_stdin"));
}
}
}
Expand All @@ -478,7 +491,7 @@ void PythonQt::qObjectNoLongerWrappedCB(QObject* o)
{
if (_self->_p->_noLongerWrappedCB) {
(*_self->_p->_noLongerWrappedCB)(o);
};
}
}

void PythonQt::setQObjectMissingAttributeCallback(PythonQtQObjectMissingAttributeCB* cb)
Expand Down Expand Up @@ -557,20 +570,20 @@ void PythonQtPrivate::createPythonQtClassWrapper(PythonQtClassInfo* info, const
PythonQtClassInfo* outerClassInfo = lookupClassInfoAndCreateIfNotPresent(outerClass);
outerClassInfo->addNestedClass(info);
} else {
if (!PyModule_AddObject(pack, info->className(), pyobj)) {
// since PyModule_AddObject steals the reference, we need a incref once more...
Py_INCREF(pyobj);
Py_INCREF(pyobj);
if (PyModule_AddObject(pack, info->className(), pyobj)) {
Py_DECREF(pyobj);
}
}
if (!module && package && strncmp(package, "Qt", 2) == 0) {

Py_INCREF(pyobj);
// put all qt objects into Qt as well
if (!PyModule_AddObject(packageByName("Qt"), info->className(), pyobj)) {
// since PyModule_AddObject steals the reference, we need a incref once more...
Py_INCREF(pyobj);
if (PyModule_AddObject(packageByName("Qt"), info->className(), pyobj)) {
Py_DECREF(pyobj);
}
}
info->setPythonQtClassWrapper(pyobj);
Py_DECREF(pyobj);
}

PyObject* PythonQtPrivate::wrapQObject(QObject* obj)
Expand Down Expand Up @@ -931,17 +944,21 @@ QVariant PythonQt::evalCode(PyObject* object, PyObject* pycode) {
QVariant result;
clearError();
if (pycode) {
PyObject* dict = NULL;
PyObject* globals = NULL;
PythonQtObjectPtr dict;
PythonQtObjectPtr globals;
if (PyModule_Check(object)) {
dict = PyModule_GetDict(object);
globals = dict;
} else if (PyDict_Check(object)) {
dict = object;
globals = dict;
} else {
dict = PyObject_GetAttrString(object, "__dict__");
globals = PyObject_GetAttrString(PyImport_ImportModule(PyString_AS_STRING(PyObject_GetAttrString(object, "__module__"))),"__dict__");
PyObject *moduleName = PyObject_GetAttrString(object, "__module__");
PyObject *module = PyImport_ImportModule(PyString_AS_STRING(moduleName));
dict.setNewRef(PyObject_GetAttrString(object, "__dict__"));
globals.setNewRef(PyObject_GetAttrString(module, "__dict__"));
Py_XDECREF(moduleName);
Py_XDECREF(module);
}
PyObject* r = NULL;
if (dict) {
Expand Down Expand Up @@ -1086,30 +1103,30 @@ PythonQtObjectPtr PythonQt::createUniqueModule()

void PythonQt::addObject(PyObject* object, const QString& name, QObject* qObject)
{
if (PyModule_Check(object)) {
auto pyobj = _p->wrapQObject(qObject);
if (PyModule_AddObject(object, QStringToPythonCharPointer(name), pyobj) < 0) {
Py_DECREF(pyobj);
PyObject *wrappedObject = _p->wrapQObject(qObject);
if (PyModule_Check(object)) {
Py_XINCREF(wrappedObject);
PyModule_AddObject(object, QStringToPythonCharPointer(name), wrappedObject);
} else if (PyDict_Check(object)) {
PyDict_SetItemString(object, QStringToPythonCharPointer(name), wrappedObject);
} else {
PyObject_SetAttrString(object, QStringToPythonCharPointer(name), wrappedObject);
}
} else if (PyDict_Check(object)) {
PyDict_SetItemString(object, QStringToPythonCharPointer(name), _p->wrapQObject(qObject));
} else {
PyObject_SetAttrString(object, QStringToPythonCharPointer(name), _p->wrapQObject(qObject));
}
Py_XDECREF(wrappedObject);
}

void PythonQt::addVariable(PyObject* object, const QString& name, const QVariant& v)
{
if (PyModule_Check(object)) {
auto pyobj = PythonQtConv::QVariantToPyObject(v);
if (PyModule_AddObject(object, QStringToPythonCharPointer(name), pyobj) < 0) {
Py_DECREF(pyobj);
}
} else if (PyDict_Check(object)) {
PyDict_SetItemString(object, QStringToPythonCharPointer(name), PythonQtConv::QVariantToPyObject(v));
} else {
PyObject_SetAttrString(object, QStringToPythonCharPointer(name), PythonQtConv::QVariantToPyObject(v));
}
PyObject *value = PythonQtConv::QVariantToPyObject(v);
if (PyModule_Check(object)) {
Py_XINCREF(value);
PyModule_AddObject(object, QStringToPythonCharPointer(name), value);
} else if (PyDict_Check(object)) {
PyDict_SetItemString(object, QStringToPythonCharPointer(name), value);
} else {
PyObject_SetAttrString(object, QStringToPythonCharPointer(name), value);
}
Py_XDECREF(value);
}

void PythonQt::removeVariable(PyObject* object, const QString& name)
Expand Down Expand Up @@ -1151,7 +1168,7 @@ QStringList PythonQt::introspection(PyObject* module, const QString& objectname,
} else {
object = lookupObject(module, objectname);
if (!object && type == CallOverloads) {
PyObject* dict = lookupObject(module, "__builtins__");
PythonQtObjectPtr dict = lookupObject(module, "__builtins__");
if (dict) {
object = PyDict_GetItemString(dict, QStringToPythonCharPointer(objectname));
}
Expand Down Expand Up @@ -1203,38 +1220,36 @@ QStringList PythonQt::introspectObject(PyObject* object, ObjectType type)
}
}
} else {
PyObject* keys = NULL;
bool isDict = false;
if (PyDict_Check(object)) {
keys = PyDict_Keys(object);
isDict = true;
} else {
#if defined(MEVISLAB) && !defined(PY3K)
int oldPy3kWarningFlag = Py_Py3kWarningFlag;
Py_Py3kWarningFlag = 0; // temporarily disable Python 3 warnings
keys = PyObject_Dir(object);
Py_Py3kWarningFlag = oldPy3kWarningFlag;
#else
keys = PyObject_Dir(object);
#endif
PythonQtObjectPtr keys;
bool isDict = false;
if (PyDict_Check(object)) {
keys.setNewRef(PyDict_Keys(object));
isDict = true;
} else {
#if defined(MEVISLAB) && !defined(PY3K)
int oldPy3kWarningFlag = Py_Py3kWarningFlag;
Py_Py3kWarningFlag = 0; // temporarily disable Python 3 warnings
keys.setNewRef(PyObject_Dir(object));
Py_Py3kWarningFlag = oldPy3kWarningFlag;
#else
keys.setNewRef(PyObject_Dir(object));
#endif
}
if (keys) {
int count = PyList_Size(keys);
PyObject* key;
PyObject* value;
QString keystr;
for (int i = 0;i<count;i++) {
key = PyList_GetItem(keys,i);
if (isDict) {
value = PyDict_GetItem(object, key);
Py_INCREF(value);
} else {
value = PyObject_GetAttr(object, key);
}
if (!value) continue;
keystr = PyString_AsString(key);
static const QString underscoreStr("__tmp");
if (!keystr.startsWith(underscoreStr)) {
PythonQtObjectPtr key = PyList_GetItem(keys,i);
PythonQtObjectPtr value;
if (isDict) {
value.setNewRef(PyDict_GetItem(object, key));
Py_INCREF(value);
} else {
value = PyObject_GetAttr(object, key);
}
if (!value) continue;
QString keystr = PyString_AsString(key);
static const QString underscoreStr("__tmp");
if (!keystr.startsWith(underscoreStr)) {
switch (type) {
case Anything:
results << keystr;
Expand Down Expand Up @@ -1277,9 +1292,7 @@ QStringList PythonQt::introspectObject(PyObject* object, ObjectType type)
std::cerr << "PythonQt: introspection: unknown case" << ", in " << __FILE__ << ":" << __LINE__ << std::endl;
}
}
Py_DECREF(value);
}
Py_DECREF(keys);
}
}
PyErr_Clear();
Expand Down Expand Up @@ -1332,6 +1345,7 @@ QStringList PythonQt::introspectType(const QString& typeName, ObjectType type)
PyObject* typeObject = getObjectByType(typeName);
if (typeObject) {
object = PyObject_GetAttrString(typeObject, QStringToPythonCharPointer(memberName));
Py_DECREF(typeObject);
}
}
if (object) {
Expand Down Expand Up @@ -1403,6 +1417,7 @@ PyObject* PythonQt::callAndReturnPyObject(PyObject* callable, const QVariantList
PyObject* arg = PythonQtConv::QVariantToPyObject(it.value());
if (arg) {
PyDict_SetItemString(pkwargs, QStringToPythonCharPointer(it.key()), arg);
Py_DECREF(arg);
} else {
err = true;
break;
Expand Down Expand Up @@ -1794,7 +1809,7 @@ void PythonQt::initPythonQtModule(bool redirectStdOut, const QByteArray& pythonQ
}
#ifdef PY3K
PythonQtModuleDef.m_name = name.constData();
_p->_pythonQtModule = PyModule_Create(&PythonQtModuleDef);
_p->_pythonQtModule.setNewRef(PyModule_Create(&PythonQtModuleDef));
#else
_p->_pythonQtModule = Py_InitModule(name.constData(), PythonQtMethods);
#endif
Expand Down Expand Up @@ -1834,7 +1849,11 @@ void PythonQt::initPythonQtModule(bool redirectStdOut, const QByteArray& pythonQ
Py_XDECREF(old_module_names);

#ifdef PY3K
PyDict_SetItem(PyObject_GetAttrString(sys.object(), "modules"), PyUnicode_FromString(name.constData()), _p->_pythonQtModule.object());
PyObject *modules = PyObject_GetAttrString(sys.object(), "modules");
PyObject *nameObj = PyUnicode_FromString(name.constData());
PyDict_SetItem(modules, nameObj, _p->_pythonQtModule.object());
Py_XDECREF(modules);
Py_XDECREF(nameObj);
#endif
}

Expand All @@ -1854,8 +1873,9 @@ QString PythonQt::getReturnTypeOfWrappedMethod(PyObject* module, const QString&

QString PythonQt::getReturnTypeOfWrappedMethod(const QString& typeName, const QString& methodName)
{
PythonQtObjectPtr typeObject = getObjectByType(typeName);
if (typeObject.isNull()) {
PythonQtObjectPtr typeObject;
typeObject.setNewRef(getObjectByType(typeName));
if (typeObject.isNull()) {
return "";
}
return getReturnTypeOfWrappedMethodHelper(typeObject, methodName, typeName + "." + methodName);
Expand Down Expand Up @@ -2125,10 +2145,12 @@ const QMetaObject* PythonQtPrivate::buildDynamicMetaObject(PythonQtClassWrapper*
builder.setClassName(((PyTypeObject*)type)->tp_name);

PyObject* dict = ((PyTypeObject*)type)->tp_dict;
Py_ssize_t pos = NULL;
Py_ssize_t pos = 0;
PyObject* value = NULL;
PyObject* key = NULL;
static PyObject* qtSlots = PyString_FromString("_qtSlots");
static PyObject* qtSlots = NULL;
if (!qtSlots)
qtSlots = PyString_FromString("_qtSlots");

bool needsMetaObject = false;
// Iterate over all members and check if they affect the QMetaObject:
Expand All @@ -2147,7 +2169,7 @@ const QMetaObject* PythonQtPrivate::buildDynamicMetaObject(PythonQtClassWrapper*
}
}
}
pos = NULL;
pos = 0;
value = NULL;
key = NULL;
// Now look for slots: (this is a bug in QMetaObjectBuilder, all signals need to be added first)
Expand Down Expand Up @@ -2181,10 +2203,11 @@ const QMetaObject* PythonQtPrivate::buildDynamicMetaObject(PythonQtClassWrapper*
}
if (PyFunction_Check(value) && PyObject_HasAttr(value, qtSlots)) {
// A function which has a "_qtSlots" signature list, add the slots to the meta object
PyObject* signatures = PyObject_GetAttr(value, qtSlots);
PythonQtObjectPtr signatures;
signatures.setNewRef(PyObject_GetAttr(value, qtSlots));
Py_ssize_t count = PyList_Size(signatures);
for (Py_ssize_t i = 0; i < count; i++) {
PyObject* signature = PyList_GET_ITEM(signatures, i);
PyObject* signature = PyList_GET_ITEM(signatures.object(), i);
QByteArray sig = PyString_AsString(signature);
// Split the return type and the rest of the signature,
// no spaces should be in the rest of the signature...
Expand Down Expand Up @@ -2230,9 +2253,11 @@ int PythonQtPrivate::handleMetaCall(QObject* object, PythonQtInstanceWrapper* wr
}
PythonQtProperty* prop = NULL;
// Get directly from the Python class, since we don't want to get the value of the property
PyObject* maybeProp = PyBaseObject_Type.tp_getattro((PyObject*)wrapper, PyString_FromString(metaProp.name()));
PythonQtObjectPtr name, maybeProp;
name.setNewRef(PyString_FromString(metaProp.name()));
maybeProp.setNewRef(PyBaseObject_Type.tp_getattro((PyObject*)wrapper, name));
if (maybeProp && PythonQtProperty_Check(maybeProp)) {
prop = (PythonQtProperty*)maybeProp;
prop = (PythonQtProperty*)maybeProp.object();
} else {
return id - methodCount;
}
Expand Down Expand Up @@ -2288,17 +2313,15 @@ QString PythonQtPrivate::getSignature(PyObject* object)
PyMethodObject* method = NULL;
PyFunctionObject* func = NULL;

bool decrefMethod = false;

if (PythonQtUtils::isPythonClassType(object)) {
method = (PyMethodObject*)PyObject_GetAttrString(object, "__init__");
decrefMethod = true;
} else if (object->ob_type == &PyFunction_Type) {
func = (PyFunctionObject*)object;
} else if (object->ob_type == &PyMethod_Type) {
method = (PyMethodObject*)object;
Py_XINCREF(method);
}
if (method) {
if (method) {
if (PyFunction_Check(method->im_func)) {
func = (PyFunctionObject*)method->im_func;
} else if (isMethodDescriptor((PyObject*)method)) {
Expand Down Expand Up @@ -2405,9 +2428,7 @@ QString PythonQtPrivate::getSignature(PyObject* object)
signature = funcName + "(" + signature + ")";
}

if (method && decrefMethod) {
Py_DECREF(method);
}
Py_XDECREF(method);
}

return signature;
Expand Down
Loading

0 comments on commit ce7b3a2

Please sign in to comment.