/*========================================================================= Library: CTK Copyright (c) Kitware Inc. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.commontk.org/LICENSE Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. =========================================================================*/ // Qt includes #include #include #include #include #include // CTK includes #include "ctkVTKObjectEventsObserver.h" #include "ctkVTKConnection.h" // VTK includes #include #include //----------------------------------------------------------------------------- class ctkVTKObjectEventsObserverPrivate { Q_DECLARE_PUBLIC(ctkVTKObjectEventsObserver); protected: ctkVTKObjectEventsObserver* const q_ptr; public: ctkVTKObjectEventsObserverPrivate(ctkVTKObjectEventsObserver& object); /// /// Check if a connection has already been added bool containsConnection(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot); /// /// Return a reference toward the corresponding connection or 0 if doesn't exist ctkVTKConnection* findConnection(const QString& id); /// /// Return a reference toward the corresponding connection or 0 if doesn't exist ctkVTKConnection* findConnection(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot); /// /// Return all the references that match the given parameters QList findConnections(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot); inline QList connections()const { Q_Q(const ctkVTKObjectEventsObserver); return q->findChildren(); } bool AllBlocked; bool ObserveDeletion; }; //----------------------------------------------------------------------------- // ctkVTKObjectEventsObserver methods //----------------------------------------------------------------------------- ctkVTKObjectEventsObserver::ctkVTKObjectEventsObserver(QObject* _parent):Superclass(_parent) , d_ptr(new ctkVTKObjectEventsObserverPrivate(*this)) { this->setProperty("QVTK_OBJECT", true); } //----------------------------------------------------------------------------- ctkVTKObjectEventsObserver::~ctkVTKObjectEventsObserver() { } //----------------------------------------------------------------------------- void ctkVTKObjectEventsObserver::printAdditionalInfo() { this->Superclass::dumpObjectInfo(); Q_D(ctkVTKObjectEventsObserver); qDebug() << "ctkVTKObjectEventsObserver:" << this << endl << " AllBlocked:" << d->AllBlocked << endl << " Parent:" << (this->parent()?this->parent()->objectName():"NULL") << endl << " Connection count:" << d->connections().count(); // Loop through all connection foreach (const ctkVTKConnection* connection, d->connections()) { qDebug() << *connection; } } //----------------------------------------------------------------------------- QString ctkVTKObjectEventsObserver::addConnection(vtkObject* old_vtk_obj, vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot, float priority) { QString connectionId; if (old_vtk_obj) { // Check that old_object and new_object are the same type if (vtk_obj && !vtk_obj->IsA(old_vtk_obj->GetClassName())) { qDebug() << "Previous vtkObject (type:" << old_vtk_obj->GetClassName() << ") to disconnect" << "and new vtkObject (type:" << vtk_obj->GetClassName() << ") to connect" << "have a different type."; return connectionId; } // Disconnect old vtkObject this->removeConnection(old_vtk_obj, vtk_event, qt_obj, qt_slot); } if (vtk_obj) { connectionId = this->addConnection(vtk_obj, vtk_event, qt_obj, qt_slot, priority); } return connectionId; } //----------------------------------------------------------------------------- QString ctkVTKObjectEventsObserver::reconnection(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot, float priority) { QString connectionId; this->removeConnection(0, vtk_event, qt_obj, qt_slot); if (vtk_obj) { connectionId = this->addConnection(vtk_obj, vtk_event, qt_obj, qt_slot, priority); } return connectionId; } //----------------------------------------------------------------------------- QString ctkVTKObjectEventsObserver::addConnection(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot, float priority) { Q_D(ctkVTKObjectEventsObserver); if (!ctkVTKConnection::isValid(vtk_obj, vtk_event, qt_obj, qt_slot)) { qDebug() << "ctkVTKObjectEventsObserver::addConnection(...) - Invalid parameters - " << ctkVTKConnection::shortDescription(vtk_obj, vtk_event, qt_obj, qt_slot); return QString(); } // Check if such event is already observed if (d->containsConnection(vtk_obj, vtk_event, qt_obj, qt_slot)) { qWarning() << "ctkVTKObjectEventsObserver::addConnection - [vtkObject:" << vtk_obj->GetClassName() << ", event:" << vtk_event << "]" << " is already connected with [qObject:" << qt_obj->objectName() << ", slot:" << qt_slot << "]"; return QString(); } // Instantiate a new connection, set its parameters and add it to the list ctkVTKConnection * connection = new ctkVTKConnection(this); connection->observeDeletion(d->ObserveDeletion); connection->setup(vtk_obj, vtk_event, qt_obj, qt_slot, priority); // If required, establish connection connection->setBlocked(d->AllBlocked); return connection->id(); } //----------------------------------------------------------------------------- bool ctkVTKObjectEventsObserver::blockAllConnections(bool block) { Q_D(ctkVTKObjectEventsObserver); if (d->AllBlocked == block) { return d->AllBlocked; } bool oldAllBlocked = d->AllBlocked; foreach (ctkVTKConnection* connection, d->connections()) { connection->setBlocked(block); } d->AllBlocked = block; return oldAllBlocked; } //----------------------------------------------------------------------------- bool ctkVTKObjectEventsObserver::connectionsBlocked()const { Q_D(const ctkVTKObjectEventsObserver); return d->AllBlocked; } //----------------------------------------------------------------------------- bool ctkVTKObjectEventsObserver::blockConnection(const QString& id, bool blocked) { Q_D(ctkVTKObjectEventsObserver); ctkVTKConnection* connection = d->findConnection(id); if (connection == 0) { qWarning() << "no connection for id " << id; return false; } bool oldBlocked = connection->isBlocked(); connection->setBlocked(blocked); return oldBlocked; } //----------------------------------------------------------------------------- int ctkVTKObjectEventsObserver::blockConnection(bool block, vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj) { Q_D(ctkVTKObjectEventsObserver); if (!vtk_obj) { qDebug() << "ctkVTKObjectEventsObserver::blockConnectionRecursive" << "- Failed to " << (block?"block":"unblock") <<" connection" << "- vtkObject is NULL"; return 0; } QList connections = d->findConnections(vtk_obj, vtk_event, qt_obj, 0); foreach (ctkVTKConnection* connection, connections) { connection->setBlocked(block); } return connections.size(); } //----------------------------------------------------------------------------- int ctkVTKObjectEventsObserver::removeConnection(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot) { Q_D(ctkVTKObjectEventsObserver); QList connections = d->findConnections(vtk_obj, vtk_event, qt_obj, qt_slot); foreach (ctkVTKConnection* connection, connections) { delete connection; } return connections.count(); } //----------------------------------------------------------------------------- // ctkVTKObjectEventsObserverPrivate methods //----------------------------------------------------------------------------- ctkVTKObjectEventsObserverPrivate::ctkVTKObjectEventsObserverPrivate(ctkVTKObjectEventsObserver& object) :q_ptr(&object) { this->AllBlocked = false; // ObserveDeletion == false hasn't been that well tested... this->ObserveDeletion = true; } //----------------------------------------------------------------------------- bool ctkVTKObjectEventsObserverPrivate::containsConnection(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot) { return (this->findConnection(vtk_obj, vtk_event, qt_obj, qt_slot) != 0); } //----------------------------------------------------------------------------- ctkVTKConnection* ctkVTKObjectEventsObserverPrivate::findConnection(const QString& id) { foreach(ctkVTKConnection* connection, this->connections()) { if (connection->id() == id) { return connection; } } return 0; } //----------------------------------------------------------------------------- ctkVTKConnection* ctkVTKObjectEventsObserverPrivate::findConnection(vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot) { QList foundConnections = this->findConnections(vtk_obj, vtk_event, qt_obj, qt_slot); return foundConnections.size() ? foundConnections[0] : 0; } //----------------------------------------------------------------------------- QList ctkVTKObjectEventsObserverPrivate::findConnections( vtkObject* vtk_obj, unsigned long vtk_event, const QObject* qt_obj, const char* qt_slot) { bool all_info = true; if(vtk_obj == NULL || qt_slot == NULL || qt_obj == NULL || vtk_event == vtkCommand::NoEvent) { all_info = false; } QList foundConnections; // Loop through all connection foreach (ctkVTKConnection* connection, this->connections()) { if (connection->isEqual(vtk_obj, vtk_event, qt_obj, qt_slot)) { foundConnections.append(connection); if (all_info) { break; } } } return foundConnections; }