| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627 | /*=========================================================================  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 <QEvent>#include <QMouseEvent>#include <QPointF>#include <QTimerEvent>// CTK includes#include "ctkVTKMagnifyView.h"#include "ctkVTKMagnifyView_p.h"#include "ctkLogger.h"// VTK includes#include <QVTKWidget.h>#include <vtkMath.h>#include <vtkRenderWindow.h>#include <vtkUnsignedCharArray.h>// STD includes#include <cmath>// Convenient macro#define VTK_CREATE(type, name) \  vtkSmartPointer<type> name = vtkSmartPointer<type>::New()//--------------------------------------------------------------------------static ctkLogger logger("org.commontk.visualization.vtk.widgets.ctkVTKMagnifyView");//--------------------------------------------------------------------------// --------------------------------------------------------------------------// ctkVTKMagnifyViewPrivate methods// --------------------------------------------------------------------------ctkVTKMagnifyViewPrivate::ctkVTKMagnifyViewPrivate(ctkVTKMagnifyView& object)  : QObject(&object), q_ptr(&object){  this->ObservedQVTKWidgets = QList<QVTKWidget *>();  this->Magnification = 1.0;  this->ObserveRenderWindowEvents = true;  this->EventHandler.EventType = NoEvent;  this->EventHandler.Position = QPointF(0,0);  this->EventHandler.UpdateInterval = 20;  this->EventHandler.TimerId = 0;}// --------------------------------------------------------------------------ctkVTKMagnifyViewPrivate::~ctkVTKMagnifyViewPrivate(){  if (this->EventHandler.TimerId != 0)    {    this->killTimer(this->EventHandler.TimerId);    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::init(){  // Start by removing the pixmap  this->EventHandler.EventType = RemovePixmapEvent;  this->removePixmap();  // Start the timer  this->restartTimer();}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::restartTimer(){  // Kill any old timers  if (this->EventHandler.TimerId != 0)    {    this->killTimer(this->EventHandler.TimerId);    this->EventHandler.TimerId = 0;    }  // Start timer if appropriate  if (this->EventHandler.UpdateInterval != 0)    {    this->EventHandler.TimerId = this->startTimer(this->EventHandler.UpdateInterval);    Q_ASSERT(this->EventHandler.TimerId);    }  // Not using any timers, process events as they come  else    {    this->EventHandler.TimerId = 0;    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::resetEventHandler(){  this->EventHandler.EventType = NoEvent;}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::timerEvent(QTimerEvent * event){  Q_ASSERT(event->timerId() == this->EventHandler.TimerId);  if (this->EventHandler.EventType == UpdatePixmapEvent)    {    this->updatePixmap();    }  else if (this->EventHandler.EventType == RemovePixmapEvent)    {    this->removePixmap();    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::pushUpdatePixmapEvent(){  if (this->EventHandler.Widget.isNull())    {    return;    }  this->pushUpdatePixmapEvent(        this->EventHandler.Widget.data()->mapFromGlobal(QCursor::pos()));}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::pushUpdatePixmapEvent(QPointF pos){  if (this->EventHandler.Widget.isNull())    {    return;    }  // Add this event to the queue  this->EventHandler.EventType = UpdatePixmapEvent;  this->EventHandler.Position = pos;  // Process the event if we handle all events  if (this->EventHandler.UpdateInterval == 0)    {    this->updatePixmap();    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::pushRemovePixmapEvent(){  // Add this event to the queue  this->EventHandler.EventType = RemovePixmapEvent;  // Process the event if we handle all events  if (this->EventHandler.UpdateInterval == 0)    {    this->removePixmap();    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::connectRenderWindow(QVTKWidget * widget){  Q_ASSERT(widget);  Q_ASSERT(this->ObserveRenderWindowEvents);  vtkRenderWindow * renderWindow = widget->GetRenderWindow();  if (renderWindow)    {    this->qvtkConnect(renderWindow, vtkCommand::EndEvent,                      this, SLOT(pushUpdatePixmapEvent()));    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::disconnectRenderWindow(QVTKWidget * widget){  Q_ASSERT(widget);  vtkRenderWindow * renderWindow = widget->GetRenderWindow();  if (renderWindow)    {    this->qvtkDisconnect(renderWindow, vtkCommand::EndEvent,                         this, SLOT(pushUpdatePixmapEvent()));    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::observe(QVTKWidget * widget){  Q_ASSERT(widget);  // If we are not already observing the widget, add it to the list and install  // the public implementation as the event filter to handle mousing  if (!this->ObservedQVTKWidgets.contains(widget))    {    this->ObservedQVTKWidgets.append(widget);    Q_Q(ctkVTKMagnifyView);    widget->installEventFilter(q);    if (this->ObserveRenderWindowEvents)      {      this->connectRenderWindow(widget);      }    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::remove(QVTKWidget * widget){  Q_ASSERT(widget);  // If we are observing the widget, remove it from the list and remove the  // public implementations event filtering  if (this->ObservedQVTKWidgets.contains(widget))    {    Q_ASSERT(this->ObservedQVTKWidgets.count(widget) == 1);    this->ObservedQVTKWidgets.removeOne(widget);    Q_Q(ctkVTKMagnifyView);    widget->removeEventFilter(q);    if (this->ObserveRenderWindowEvents)      {      this->disconnectRenderWindow(widget);      }    }}// --------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::removePixmap(){  Q_ASSERT(this->EventHandler.EventType == RemovePixmapEvent);  Q_Q(ctkVTKMagnifyView);  QPixmap nullPixmap;  q->setPixmap(nullPixmap);  q->update();  this->resetEventHandler();}// -------------------------------------------------------------------------void ctkVTKMagnifyViewPrivate::updatePixmap(){  Q_ASSERT(this->EventHandler.EventType == UpdatePixmapEvent);  Q_ASSERT(!this->EventHandler.Widget.isNull());  Q_Q(ctkVTKMagnifyView);  // Retrieve buffer of given QVTKWidget from its render window  vtkRenderWindow * renderWindow = this->EventHandler.Widget.data()->GetRenderWindow();  if (!renderWindow)    {    return;    }  // Get the window size and mouse position, and do error checking  QPointF pos = this->EventHandler.Position;  int * windowSize = renderWindow->GetSize();  QPointF mouseWindowPos(pos.x(), static_cast<double>(windowSize[1]-1)-pos.y());  if (mouseWindowPos.x() < 0 || mouseWindowPos.x() >= windowSize[0] ||      mouseWindowPos.y() < 0 || mouseWindowPos.y() >= windowSize[1])    {    return;    }  // Compute indices into the render window's data array  // Given a magnification and the label's widget size, compute the number of  // pixels we can show.  We should round to get a larger integer extent, since  // we will later adjust the pixmap's location in paintEvent().  // Left-right and up-down are in the render window coordinate frame.  // (which is different in the y-direction compared to Qt coordinates).  QSizeF sizeToMagnify = QSizeF(q->size()) / this->Magnification;  double posLeft = (mouseWindowPos.x() - ((sizeToMagnify.width()-1.0) / 2.0));  double posRight = (mouseWindowPos.x() + ((sizeToMagnify.width()-1.0) / 2.0));  double posBottom = (mouseWindowPos.y() - ((sizeToMagnify.height()-1.0) / 2.0));  double posTop = (mouseWindowPos.y() + ((sizeToMagnify.height()-1.0) / 2.0));  // Round to ints, for indexing into the pixel array  int indexLeft = std::floor(posLeft);  int indexRight = std::ceil(posRight);  int indexBottom = std::floor(posBottom);  int indexTop = std::ceil(posTop);  // Handle when mouse is near the border  int minLeft = 0;  int maxRight = windowSize[0]-1;  int minBottom = 0;  int maxTop = windowSize[1]-1;  bool overLeft = indexLeft < minLeft;  bool overRight = indexRight > maxRight;  bool overBottom = indexBottom < minBottom;  bool overTop = indexTop > maxTop;  // Ensure we don't access nonexistant indices  if (overLeft)    {    indexLeft = minLeft;    posLeft = minLeft;    }  if (overRight)    {    indexRight = maxRight;    posRight = maxRight;    }  if (overBottom)    {    indexBottom = minBottom;    posBottom = minBottom;    }  if (overTop)    {    indexTop = maxTop;    posTop = maxTop;    }  // Error case  if (indexLeft > indexRight || indexBottom > indexTop)    {    return;    }  // Setup the pixelmap's position in the label  Qt::Alignment alignment;  if (overLeft && !overRight)    {    alignment = Qt::AlignRight;    }  else if (overRight && !overLeft)    {    alignment = Qt::AlignLeft;    }  else    {    alignment = Qt::AlignLeft;    }  if (overBottom && !overTop)    {    alignment = alignment | Qt::AlignTop;    }  else if (overTop && !overBottom)    {    alignment = alignment | Qt::AlignBottom;    }  else    {    alignment = alignment | Qt::AlignTop;    }  q->setAlignment(alignment);  // Retrieve the pixel data into a QImage (flip vertically to move from render  // window coordinates to Qt coordinates)  QSize actualSize(indexRight-indexLeft+1, indexTop-indexBottom+1);  QImage image(actualSize.width(), actualSize.height(), QImage::Format_RGB32);  vtkUnsignedCharArray * pixelData = vtkUnsignedCharArray::New();  pixelData->SetArray(image.bits(), actualSize.width() * actualSize.height() * 4, 1);  int front = renderWindow->GetDoubleBuffer();  int success = renderWindow->GetRGBACharPixelData(      indexLeft, indexBottom, indexRight, indexTop, front, pixelData);  if (!success)    {    return;    }  pixelData->Delete();  image = image.rgbSwapped();  image = image.mirrored();  // Scale the image to zoom, using FastTransformation to prevent smoothing  QSize imageSize = actualSize * this->Magnification;  image = image.scaled(imageSize, Qt::KeepAspectRatioByExpanding,                       Qt::FastTransformation);  // Crop the magnified image to solve the problem of magnified partial pixels  double errorLeft      = (posLeft - static_cast<double>(indexLeft)) * this->Magnification;  double errorRight      = (static_cast<double>(indexRight) - posRight) * this->Magnification;  double errorBottom      = (posBottom - static_cast<double>(indexBottom)) * this->Magnification;  double errorTop      = (static_cast<double>(indexTop) - posTop) * this->Magnification;  // When cropping the Qt image, the 'adjust' variables are in Qt coordinates,  // not render window coordinates (bottom and top switch).  int cropIndexLeft = vtkMath::Round(errorLeft);  int cropIndexRight = imageSize.width() - vtkMath::Round(errorRight) - 1;  int cropIndexTop = vtkMath::Round(errorTop);  int cropIndexBottom = imageSize.height() - vtkMath::Round(errorBottom) - 1;  // Handle case when label size and magnification are not both even or odd  // (errorLeft/errorRight/errorBottom/errorTop will have fractional component,  // so cropped image wouldn't be the correct size unless we adjust further).  int requiredWidth = vtkMath::Round((posRight - posLeft + 1) * this->Magnification);  int requiredHeight = vtkMath::Round((posTop - posBottom + 1) * this->Magnification);  int actualWidth = cropIndexRight - cropIndexLeft + 1;  int actualHeight = cropIndexBottom - cropIndexTop + 1;  int diffWidth = requiredWidth - actualWidth;  int diffHeight = requiredHeight - actualHeight;  // Too wide  if (diffWidth < 0 && cropIndexRight != imageSize.width()-1)    {    Q_ASSERT(actualWidth - requiredWidth <= 1);    cropIndexRight += diffWidth;    }  // Too narrow  else if (diffWidth > 0 && cropIndexLeft != 0)    {    Q_ASSERT(requiredWidth - actualWidth <= 1);    cropIndexLeft -= diffWidth;    }  // Too tall  if (diffHeight < 0 && cropIndexBottom != imageSize.height()-1)    {    Q_ASSERT(actualHeight - requiredHeight <= 1);    cropIndexBottom += diffHeight;    }  // Too short  else if (diffHeight > 0 && cropIndexTop != 0)    {    Q_ASSERT(requiredHeight - actualHeight <= 1);    cropIndexTop -= diffHeight;    }  // Finally crop the QImage for display  QRect cropRect(QPoint(cropIndexLeft, cropIndexTop),                 QPoint(cropIndexRight, cropIndexBottom));  image = image.copy(cropRect);  // Finally, set the pixelmap to the new one we have created and update  q->setPixmap(QPixmap::fromImage(image));  q->update();  this->resetEventHandler();}//---------------------------------------------------------------------------// ctkVTKMagnifyView methods// --------------------------------------------------------------------------ctkVTKMagnifyView::ctkVTKMagnifyView(QWidget* parentWidget)  : Superclass(parentWidget)  , d_ptr(new ctkVTKMagnifyViewPrivate(*this)){  Q_D(ctkVTKMagnifyView);  d->init();}// --------------------------------------------------------------------------ctkVTKMagnifyView::~ctkVTKMagnifyView(){}// --------------------------------------------------------------------------CTK_GET_CPP(ctkVTKMagnifyView, double, magnification, Magnification)// --------------------------------------------------------------------------void ctkVTKMagnifyView::setMagnification(double newMagnification){  Q_D(ctkVTKMagnifyView);  if (newMagnification == d->Magnification || newMagnification <= 0)    {    return;    }  d->Magnification = newMagnification;  this->update();}// --------------------------------------------------------------------------CTK_GET_CPP(ctkVTKMagnifyView, bool,            observeRenderWindowEvents, ObserveRenderWindowEvents)// --------------------------------------------------------------------------void ctkVTKMagnifyView::setObserveRenderWindowEvents(bool newObserve){  Q_D(ctkVTKMagnifyView);  if (newObserve == d->ObserveRenderWindowEvents)    {    return;    }  d->ObserveRenderWindowEvents = newObserve;  // Connect/disconnect observations on vtkRenderWindow EndEvents, depending  // on whether we are switching from not-observing to observing or from  // observing to not-observing  QList<QVTKWidget *>::iterator it = d->ObservedQVTKWidgets.begin();  while (it != d->ObservedQVTKWidgets.end())    {    if (newObserve)      {      d->connectRenderWindow(*it);      }    else      {      d->disconnectRenderWindow(*it);      }    ++it;    }}// --------------------------------------------------------------------------int ctkVTKMagnifyView::updateInterval() const  {  Q_D(const ctkVTKMagnifyView);  return d->EventHandler.UpdateInterval;  }// --------------------------------------------------------------------------void ctkVTKMagnifyView::setUpdateInterval(int newInterval){  Q_D(ctkVTKMagnifyView);  if (newInterval == d->EventHandler.UpdateInterval || newInterval < 0)    {    return;    }  d->EventHandler.UpdateInterval = newInterval;  d->restartTimer();}// --------------------------------------------------------------------------void ctkVTKMagnifyView::observe(QVTKWidget * widget){  Q_D(ctkVTKMagnifyView);  if (widget)    {    d->observe(widget);    }}// --------------------------------------------------------------------------void ctkVTKMagnifyView::observe(QList<QVTKWidget *> widgets){  foreach(QVTKWidget * widget, widgets)    {    this->observe(widget);    }}// --------------------------------------------------------------------------bool ctkVTKMagnifyView::isObserved(QVTKWidget * widget) const{  if (!widget)    {    return false;    }  Q_D(const ctkVTKMagnifyView);  return d->ObservedQVTKWidgets.contains(widget);}// --------------------------------------------------------------------------int ctkVTKMagnifyView::numberObserved() const  {  Q_D(const ctkVTKMagnifyView);  return d->ObservedQVTKWidgets.length();  }// --------------------------------------------------------------------------void ctkVTKMagnifyView::remove(QVTKWidget * widget){  Q_D(ctkVTKMagnifyView);  if (widget)    {    d->remove(widget);    }}// --------------------------------------------------------------------------void ctkVTKMagnifyView::remove(QList<QVTKWidget *> widgets){  foreach(QVTKWidget * widget, widgets)    {    this->remove(widget);    }}// --------------------------------------------------------------------------bool ctkVTKMagnifyView::eventFilter(QObject * obj, QEvent * event){  // The given object should be a QVTKWidget in our list  QVTKWidget * widget = static_cast<QVTKWidget *>(obj);  Q_ASSERT(widget);  Q_D(ctkVTKMagnifyView);  Q_ASSERT(d->ObservedQVTKWidgets.contains(widget));  d->EventHandler.Widget = QWeakPointer<QVTKWidget>(widget);  QEvent::Type eventType = event->type();  // On mouse move, update the pixmap with the zoomed image  if (eventType == QEvent::MouseMove)    {    QMouseEvent * mouseEvent = static_cast<QMouseEvent *>(event);    Q_ASSERT(mouseEvent);    d->pushUpdatePixmapEvent(mouseEvent->posF());    }  // On enter, update the pixmap with the zoomed image (required for zooming when  // widget is created with mouse already within it), and emit signal of enter event  else if (eventType == QEvent::Enter)    {    d->pushUpdatePixmapEvent();    emit enteredObservedWidget(widget);    }  // On leave, fill the pixmap with a solid color and emit signal of leave event  else if (eventType == QEvent::Leave)    {    d->pushRemovePixmapEvent();    emit leftObservedWidget(widget);    }  // For other event types, use standard event processing  else    {    return QObject::eventFilter(obj, event);    }  return false;}
 |