| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487 | /*=========================================================================  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.apache.org/licenses/LICENSE-2.0.txt  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 <QDebug>#include <QPointer>#include <QRegExp>#include <QString>#include <QTextStream>// CTK includes#include "ctkVTKConnection.h"#include "ctkVTKConnection_p.h"// VTK includes#include <vtkObject.h>#include <vtkSmartPointer.h>#include <vtkCallbackCommand.h>//-----------------------------------------------------------------------------QString convertPointerToString(void* pointer){  QString pointerAsString;  QTextStream(&pointerAsString) << pointer;  return pointerAsString;}//-----------------------------------------------------------------------------// ctkVTKConnectionPrivate methods//-----------------------------------------------------------------------------ctkVTKConnectionPrivate::ctkVTKConnectionPrivate(ctkVTKConnection& object)  :q_ptr(&object){  this->Callback    = vtkSmartPointer<vtkCallbackCommand>::New();  this->Callback->SetCallback(ctkVTKConnectionPrivate::DoCallback);  this->Callback->SetClientData(this);  this->QtObject    = 0;  this->VTKEvent    = vtkCommand::NoEvent;  this->Priority    = 0.0;  this->ConnectionType = Qt::AutoConnection;  this->SlotType    = ARG_UNKNOWN;  this->Connected   = false;  this->Blocked     = false;  this->Id          = convertPointerToString(this);  this->ObserveDeletion = false;}//-----------------------------------------------------------------------------ctkVTKConnectionPrivate::~ctkVTKConnectionPrivate(){}//-----------------------------------------------------------------------------void ctkVTKConnectionPrivate::connect(){  Q_Q(ctkVTKConnection);    if (this->Connected)    {    qDebug() << "ctkVTKConnection already connected.";    return;    }  switch (this->SlotType)    {    case ctkVTKConnectionPrivate::ARG_VTKOBJECT_AND_VTKOBJECT:      QObject::connect(q, SIGNAL(emitExecute(vtkObject*,vtkObject*)),        this->QtObject, this->QtSlot.toLatin1(), this->ConnectionType);      break;    case ctkVTKConnectionPrivate::ARG_VTKOBJECT_VOID_ULONG_VOID:      QObject::connect(q, SIGNAL(emitExecute(vtkObject*,void*,ulong,void*)),                       this->QtObject, this->QtSlot.toLatin1(), this->ConnectionType);      break;    default:      Q_ASSERT(false);      qCritical() << "Failed to connect - "                  << "The slot (" << this->QtSlot <<  ") owned by "                  << "QObject(" << this->QtObject->objectName() << ")"                  << " seems to have a wrong signature.";      break;    }  // Make a connection between this and the vtk object  q->addObserver(this->VTKObject, this->VTKEvent, this->Callback, this->Priority);  // If necessary, observe vtk DeleteEvent  if(this->ObserveDeletion)    {    // don't observe it twice    if (this->VTKEvent != vtkCommand::DeleteEvent)      {      this->VTKObject->AddObserver(vtkCommand::DeleteEvent, this->Callback);      }    }  // When Qt object is destroyed: (1) remove VTK observers and  // (2) set QtObject pointer to 0.  QObject::connect(this->QtObject, SIGNAL(destroyed(QObject*)),                   q, SLOT(qobjectDeleted()));  this->Connected = true;}//-----------------------------------------------------------------------------void ctkVTKConnectionPrivate::disconnectSlots(){  Q_Q(ctkVTKConnection);    if (!this->Connected)     {     return;     }  if (this->QtObject)    {    switch (this->SlotType)      {      case ctkVTKConnectionPrivate::ARG_VTKOBJECT_AND_VTKOBJECT:        QObject::disconnect(q, SIGNAL(emitExecute(vtkObject*,vtkObject*)),                            this->QtObject,this->QtSlot.toLatin1().data());        break;      case ctkVTKConnectionPrivate::ARG_VTKOBJECT_VOID_ULONG_VOID:        QObject::disconnect(q, SIGNAL(emitExecute(vtkObject*,void*,ulong,void*)),                            this->QtObject, this->QtSlot.toLatin1().data());        break;      default:        Q_ASSERT(false);        qCritical() << "Failed to disconnect - "                    << "The slot (" << this->QtSlot <<  ") owned by "                    << "QObject(" << this->QtObject->objectName() << ")"                    << " seems to have a wrong signature.";        break;      }    }  this->Connected = false;}//-----------------------------------------------------------------------------void ctkVTKConnectionPrivate::disconnectVTKObject(){  Q_Q(ctkVTKConnection);  if (this->VTKObject)    {    q->removeObserver(this->VTKObject, this->VTKEvent, this->Callback);    if (this->ObserveDeletion)      {      this->VTKObject->RemoveObservers(vtkCommand::DeleteEvent, this->Callback);      }    }}//-----------------------------------------------------------------------------bool ctkVTKConnectionPrivate::IsSameQtSlot(const char* qt_slot)const{  if (qt_slot == 0)    {    return true;    }  const char* ptr = qt_slot;  for (QString::const_iterator it = this->QtSlot.begin();       it != this->QtSlot.end() && *ptr != '\0'; )    {    if (*it == *ptr)      {      ++it;      ++ptr;      }    else if (*it == ' ')      {      ++it;      }    else if (*ptr == ' ')      {      ++ptr;      }    else      {      return false;      }    }  return true;}//-----------------------------------------------------------------------------// ctkVTKConnection methods//-----------------------------------------------------------------------------ctkVTKConnection::ctkVTKConnection(QObject* _parent):  Superclass(_parent)  , d_ptr(new ctkVTKConnectionPrivate(*this)){}// --------------------------------------------------------------------------ctkVTKConnection::ctkVTKConnection(ctkVTKConnectionPrivate* pimpl, QObject* _parent)  : Superclass(_parent)  , d_ptr(pimpl){}//-----------------------------------------------------------------------------ctkVTKConnection::~ctkVTKConnection(){  Q_D(ctkVTKConnection);  d->disconnectVTKObject();}//-----------------------------------------------------------------------------QString ctkVTKConnection::id()const{  Q_D(const ctkVTKConnection);  return d->Id;}//-----------------------------------------------------------------------------QObject* ctkVTKConnection::object()const{  Q_D(const ctkVTKConnection);  return const_cast<QObject*>(d->QtObject);}//-----------------------------------------------------------------------------vtkObject* ctkVTKConnection::vtkobject() const{  Q_D(const ctkVTKConnection);  return const_cast<vtkObject*>(d->VTKObject.GetPointer());}//-----------------------------------------------------------------------------QDebug operator<<(QDebug dbg, const ctkVTKConnection& connection){  const ctkVTKConnectionPrivate* d = connection.d_func();  dbg.nospace() << "ctkVTKConnection:" << &connection << endl                << "Id:" << d->Id << endl                << " VTKObject:" << (d->VTKObject ? d->VTKObject->GetClassName() : "<null>")                << "(" << d->VTKObject << ")" << endl                << " QtObject:" << d->QtObject << endl                << " VTKEvent:" << d->VTKEvent << endl                << " QtSlot:" << d->QtSlot << endl                << " SlotType:" << d->SlotType << endl                << " Priority:" << d->Priority << endl                << " Connected:" << d->Connected << endl                << " Blocked:" << d->Blocked;  return dbg.space();}//-----------------------------------------------------------------------------QString ctkVTKConnection::shortDescription(){  Q_D(ctkVTKConnection);    return ctkVTKConnection::shortDescription(d->VTKObject, d->VTKEvent, d->QtObject, d->QtSlot.toLatin1());}//-----------------------------------------------------------------------------QString ctkVTKConnection::shortDescription(vtkObject* vtk_obj, unsigned long vtk_event,    const QObject* qt_obj, const char* qt_slot){  QString ret;  QTextStream ts( &ret );  ts << (vtk_obj ? vtk_obj->GetClassName() : "NULL") << " "     << vtk_event << " " << qt_obj << " " << (qt_slot ? qt_slot : "");  return ret;}//-----------------------------------------------------------------------------bool ctkVTKConnection::isValid(vtkObject* vtk_obj, unsigned long vtk_event,                               const QObject* qt_obj, const char* qt_slot){  Q_UNUSED(vtk_event);  if (!vtk_obj)    {    return false;    }  if (!qt_obj)    {    return false;    }  if (qt_slot == 0 || strlen(qt_slot) == 0)    {    return false;    }  return true;}//-----------------------------------------------------------------------------void ctkVTKConnection::setup(vtkObject* vtk_obj, unsigned long vtk_event,                             const QObject* qt_obj, const char* qt_slot,                              float priority,                             Qt::ConnectionType connectionType){  Q_D(ctkVTKConnection);    if (!ctkVTKConnection::isValid(vtk_obj, vtk_event, qt_obj, qt_slot))     {     return;     }  d->VTKObject = vtk_obj;  d->QtObject = qt_obj;  d->VTKEvent = vtk_event;  d->QtSlot = qt_slot;  d->Priority = priority;  d->ConnectionType = connectionType;  if (d->QtSlot.contains(QRegExp(QString("\\( ?vtkObject ?\\* ?, ?vtkObject ?\\* ?\\)"))))    {    d->SlotType = ctkVTKConnectionPrivate::ARG_VTKOBJECT_AND_VTKOBJECT;    }  else    {    d->SlotType = ctkVTKConnectionPrivate::ARG_VTKOBJECT_VOID_ULONG_VOID;    }  d->connect();}//-----------------------------------------------------------------------------void ctkVTKConnection::setBlocked(bool block){  Q_D(ctkVTKConnection);  d->Blocked = block;}//-----------------------------------------------------------------------------bool ctkVTKConnection::isBlocked()const{  Q_D(const ctkVTKConnection);  return d->Blocked;}//-----------------------------------------------------------------------------bool ctkVTKConnection::isEqual(vtkObject* vtk_obj, unsigned long vtk_event,    const QObject* qt_obj, const char* qt_slot)const{  Q_D(const ctkVTKConnection);    if (vtk_obj && d->VTKObject != vtk_obj)    {    return false;    }  if (vtk_event != vtkCommand::NoEvent && d->VTKEvent != vtk_event)    {    return false;    }  if (qt_obj && d->QtObject != qt_obj)    {    return false;    }  if (!d->IsSameQtSlot(qt_slot))    {    return false;    }  return true;}//-----------------------------------------------------------------------------void ctkVTKConnectionPrivate::DoCallback(vtkObject* vtk_obj, unsigned long event,                                 void* client_data, void* call_data){  ctkVTKConnectionPrivate* conn = reinterpret_cast<ctkVTKConnectionPrivate*>(client_data);  Q_ASSERT(conn);  conn->execute(vtk_obj, event, client_data, call_data);}//-----------------------------------------------------------------------------// callback from VTK to emit signalvoid ctkVTKConnectionPrivate::execute(vtkObject* vtk_obj, unsigned long vtk_event,  void* client_data, void* call_data){  Q_Q(ctkVTKConnection);    Q_ASSERT(this->Connected);  if (this->Blocked)     {     return;     }  QPointer<ctkVTKConnection> connection(q);  if(!this->ObserveDeletion ||     vtk_event != vtkCommand::DeleteEvent ||     this->VTKEvent == vtkCommand::DeleteEvent)    {    vtkObject* callDataAsVtkObject = 0;    switch (this->SlotType)      {      case ctkVTKConnectionPrivate::ARG_VTKOBJECT_AND_VTKOBJECT:        if (this->VTKEvent == vtk_event)          {          callDataAsVtkObject = reinterpret_cast<vtkObject*>( call_data );          if (!callDataAsVtkObject)            {            qCritical() << "The VTKEvent(" << this->VTKEvent<< ") triggered by vtkObject("              << this->VTKObject->GetClassName() << ") "              << "doesn't return data of type vtkObject." << endl              << "The slot (" << this->QtSlot <<  ") owned by "              << "QObject(" << this->QtObject->objectName() << ")"              << " may be incorrect.";            }          emit q->emitExecute(vtk_obj, callDataAsVtkObject);          }        break;      case ctkVTKConnectionPrivate::ARG_VTKOBJECT_VOID_ULONG_VOID:        emit q->emitExecute(vtk_obj, call_data, vtk_event, client_data);        break;      default:        // Should never reach        qCritical() << "Unknown SlotType:" << this->SlotType;        return;        break;      }    }  if (!connection.isNull() &&      vtk_event == vtkCommand::DeleteEvent)    {    q->vtkObjectDeleted();    }}//-----------------------------------------------------------------------------void ctkVTKConnection::observeDeletion(bool enable){  Q_D(ctkVTKConnection);  d->ObserveDeletion = enable;}//-----------------------------------------------------------------------------bool ctkVTKConnection::deletionObserved()const{  Q_D(const ctkVTKConnection);  return d->ObserveDeletion;}//-----------------------------------------------------------------------------void ctkVTKConnection::disconnect(){  Q_D(ctkVTKConnection);  d->disconnectVTKObject();}//-----------------------------------------------------------------------------void ctkVTKConnection::vtkObjectDeleted(){  Q_D(ctkVTKConnection);  d->disconnectSlots();}//-----------------------------------------------------------------------------void ctkVTKConnection::qobjectDeleted(){  Q_D(ctkVTKConnection);  d->QtObject = 0;  d->disconnectVTKObject();}//-----------------------------------------------------------------------------void ctkVTKConnection::addObserver(vtkObject* caller, unsigned long vtk_event,  vtkCallbackCommand* callback, float priority){  caller->AddObserver(vtk_event, callback, priority);}//-----------------------------------------------------------------------------void ctkVTKConnection::removeObserver(vtkObject* caller, unsigned long vtk_event, vtkCallbackCommand* callback){  caller->RemoveObservers(vtk_event, callback);}
 |