#ifndef _PYTHONQTCLASSINFO_H #define _PYTHONQTCLASSINFO_H /* * * 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 * */ #include "PythonQtPythonInclude.h" #include "PythonQt.h" #include #include #include #include #include class PythonQtSlotInfo; class PythonQtClassInfo; struct PythonQtDynamicClassInfo { PythonQtDynamicClassInfo() { _dynamicMetaObject = nullptr; _classInfo = nullptr; } ~PythonQtDynamicClassInfo(); const QMetaObject* _dynamicMetaObject; PythonQtClassInfo* _classInfo; }; struct PythonQtMemberInfo { enum Type { Invalid, Slot, Signal, EnumValue, EnumWrapper, Property, NestedClass, NotFound }; PythonQtMemberInfo():_type(Invalid),_slot(nullptr),_pythonType(nullptr),_enumValue(nullptr) { } PythonQtMemberInfo(PythonQtSlotInfo* info); PythonQtMemberInfo(const PythonQtObjectPtr& enumValue); PythonQtMemberInfo(const QMetaProperty& prop); Type _type; // TODO: this could be a union... PythonQtSlotInfo* _slot; PyObject* _pythonType; PythonQtObjectPtr _enumValue; QMetaProperty _property; }; //! a class that stores all required information about a Qt object (and an optional associated C++ class name) /*! for fast lookup of slots when calling the object from Python */ class PYTHONQT_EXPORT PythonQtClassInfo { public: PythonQtClassInfo(); ~PythonQtClassInfo(); //! store information about parent classes struct ParentClassInfo { ParentClassInfo(PythonQtClassInfo* parent, int upcastingOffset=0):_parent(parent),_upcastingOffset(upcastingOffset) {}; PythonQtClassInfo* _parent; int _upcastingOffset; }; //! setup as a QObject, taking the meta object as meta information about the QObject void setupQObject(const QMetaObject* meta); //! setup as a CPP (non-QObject), taking the classname void setupCPPObject(const QByteArray& classname); //! set the type capabilities void setTypeSlots(int typeSlots) { _typeSlots = typeSlots; } //! get the type capabilities int typeSlots() const { return _typeSlots; } //! get the Python method definition for a given slot name (without return type and signature) PythonQtMemberInfo member(const char* member); //! get access to the constructor slot (which may be overloaded if there are multiple constructors) PythonQtSlotInfo* constructors(); //! get access to the destructor slot PythonQtSlotInfo* destructor(); //! add a constructor, ownership is passed to classinfo void addConstructor(PythonQtSlotInfo* info); //! set a destructor, ownership is passed to classinfo void setDestructor(PythonQtSlotInfo* info); //! add a decorator slot, ownership is passed to classinfo void addDecoratorSlot(PythonQtSlotInfo* info); //! add a nested class, so that it can be shown as outer class member void addNestedClass(PythonQtClassInfo* info); //! get the classname (either of the QObject or of the wrapped CPP object) const QByteArray& className() const; //! get the unscoped classname (without ParentClass::) for nested classes QByteArray unscopedClassName() const; //! returns if the QObject bool isQObject() { return _isQObject; } //! returns if the class is a CPP wrapper bool isCPPWrapper() { return !_isQObject; } //! get the meta object const QMetaObject* metaObject() { return _meta; } //! set the meta object, this will reset the caching void setMetaObject(const QMetaObject* meta); //! returns if this class inherits from the given classname bool inherits(const char* classname); //! returns if this class inherits from the given classinfo bool inherits(PythonQtClassInfo* info); //! casts the given \c ptr to an object of type \c classname, returns the new pointer //! which might be different to \c ptr due to C++ multiple inheritance //! (if the cast is not possible or if ptr is NULL, NULL is returned) void* castTo(void* ptr, const char* classname); //! get help string for the metaobject QString help(); //! get list of all properties (on QObjects only, otherwise the list is empty) QStringList propertyList(); //! get list of all members (excluding properties, which can be listed with propertyList()) QStringList memberList(); //! get the meta type id of this class (only valid for isCPPWrapper() == true) int metaTypeId() { return _metaTypeId; } //! set an additional decorator provider that offers additional decorator slots for this class void setDecoratorProvider(PythonQtQObjectCreatorFunctionCB* cb); //! get the decorator qobject instance QObject* decorator(); //! add the parent class info of a CPP object void addParentClass(const ParentClassInfo& info) { _parentClasses.append(info); } //! set the associated PythonQtClassWrapper (which handles instance creation of this type) void setPythonQtClassWrapper(PyObject* obj) { _pythonQtClassWrapper = obj; } //! get the associated PythonQtClassWrapper (which handles instance creation of this type) PyObject* pythonQtClassWrapper() { return _pythonQtClassWrapper; } //! set the shell set instance wrapper cb void setShellSetInstanceWrapperCB(PythonQtShellSetInstanceWrapperCB* cb) { _shellSetInstanceWrapperCB = cb; } //! get the shell set instance wrapper cb PythonQtShellSetInstanceWrapperCB* shellSetInstanceWrapperCB() { return _shellSetInstanceWrapperCB; } //! add a handler for polymorphic downcasting void addPolymorphicHandler(PythonQtPolymorphicHandlerCB* cb) { _polymorphicHandlers.append(cb); } //! cast the pointer down in the class hierarchy if a polymorphic handler allows to do that void* castDownIfPossible(void* ptr, PythonQtClassInfo** resultClassInfo); //! returns if the localScope has an enum of that type name or if the enum contains a :: scope, if that class contails the enum static PyObject* findEnumWrapper(const QByteArray& name, PythonQtClassInfo* localScope, bool* isLocalEnum = nullptr); //! clear all members that where cached as "NotFound" void clearNotFoundCachedMembers(); //! get nested classes const QList& nestedClasses() { return _nestedClasses; } //! Create a copy of the given C++ object (which is known to be of a derived class), wrapped by Python and owned by PythonQt. //! This will downcast if possible and return a copy of the down casted object. //! This either requires a copy constructor on the class or it needs to be registered //! as a meta type. PyObject* copyObject(void* cppObject); //! Get the copy constructor for this class PythonQtSlotInfo* getCopyConstructor(); //! Sets reference counting callbacks for this class and all its subclasses void setReferenceCounting(PythonQtVoidPtrCB* refCB, PythonQtVoidPtrCB* unrefCB); //! Returns the ref counting CB, if there is any PythonQtVoidPtrCB* referenceCountingRefCB(); //! Returns the unref counting CB, if there is any PythonQtVoidPtrCB* referenceCountingUnrefCB(); //! Returns the Python type object for a given property. //! (the returned object does not get an extra reference count) PyObject* getPythonTypeForProperty(const QString& name); //! Returns the class info for given property, if available. PythonQtClassInfo* getClassInfoForProperty( const QString& name ); //! Returns if the class supports rich compare. This tests for //! __eq__, __ne__, __lt__, __le__, __gt__, __ge__ slots and if //! any of the slots is present it returns true and modifies the //! _typeSlots with Type_RichCompare. The result is cached internally. bool supportsRichCompare(); //! Sometimes enum values use a reserved name in Python. In this case //! replace it with something that is not reserved static QByteArray escapeReservedNames(const QByteArray& name); //! Add a wrapper that contains global enums static void addGlobalNamespaceWrapper(PythonQtClassInfo* namespaceWrapper); private: void updateRefCountingCBs(); void createEnumWrappers(const QObject* decoratorProvider); void createEnumWrappers(const QMetaObject* meta); PyObject* findEnumWrapper(const char* name); //! clear all cached members void clearCachedMembers(); void* recursiveCastDownIfPossible(void* ptr, const char** resultClassName); PythonQtSlotInfo* findDecoratorSlotsFromDecoratorProvider(const char* memberName, PythonQtSlotInfo* inputInfo, bool &found, QHash& memberCache, int upcastingOffset); void listDecoratorSlotsFromDecoratorProvider(QStringList& list, bool metaOnly); PythonQtSlotInfo* recursiveFindDecoratorSlotsFromDecoratorProvider(const char* memberName, PythonQtSlotInfo* inputInfo, bool &found, QHash& memberCache, int upcastingOffset); void recursiveCollectClassInfos(QList& classInfoObjects); void recursiveCollectDecoratorObjects(QList& decoratorObjects); bool lookForPropertyAndCache(const char* memberName); bool lookForMethodAndCache(const char* memberName); bool lookForEnumAndCache(const QMetaObject* m, const char* memberName); PythonQtSlotInfo* findDecoratorSlots(const char* memberName, PythonQtSlotInfo* tail, bool &found, QHash& memberCache, int upcastingOffset); int findCharOffset(const char* sigStart, char someChar); QHash _cachedMembers; PythonQtSlotInfo* _constructors; PythonQtSlotInfo* _destructor; PythonQtVoidPtrCB* _refCallback; PythonQtVoidPtrCB* _unrefCallback; QList _decoratorSlots; QList _enumWrappers; const QMetaObject* _meta; QByteArray _wrappedClassName; QList _parentClasses; QList _polymorphicHandlers; QList _nestedClasses; QObject* _decoratorProvider; PythonQtQObjectCreatorFunctionCB* _decoratorProviderCB; PyObject* _pythonQtClassWrapper; PythonQtShellSetInstanceWrapperCB* _shellSetInstanceWrapperCB; int _metaTypeId; int _typeSlots; bool _isQObject; bool _enumsCreated; bool _richCompareDetectionDone; bool _searchPolymorphicHandlerOnParent; bool _searchRefCountCB; static QList _globalNamespaceWrappers; static QSet _reservedNames; }; //--------------------------------------------------------------- #endif