/**************************************************************************** ** ** Copyright (C) 2014 Klaralvdalens Datakonsult AB (KDAB). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the Qt3D module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL3$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see http://www.qt.io/terms-conditions. For further ** information use the contact form at http://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPLv3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or later as published by the Free ** Software Foundation and appearing in the file LICENSE.GPL included in ** the packaging of this file. Please review the following information to ** ensure the GNU General Public License version 2.0 requirements will be ** met: http://www.gnu.org/licenses/gpl-2.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qnode.h" #include "qnode_p.h" #include #include #include #include #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE namespace Qt3DCore { QHash QNodePrivate::m_clonesLookupTable = QHash(); /*! \class Qt3DCore::QNodePrivate \internal */ QNodePrivate::QNodePrivate() : QObjectPrivate() , m_changeArbiter(Q_NULLPTR) , m_scene(Q_NULLPTR) , m_id(QNodeId::createId()) , m_blockNotifications(false) , m_wasCleanedUp(false) , m_enabled(true) , m_propertyChangesSetup(false) , m_signals(this) { } // Called by QNodePrivate::ctor or setParent (main thread) void QNodePrivate::_q_addChild(QNode *childNode) { Q_ASSERT(childNode); if (childNode == q_func()) return ; // If the scene is null it means that the current node is part of a subtree // that has been pre-prepared. Therefore the node shouldn't be added by // itself but only when the root of the said subtree is inserted into an // existing node whose m_scene member is valid if (m_scene == Q_NULLPTR) return; QNodeVisitor visitor; // Recursively set scene and change arbiter for the node subtree visitor.traverse(childNode, this, &QNodePrivate::setSceneHelper); // We notify only if we have a QChangeArbiter if (m_changeArbiter != Q_NULLPTR) { QScenePropertyChangePtr e(new QScenePropertyChange(NodeCreated, QSceneChange::Node, m_id)); e->setPropertyName("node"); // We need to clone the parent of the childNode we send QNode *parentClone = QNode::clone(q_func()); QNode *childClone = Q_NULLPTR; Q_FOREACH (QObject *c, parentClone->children()) { QNode *clone = qobject_cast(c); if (clone != Q_NULLPTR && clone->id() == childNode->id()) { childClone = clone; break; } } e->setValue(QVariant::fromValue(QNodePtr(childClone, &QNodePrivate::nodePtrDeleter))); notifyObservers(e); } // Handle Entity - Components visitor.traverse(childNode, this, &QNodePrivate::addEntityComponentToScene); } // Called by setParent or cleanup (main thread) (could be other thread if created on the backend in a job) void QNodePrivate::_q_removeChild(QNode *childNode) { Q_ASSERT(childNode); if (childNode->parent() != q_func()) qCWarning(Nodes) << Q_FUNC_INFO << "not a child of " << this; // Notify only if child isn't a clone if (m_changeArbiter != Q_NULLPTR) { QScenePropertyChangePtr e(new QScenePropertyChange(NodeAboutToBeDeleted, QSceneChange::Node, m_id)); e->setPropertyName("node"); // We need to clone the parent of the childNode we send // QNode *parentClone = QNode::clone(childNode->parentNode()); // QNode *childClone = Q_NULLPTR; // Q_FOREACH (QObject *c, parentClone->children()) { // QNode *clone = qobject_cast(c); // if (clone != Q_NULLPTR && clone->id() == childNode->id()) { // childClone = clone; // break; // } // } // We cannot clone the parent as it seems that the childNode is already removed // from the parent when the ChildRemoved event is triggered // and that would therefore return us a childNode NULL (because not found in the parent's children list) // and crash the backend QNode *childClone = QNode::clone(childNode); e->setValue(QVariant::fromValue(QNodePtr(childClone, &QNodePrivate::nodePtrDeleter))); notifyObservers(e); } // Recursively unset the scene on all children QNodeVisitor visitor; visitor.traverse(childNode, this, &QNodePrivate::unsetSceneHelper); } void QNodePrivate::registerNotifiedProperties() { Q_Q(QNode); if (m_propertyChangesSetup) return; const int offset = QNode::staticMetaObject.propertyOffset(); const int count = q->metaObject()->propertyCount(); for (int index = offset; index < count; index++) m_signals.connectToPropertyChange(q, index); m_propertyChangesSetup = true; } void QNodePrivate::unregisterNotifiedProperties() { Q_Q(QNode); if (!m_propertyChangesSetup) return; const int offset = QNode::staticMetaObject.propertyOffset(); const int count = q->metaObject()->propertyCount(); for (int index = offset; index < count; index++) m_signals.disconnectFromPropertyChange(q, index); m_propertyChangesSetup = false; } void QNodePrivate::propertyChanged(int propertyIndex) { // Bail out early if we can to avoid the cost below if (m_blockNotifications) return; Q_Q(QNode); const QMetaProperty property = q->metaObject()->property(propertyIndex); const QVariant data = property.read(q); if (data.canConvert()) { const QNode * const node = data.value(); const QNodeId id = node ? node->id() : QNodeId(); notifyPropertyChange(property.name(), QVariant::fromValue(id)); } else { notifyPropertyChange(property.name(), data); } } /*! \internal Recursively sets and adds the nodes in the subtree of base node \a root to the scene. Also takes care of connecting Components and Entities together in the scene. */ void QNodePrivate::setSceneHelper(QNode *root) { // Sets the scene root->d_func()->setScene(m_scene); // addObservable sets the QChangeArbiter m_scene->addObservable(root); // We also need to handle QEntity <-> QComponent relationships if (QComponent *c = qobject_cast(root)) { const QVector entities = c->entities(); Q_FOREACH (QEntity *entity, entities) { if (!m_scene->hasEntityForComponent(c->id(), entity->id())) { if (!c->shareable() && !m_scene->entitiesForComponent(c->id()).isEmpty()) qWarning() << "Trying to assign a non shareable component to more than one Entity"; m_scene->addEntityForComponent(c->id(), entity->id()); } } } } /*! \internal Recursively unsets and remove nodes in the subtree of base node \a root from the scene. Also takes care of removing Components and Entities connections. */ void QNodePrivate::unsetSceneHelper(QNode *root) { // We also need to handle QEntity <-> QComponent relationships removal if (QComponent *c = qobject_cast(root)) { const QVector entities = c->entities(); Q_FOREACH (QEntity *entity, entities) { if (m_scene) m_scene->removeEntityForComponent(c->id(), entity->id()); } } if (m_scene != Q_NULLPTR) m_scene->removeObservable(root); root->d_func()->setScene(Q_NULLPTR); } /*! \internal */ void QNodePrivate::addEntityComponentToScene(QNode *root) { if (QEntity *e = qobject_cast(root)) { Q_FOREACH (QComponent *c, e->components()) { if (!m_scene->hasEntityForComponent(c->id(), e->id())) m_scene->addEntityForComponent(c->id(), e->id()); } } } /*! \internal */ // Called in the main thread by QScene -> following QEvent::childAdded / addChild void QNodePrivate::setArbiter(QLockableObserverInterface *arbiter) { if (m_changeArbiter && m_changeArbiter != arbiter) unregisterNotifiedProperties(); m_changeArbiter = static_cast(arbiter); if (m_changeArbiter) registerNotifiedProperties(); } /*! Called when one or more backend aspects sends a notification \a change to the current Qt3DCore::QNode instance. \note This method should be reimplemented in your subclasses to properly handle the \a change. */ void QNode::sceneChangeEvent(const QSceneChangePtr &change) { Q_UNUSED(change); qWarning() << Q_FUNC_INFO << "sceneChangeEvent should have been subclassed"; } /*! \internal */ void QNodePrivate::setScene(QScene *scene) { if (m_scene != scene) m_scene = scene; } /*! \internal */ QScene *QNodePrivate::scene() const { return m_scene; } /*! \internal */ void QNodePrivate::notifyPropertyChange(const char *name, const QVariant &value) { // Bail out early if we can to avoid operator new if (m_blockNotifications) return; QScenePropertyChangePtr e(new QScenePropertyChange(NodeUpdated, QSceneChange::Node, m_id)); e->setPropertyName(name); e->setValue(value); notifyObservers(e); } /*! \internal */ // Called by the main thread void QNodePrivate::notifyObservers(const QSceneChangePtr &change) { Q_ASSERT(change); // Don't send notifications if we are blocking if (m_blockNotifications && change->type() == NodeUpdated) return; if (m_changeArbiter != Q_NULLPTR) { QAbstractPostman *postman = m_changeArbiter->postman(); if (postman != Q_NULLPTR) postman->notifyBackend(change); } } // Inserts this tree into the main Scene tree. // Needed when SceneLoaders provide a cloned tree from the backend // and need to insert it in the main scene tree // QNode *root; // QNode *subtree; // QNodePrivate::get(root)->insertTree(subtree); /*! \internal */ void QNodePrivate::insertTree(QNode *treeRoot, int depth) { if (m_scene != Q_NULLPTR) { treeRoot->d_func()->setScene(m_scene); m_scene->addObservable(treeRoot); } Q_FOREACH (QObject *c, treeRoot->children()) { QNode *n = Q_NULLPTR; if ((n = qobject_cast(c)) != Q_NULLPTR) insertTree(n, depth + 1); } if (depth == 0) treeRoot->setParent(q_func()); } /*! \internal */ QNodePrivate *QNodePrivate::get(QNode *q) { return q->d_func(); } /*! \internal */ void QNodePrivate::nodePtrDeleter(QNode *q) { QObject *p = q->parent(); if (p == Q_NULLPTR) p = q; p->deleteLater(); } /*! \class Qt3DCore::QNode \inherits QObject \inmodule Qt3DCore \since 5.5 \brief QNode is the base class of all Qt3D node classes used to build a Qt3D scene. The owernship of QNode is determined by the QObject parent/child relationship between nodes. By itself, a QNode has no visual appearance and no particular meaning, it is there as a way of building a node based tree structure. The parent of a QNode instance can only be another QNode instance. Each QNode instance has a unique id that allows it to be recognizable from other instances. When properties are defined on a QNode subclass, their NOTIFY signal will automatically generate notifications that the Qt3D backend aspects will receive. When subclassing QNode, make sure to call QNode::cleanup() from your subclass's destructor to ensure proper notification to backend aspects. Faiure to do so will result in crashes when one of your QNode subclass instance is eventually destroyed. \sa QEntity, QComponent */ /*! Creates a new QNode instance with parent \a parent. \note The backend aspects will be notified that a QNode instance is part of the scene only if it has a parent; unless this is the root node of the Qt3D scene. \sa setParent() */ QNode::QNode(QNode *parent) : QObject(*new QNodePrivate, parent) { qRegisterMetaType("QNode*"); // We need to add ourselves with the parent if it is valid // This will notify the backend about the new child if (parent) { // This needs to be invoked only after the QNode has been fully QMetaObject::invokeMethod(parent, "_q_addChild", Qt::QueuedConnection, Q_ARG(QNode*, this)); } } /*! \internal */ QNode::QNode(QNodePrivate &dd, QNode *parent) : QObject(dd, parent) { qRegisterMetaType("QNode*"); // We need to add ourselves with the parent if it is valid // This will notify the backend about the new child if (parent) { // This needs to be invoked only after the QNode has been fully QMetaObject::invokeMethod(parent, "_q_addChild", Qt::QueuedConnection, Q_ARG(QNode*, this)); } } /*! Copies all the attributes from \a ref to the current Qt3DCore::QNode instance. \note When subclassing QNode, you should reimplement this method and always call the copy method on the base class. This will ensure that when cloned, the QNode is properly initialized. */ void QNode::copy(const QNode *ref) { if (ref) { d_func()->m_id = ref->d_func()->m_id; d_func()->m_enabled = ref->d_func()->m_enabled; setObjectName(ref->objectName()); } } QNode::~QNode() { Q_ASSERT_X(QNodePrivate::get(this)->m_wasCleanedUp, Q_FUNC_INFO, "QNode::cleanup should have been called by now. A Qt3DCore::QNode subclass didn't call QNode::cleanup in its destructor"); } /*! Returns the id that uniquely identifies the QNode instance. */ const QNodeId QNode::id() const { Q_D(const QNode); return d->m_id; } /*! Returns the immediate QNode parent, or null if the node has no parent. */ QNode *QNode::parentNode() const { return qobject_cast(parent()); } /*! Returns \c true if aspect notifications are blocked; otherwise returns \c false. By default, notifications are \e not blocked. \sa blockNotifications() */ bool QNode::notificationsBlocked() const { Q_D(const QNode); return d->m_blockNotifications; } /*! If \a block is \c true, property change notifications sent by this object to aspects are blocked. If \a block is \c false, no such blocking will occur. The return value is the previous value of notificationsBlocked(). Note that the other notification types will be sent even if the notifications for this object have been blocked. \sa notificationsBlocked() */ bool QNode::blockNotifications(bool block) { Q_D(QNode); bool previous = d->m_blockNotifications; d->m_blockNotifications = block; return previous; } /*! * Sets the parent node of the current QNode instance to \a parent. * Setting the parent will notify the backend aspects about current QNode * instance's parent change. * * \note if \a parent happens to be null, this will actually notify that the * current QNode instance was removed from the scene. */ void QNode::setParent(QNode *parent) { if (parentNode()) QNodePrivate::get(parentNode())->_q_removeChild(this); QObject::setParent(parent); if (parentNode()) QNodePrivate::get(parentNode())->_q_addChild(this); emit parentChanged(parent); } /*! * Returns a list filled with the QNode children of the current * QNode instance. */ QNodeList QNode::childrenNodes() const { QNodeList nodeChildrenList; const QObjectList objectChildrenList = QObject::children(); nodeChildrenList.reserve(objectChildrenList.size()); Q_FOREACH (QObject *c, objectChildrenList) { if (QNode *n = qobject_cast(c)) nodeChildrenList.push_back(n); } return nodeChildrenList; } /*! Set the QNode to enabled if \a isEnabled is \c true. By default a QNode is always enabled. \note the interpretation of what enabled means is aspect-dependent. Even if enabled is set to \c false, some aspects may still consider the node in some manner. This is documented on a class by class basis. */ void QNode::setEnabled(bool isEnabled) { Q_D(QNode); if (d->m_enabled == isEnabled) return; d->m_enabled = isEnabled; emit enabledChanged(isEnabled); } /*! Returns whether the QNode is enabled or not. */ bool QNode::isEnabled() const { Q_D(const QNode); return d->m_enabled; } /*! Returns a clone of \a node. All the children of \a node are cloned as well. \note This is the only way to create two nodes with the same id. */ QNode *QNode::clone(QNode *node) { if (node == Q_NULLPTR) return Q_NULLPTR; static int clearLock = 0; clearLock++; // We keep a reference of clones for the current subtree // In order to preserve relationships when multiple entities // reference the same component QNode *clonedNode = QNodePrivate::m_clonesLookupTable.value(node->id()); if (clonedNode == Q_NULLPTR) { clonedNode = node->doClone(); // doClone, returns new instance with content copied // and relationships added QNodePrivate::m_clonesLookupTable.insert(clonedNode->id(), clonedNode); } Q_FOREACH (QObject *c, node->children()) { QNode *childNode = qobject_cast(c); if (childNode != Q_NULLPTR) { QNode *cclone = QNode::clone(childNode); // We use QObject::setParent instead of QNode::setParent to avoid the // whole overhead generated by the latter as we are only dealing with clones if (cclone != Q_NULLPTR) static_cast(cclone)->setParent(clonedNode); } } if (--clearLock == 0) // Cloning done QNodePrivate::m_clonesLookupTable.clear(); return clonedNode; } /*! * This methods can only be called once and takes care of notyfing the backend * aspects that the current Qt3DCore::QNode instance is about to be destroyed. * * \note It must be called by the destructor of every class subclassing * QNode that is clonable (using the QT3D_CLONEABLE macro). */ void QNode::cleanup() { Q_D(QNode); if (!d->m_wasCleanedUp) { d->m_wasCleanedUp = true; qCDebug(Nodes) << Q_FUNC_INFO << this; if (parentNode()) QNodePrivate::get(parentNode())->_q_removeChild(this); // Root element has no parent and therefore we cannot // call parent->_q_removeChild(); } } } // namespace Qt3DCore QT_END_NAMESPACE #include "moc_qnode.cpp"