123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389 |
- /*=========================================================================
- 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 <QEvent>
- #include <QMouseEvent>
- #include <QPointF>
- // CTK includes
- #include "ctkVTKMagnifyWidget.h"
- #include "ctkLogger.h"
- // VTK includes
- #include <QVTKWidget.h>
- #include <vtkRenderWindow.h>
- #include <vtkUnsignedCharArray.h>
- // Convenient macro
- #define VTK_CREATE(type, name) \
- vtkSmartPointer<type> name = vtkSmartPointer<type>::New()
- //--------------------------------------------------------------------------
- static ctkLogger logger("org.commontk.visualization.vtk.widgets.ctkVTKMagnifyWidget");
- //--------------------------------------------------------------------------
- //-----------------------------------------------------------------------------
- class ctkVTKMagnifyWidgetPrivate
- {
- Q_DECLARE_PUBLIC(ctkVTKMagnifyWidget);
- protected:
- ctkVTKMagnifyWidget* const q_ptr;
- public:
- ctkVTKMagnifyWidgetPrivate(ctkVTKMagnifyWidget& object);
- void init();
- void removePixmap();
- void observe(QVTKWidget * widget);
- void remove(QVTKWidget * widget);
- void updatePixmap(QVTKWidget * widget, QPointF pos);
- QList<QVTKWidget *> ObservedQVTKWidgets;
- double Magnification;
- };
- // --------------------------------------------------------------------------
- // ctkVTKMagnifyWidgetPrivate methods
- // --------------------------------------------------------------------------
- ctkVTKMagnifyWidgetPrivate::ctkVTKMagnifyWidgetPrivate(ctkVTKMagnifyWidget& object)
- :q_ptr(&object)
- {
- this->ObservedQVTKWidgets = QList<QVTKWidget *>();
- this->Magnification = 1.0;
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidgetPrivate::init()
- {
- this->removePixmap();
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidgetPrivate::removePixmap()
- {
- Q_Q(ctkVTKMagnifyWidget);
- QPixmap nullPixmap;
- q->setPixmap(nullPixmap);
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidgetPrivate::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(ctkVTKMagnifyWidget);
- widget->installEventFilter(q);
- }
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidgetPrivate::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(ctkVTKMagnifyWidget);
- widget->removeEventFilter(q);
- }
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidgetPrivate::updatePixmap(QVTKWidget * widget, QPointF pos)
- {
- Q_ASSERT(widget);
- Q_Q(ctkVTKMagnifyWidget);
- // Retrieve buffer of given QVTKWidget from its render window
- vtkRenderWindow * renderWindow = widget->GetRenderWindow();
- if (!renderWindow)
- {
- this->removePixmap();
- return;
- }
- // Get the window size and mouse position, and do error checking
- 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])
- {
- this->removePixmap();
- 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().
- QSizeF sizeToMagnify = QSizeF(q->size()) / this->Magnification;
- double dx0 = (mouseWindowPos.x() - ((sizeToMagnify.width()-1.0) / 2.0));
- double dx1 = (mouseWindowPos.x() + ((sizeToMagnify.width()-1.0) / 2.0));
- double dy0 = (mouseWindowPos.y() - ((sizeToMagnify.height()-1.0) / 2.0));
- double dy1 = (mouseWindowPos.y() + ((sizeToMagnify.height()-1.0) / 2.0));
- // Round to ints, for indexing into the pixel array
- int ix0 = floor(dx0);
- int ix1 = ceil(dx1);
- int iy0 = floor(dy0);
- int iy1 = ceil(dy1);
- // Handle when mouse is near the border
- int min_x0 = 0;
- int max_x1 = windowSize[0]-1;
- int min_y0 = 0;
- int max_y1 = windowSize[1]-1;
- bool overLeft = ix0 < min_x0;
- bool overRight = ix1 > max_x1;
- bool overBottom = iy0 < min_y0;
- bool overTop = iy1 > max_y1;
- // Ensure we don't access nonexistant indices
- if (overLeft)
- {
- ix0 = min_x0;
- dx0 = min_x0;
- }
- if (overRight)
- {
- ix1 = max_x1;
- dx1 = max_x1;
- }
- if (overBottom)
- {
- iy0 = min_y0;
- dy0 = min_y0;
- }
- if (overTop)
- {
- iy1 = max_y1;
- dy1 = max_y1;
- }
- // Error case
- if (ix0 > ix1 || iy0 > iy1)
- {
- this->removePixmap();
- 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
- QSize actualSize(ix1-ix0+1, iy1-iy0+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(ix0, iy0, ix1, iy1, front, pixelData);
- if (!success)
- {
- this->removePixmap();
- 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
- int x0diff = round((dx0 - static_cast<double>(ix0)) * this->Magnification);
- int x1diff = imageSize.width()
- - round((static_cast<double>(ix1) - dx1) * this->Magnification);
- int y0diff = round((dy0 - static_cast<double>(iy0)) * this->Magnification);
- int y1diff = imageSize.height()
- - round((static_cast<double>(iy1) - dy1) * this->Magnification);
- QRect cropRect(QPoint(x0diff, y0diff), QPoint(x1diff, y1diff));
- image = image.copy(cropRect);
- // Finally, set the pixelmap to the new one we have created
- q->setPixmap(QPixmap::fromImage(image));
- }
- //---------------------------------------------------------------------------
- // ctkVTKMagnifyWidget methods
- // --------------------------------------------------------------------------
- ctkVTKMagnifyWidget::ctkVTKMagnifyWidget(QWidget* parentWidget)
- : Superclass(parentWidget)
- , d_ptr(new ctkVTKMagnifyWidgetPrivate(*this))
- {
- Q_D(ctkVTKMagnifyWidget);
- d->init();
- }
- // --------------------------------------------------------------------------
- ctkVTKMagnifyWidget::~ctkVTKMagnifyWidget()
- {
- }
- // --------------------------------------------------------------------------
- CTK_GET_CPP(ctkVTKMagnifyWidget, double, magnification, Magnification);
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidget::setMagnification(double newMagnification)
- {
- if (newMagnification > 0)
- {
- Q_D(ctkVTKMagnifyWidget);
- d->Magnification = newMagnification;
- }
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidget::observe(QVTKWidget * widget)
- {
- Q_D(ctkVTKMagnifyWidget);
- if (widget)
- {
- d->observe(widget);
- }
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidget::observe(QList<QVTKWidget *> widgets)
- {
- foreach(QVTKWidget * widget, widgets)
- {
- this->observe(widget);
- }
- }
- // --------------------------------------------------------------------------
- bool ctkVTKMagnifyWidget::isObserved(QVTKWidget * widget) const
- {
- if (!widget)
- {
- return false;
- }
- Q_D(const ctkVTKMagnifyWidget);
- return d->ObservedQVTKWidgets.contains(widget);
- }
- // --------------------------------------------------------------------------
- int ctkVTKMagnifyWidget::numberObserved() const
- {
- Q_D(const ctkVTKMagnifyWidget);
- return d->ObservedQVTKWidgets.length();
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidget::remove(QVTKWidget * widget)
- {
- Q_D(ctkVTKMagnifyWidget);
- if (widget)
- {
- d->remove(widget);
- }
- }
- // --------------------------------------------------------------------------
- void ctkVTKMagnifyWidget::remove(QList<QVTKWidget *> widgets)
- {
- foreach(QVTKWidget * widget, widgets)
- {
- this->remove(widget);
- }
- }
- // --------------------------------------------------------------------------
- bool ctkVTKMagnifyWidget::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(ctkVTKMagnifyWidget);
- Q_ASSERT(d->ObservedQVTKWidgets.contains(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->updatePixmap(widget, mouseEvent->posF());
- this->update();
- return false;
- }
- // 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)
- {
- QPointF posF = widget->mapFromGlobal(QCursor::pos());
- d->updatePixmap(widget, posF);
- this->update();
- emit enteredObservedWidget(widget);
- return false;
- }
- // On leave, fill the pixmap with a solid color and emit signal of leave event
- else if (eventType == QEvent::Leave)
- {
- d->removePixmap();
- this->update();
- emit leftObservedWidget(widget);
- return false;
- }
- // For other event types, use standard event processing
- else
- {
- return QObject::eventFilter(obj, event);
- }
- }
|