diff --git a/Code/Mantid/Images/images.qrc b/Code/Mantid/Images/images.qrc index ad7484f85594ac36d8e5319fe7d0abac8d2fbb7e..4919bd69c64475181f1a18190de8ffb988abb23d 100644 --- a/Code/Mantid/Images/images.qrc +++ b/Code/Mantid/Images/images.qrc @@ -1,24 +1,29 @@ <RCC> - <qresource prefix="/"> - <file>MantidSplashScreen.png</file> - <file>MantidPlot_Icon_32offset.png</file> - <file>SNS_logo_trans_back.gif</file> - <file>Tessella_Logo_Transparent.gif</file> - <file>Mantid_Logo_Transparent_Cropped.png</file> - <file>HFIR_logo_small.png</file> - <file>ISIS_Logo_Transparent.gif</file> - <file>Mantid_Logo_Transparent.png</file> - <file>data_replace.png</file> - </qresource> - <qresource prefix="/Icons"> - <file>LoadFile.png</file> - </qresource> - <qresource prefix="/PickTools"> - <file>selection-tube.png</file> - <file>selection-box.png</file> - <file>selection-circle.png</file> - <file>selection-pointer.png</file> - <file>selection-text.png</file> - <file>selection-peak.png</file> - </qresource> + <qresource prefix="/"> + <file>MantidSplashScreen.png</file> + <file>MantidPlot_Icon_32offset.png</file> + <file>SNS_logo_trans_back.gif</file> + <file>Tessella_Logo_Transparent.gif</file> + <file>Mantid_Logo_Transparent_Cropped.png</file> + <file>HFIR_logo_small.png</file> + <file>ISIS_Logo_Transparent.gif</file> + <file>Mantid_Logo_Transparent.png</file> + <file>data_replace.png</file> + </qresource> + <qresource prefix="/Icons"> + <file>LoadFile.png</file> + </qresource> + <qresource prefix="/PickTools"> + <file>selection-tube.png</file> + <file>selection-box.png</file> + <file>selection-circle.png</file> + <file>selection-pointer.png</file> + <file>selection-text.png</file> + <file>selection-peak.png</file> + </qresource> + <qresource prefix="/MaskTools"> + <file>selection-pointer.png</file> + <file>selection-circle.png</file> + <file>selection-box.png</file> + </qresource> </RCC> diff --git a/Code/Mantid/MantidPlot/CMakeLists.txt b/Code/Mantid/MantidPlot/CMakeLists.txt index 7118832bae77fb25a93e48ebc525d1ade95d38c2..415fa0a1cd7f820b2fe0282388d27c60d1f44e73 100644 --- a/Code/Mantid/MantidPlot/CMakeLists.txt +++ b/Code/Mantid/MantidPlot/CMakeLists.txt @@ -195,6 +195,7 @@ set ( MANTID_SRCS src/Mantid/AbstractMantidLog.cpp src/Mantid/InstrumentWidget/InstrumentWindow.cpp src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.cpp src/Mantid/InstrumentWidget/InstrumentWindowPickTab.cpp + src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.cpp src/Mantid/InstrumentWidget/MantidColorMap.cpp src/Mantid/InstrumentWidget/MantidGLWidget.cpp src/Mantid/InstrumentWidget/ColorMapWidget.cpp @@ -211,6 +212,8 @@ set ( MANTID_SRCS src/Mantid/AbstractMantidLog.cpp src/Mantid/InstrumentWidget/CollapsiblePanel.cpp src/Mantid/InstrumentWidget/DetSelector.cpp src/Mantid/InstrumentWidget/XIntegrationControl.cpp + src/Mantid/InstrumentWidget/Shape2D.cpp + src/Mantid/InstrumentWidget/Shape2DCollection.cpp ) ########################################################################### @@ -406,6 +409,7 @@ set ( MANTID_HDRS src/Mantid/AbstractMantidLog.h src/Mantid/InstrumentWidget/InstrumentWindow.h src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.h src/Mantid/InstrumentWidget/InstrumentWindowPickTab.h + src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.h src/Mantid/InstrumentWidget/MantidColorMap.h src/Mantid/InstrumentWidget/MantidGLWidget.h src/Mantid/InstrumentWidget/ColorMapWidget.h @@ -422,6 +426,8 @@ set ( MANTID_HDRS src/Mantid/AbstractMantidLog.h src/Mantid/InstrumentWidget/CollapsiblePanel.h src/Mantid/InstrumentWidget/DetSelector.h src/Mantid/InstrumentWidget/XIntegrationControl.h + src/Mantid/InstrumentWidget/Shape2D.h + src/Mantid/InstrumentWidget/Shape2DCollection.h ) ########################################################################### @@ -656,12 +662,14 @@ set ( MANTID_MOC_FILES src/Mantid/AlgMonitor.h src/Mantid/InstrumentWidget/InstrumentWindow.h src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.h src/Mantid/InstrumentWidget/InstrumentWindowPickTab.h + src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.h src/Mantid/InstrumentWidget/ColorMapWidget.h src/Mantid/InstrumentWidget/MantidGLWidget.h src/Mantid/InstrumentWidget/OneCurvePlot.h src/Mantid/InstrumentWidget/CollapsiblePanel.h src/Mantid/InstrumentWidget/InstrumentActor.h src/Mantid/InstrumentWidget/ProjectionSurface.h + src/Mantid/InstrumentWidget/Shape2DCollection.h ) set ( UI_FILES src/Mantid/FirstTimeSetup.ui diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.cpp index fefaa65609ccaf4b92fa919cedc3d4e0ccfc3ec3..16864cd1773817628ac91706459a00370b84161f 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.cpp +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.cpp @@ -1,6 +1,7 @@ #include "InstrumentWindow.h" #include "InstrumentWindowRenderTab.h" #include "InstrumentWindowPickTab.h" +#include "InstrumentWindowMaskTab.h" #include "XIntegrationControl.h" #include "InstrumentActor.h" #include "UnwrappedCylinder.h" @@ -61,7 +62,6 @@ InstrumentWindow::InstrumentWindow(const QString& label, ApplicationWindow *app controlPanelLayout->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding); // Create the display widget - //mInstrumentDisplay = new Instrument3DWidget(this); m_InstrumentDisplay = new MantidGLWidget(this); controlPanelLayout->addWidget(m_InstrumentDisplay); mainLayout->addWidget(controlPanelLayout); @@ -85,6 +85,10 @@ InstrumentWindow::InstrumentWindow(const QString& label, ApplicationWindow *app m_pickTab = new InstrumentWindowPickTab(this); mControlsTab->addTab( m_pickTab, QString("Pick")); + // Mask controls + m_maskTab = new InstrumentWindowMaskTab(this); + mControlsTab->addTab( m_maskTab, QString("Mask")); + // Instrument tree controls QFrame* instrumentTree=createInstrumentTreeTab(mControlsTab); mControlsTab->addTab( instrumentTree, QString("Instrument Tree")); @@ -202,8 +206,9 @@ void InstrumentWindow::setSurfaceType(int type) surface = new UnwrappedSphere(m_instrumentActor,sample_pos,axis); } m_InstrumentDisplay->setSurface(surface); - //set3DAxesState(m_renderTab->areAxesOn()); m_InstrumentDisplay->update(); + m_maskTab->init(); + connect(surface,SIGNAL(singleDetectorTouched(int)),this,SLOT(singleDetectorTouched(int))); connect(surface,SIGNAL(singleDetectorPicked(int)),this,SLOT(singleDetectorPicked(int))); connect(surface,SIGNAL(multipleDetectorsSelected(QList<int>&)),this,SLOT(multipleDetectorsSelected(QList<int>&))); @@ -248,23 +253,23 @@ void InstrumentWindow::tabChanged(int i) QString text; if (i == 1) // picking { - if (surface) - { - surface->setInteractionModePick(); - } - m_InstrumentDisplay->setMouseTracking(true); + //if (surface) + //{ + // surface->setInteractionModePick(); + //} + //m_InstrumentDisplay->setMouseTracking(true); } else // no picking { if (surface) { - surface->setInteractionModeMove(); + //surface->setInteractionModeMove(); if (i == 0) { surface->componentSelected(); } } - m_InstrumentDisplay->setMouseTracking(false); + //m_InstrumentDisplay->setMouseTracking(false); } if (surface) { diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.h index 73624dc1e35e4527efab6748c2484f1908f44dc9..3d97a4a7fe26d3a5d3bcef7912f820f9bc147888 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.h +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindow.h @@ -33,6 +33,7 @@ class OneCurvePlot; class CollapsiblePanel; class InstrumentWindowRenderTab; class InstrumentWindowPickTab; +class InstrumentWindowMaskTab; class XIntegrationControl; // Qt forward declarations @@ -154,7 +155,7 @@ private slots: private: - QFrame * createPickTab(QTabWidget* ControlsTab); + //QFrame * createPickTab(QTabWidget* ControlsTab); QFrame * createInstrumentTreeTab(QTabWidget* ControlsTab); void loadSettings(); @@ -188,6 +189,7 @@ private: InstrumentWindowRenderTab * m_renderTab; InstrumentWindowPickTab * m_pickTab; + InstrumentWindowMaskTab * m_maskTab; XIntegrationControl * m_xIntegration; bool mViewChanged; ///< stores whether the user changed the view (so don't automatically change it) diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.cpp new file mode 100644 index 0000000000000000000000000000000000000000..3be2e5e627d0a5dab232eb486421b9109af475cf --- /dev/null +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.cpp @@ -0,0 +1,140 @@ +#include "InstrumentWindow.h" +#include "InstrumentWindowMaskTab.h" +#include "InstrumentActor.h" +#include "ProjectionSurface.h" + +#include <QVBoxLayout> +#include <QPushButton> +#include <QTextEdit> +#include <QMenu> +#include <QAction> +#include <QLabel> +#include <QMessageBox> + +#include <numeric> +#include <cfloat> +#include <algorithm> + +InstrumentWindowMaskTab::InstrumentWindowMaskTab(InstrumentWindow* instrWindow): +QFrame(instrWindow), +m_instrumentWindow(instrWindow), +m_activity(Select) +{ + m_instrumentDisplay = m_instrumentWindow->getInstrumentDisplay(); + + QVBoxLayout* layout=new QVBoxLayout(this); + + m_move = new QPushButton(); + m_move->setCheckable(true); + m_move->setAutoExclusive(true); + m_move->setIcon(QIcon(":/PickTools/selection-tube.png")); + m_move->setToolTip("Move the instrument"); + + m_pointer = new QPushButton(); + m_pointer->setCheckable(true); + m_pointer->setAutoExclusive(true); + m_pointer->setIcon(QIcon(":/MaskTools/selection-pointer.png")); + m_pointer->setToolTip("Select and edit shapes"); + + m_ellipse = new QPushButton(); + m_ellipse->setCheckable(true); + m_ellipse->setAutoExclusive(true); + m_ellipse->setIcon(QIcon(":/MaskTools/selection-circle.png")); + m_ellipse->setToolTip("Draw an ellipse"); + + m_rectangle = new QPushButton(); + m_rectangle->setCheckable(true); + m_rectangle->setAutoExclusive(true); + m_rectangle->setIcon(QIcon(":/MaskTools/selection-box.png")); + m_rectangle->setToolTip("Draw a rectangle"); + + m_ring_ellipse = new QPushButton(); + m_ring_ellipse->setCheckable(true); + m_ring_ellipse->setAutoExclusive(true); + m_ring_ellipse->setIcon(QIcon(":/MaskTools/selection-circle.png")); + m_ring_ellipse->setToolTip("Draw an ellipse"); + + m_ring_rectangle = new QPushButton(); + m_ring_rectangle->setCheckable(true); + m_ring_rectangle->setAutoExclusive(true); + m_ring_rectangle->setIcon(QIcon(":/MaskTools/selection-box.png")); + m_ring_rectangle->setToolTip("Draw a rectangle"); + + QHBoxLayout* toolBox = new QHBoxLayout(); + toolBox->addWidget(m_move); + toolBox->addWidget(m_pointer); + toolBox->addWidget(m_ellipse); + toolBox->addWidget(m_rectangle); + toolBox->addWidget(m_ring_ellipse); + toolBox->addWidget(m_ring_rectangle); + toolBox->addStretch(); + toolBox->setSpacing(2); + + connect(m_move,SIGNAL(clicked()),this,SLOT(setActivity())); + connect(m_pointer,SIGNAL(clicked()),this,SLOT(setActivity())); + connect(m_ellipse,SIGNAL(clicked()),this,SLOT(setActivity())); + connect(m_rectangle,SIGNAL(clicked()),this,SLOT(setActivity())); + connect(m_ring_ellipse,SIGNAL(clicked()),this,SLOT(setActivity())); + connect(m_ring_rectangle,SIGNAL(clicked()),this,SLOT(setActivity())); + m_move->setChecked(true); + + layout->addLayout(toolBox); + layout->addStretch(); +} + +void InstrumentWindowMaskTab::init() +{ + connect(m_instrumentDisplay->getSurface(),SIGNAL(shapeCreated()),this,SLOT(shapeCreated())); +} + +void InstrumentWindowMaskTab::setActivity() +{ + const QColor borderColor = Qt::red; + const QColor fillColor = QColor(255,255,255,100); + if (m_move->isChecked()) + { + m_activity = Move; + m_instrumentDisplay->getSurface()->setInteractionModeMove(); + } + else if (m_pointer->isChecked()) + { + m_activity = Select; + m_instrumentDisplay->getSurface()->setInteractionModeDraw(); + } + else if (m_ellipse->isChecked()) + { + m_activity = DrawEllipse; + m_instrumentDisplay->getSurface()->startCreatingShape2D("ellipse",borderColor,fillColor); + m_instrumentDisplay->getSurface()->setInteractionModeDraw(); + } + else if (m_rectangle->isChecked()) + { + m_activity = DrawEllipse; + m_instrumentDisplay->getSurface()->startCreatingShape2D("rectangle",borderColor,fillColor); + m_instrumentDisplay->getSurface()->setInteractionModeDraw(); + } + else if (m_ring_ellipse->isChecked()) + { + m_activity = DrawEllipse; + m_instrumentDisplay->getSurface()->startCreatingShape2D("ring ellipse",borderColor,fillColor); + m_instrumentDisplay->getSurface()->setInteractionModeDraw(); + } + else if (m_ring_rectangle->isChecked()) + { + m_activity = DrawEllipse; + m_instrumentDisplay->getSurface()->startCreatingShape2D("ring rectangle",borderColor,fillColor); + m_instrumentDisplay->getSurface()->setInteractionModeDraw(); + } +} + +void InstrumentWindowMaskTab::shapeCreated() +{ + m_pointer->setChecked(true); + setActivity(); +} + +void InstrumentWindowMaskTab::showEvent (QShowEvent *) +{ + setActivity(); + m_instrumentDisplay->setMouseTracking(true); +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.h new file mode 100644 index 0000000000000000000000000000000000000000..3d7353abd39fa937ddb6e0006ab1241efe98026f --- /dev/null +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowMaskTab.h @@ -0,0 +1,49 @@ +#ifndef INSTRUMENTWINDOWMASKTAB_H_ +#define INSTRUMENTWINDOWMASKTAB_H_ + +#include "MantidGLWidget.h" +#include "DetSelector.h" + +#include <QFrame> + +class InstrumentWindow; +class Instrument3DWidget; +class CollapsiblePanel; +class OneCurvePlot; + +class QPushButton; +class QTextEdit; +class QComboBox; +class QCheckBox; +class QLabel; + +/** + * Implements the Mask tab in InstrumentWindow + */ +class InstrumentWindowMaskTab: public QFrame +{ + Q_OBJECT +public: + enum Activity {Move = 0, Select = 1, DrawEllipse}; + InstrumentWindowMaskTab(InstrumentWindow* instrWindow); + void init(); +protected slots: + void setActivity(); + void shapeCreated(); +protected: + void showEvent (QShowEvent *); + + InstrumentWindow* m_instrumentWindow; + MantidGLWidget *m_instrumentDisplay; + + Activity m_activity; + QPushButton* m_move; + QPushButton* m_pointer; + QPushButton* m_ellipse; + QPushButton* m_rectangle; + QPushButton* m_ring_ellipse; + QPushButton* m_ring_rectangle; +}; + + +#endif /*INSTRUMENTWINDOWMASKTAB_H_*/ diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.cpp index 7d11f91c7a4fae5cadf1d38f95c848a62b5e4e8c..b01fd59125c6c702642b3720ea90534bef6cbbf4 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.cpp +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.cpp @@ -3,6 +3,7 @@ #include "OneCurvePlot.h" #include "CollapsiblePanel.h" #include "InstrumentActor.h" +#include "ProjectionSurface.h" #include "MantidKernel/ConfigService.h" #include "MantidAPI/AnalysisDataService.h" @@ -452,3 +453,13 @@ void InstrumentWindowPickTab::addPeak(double x,double y) //std::cerr << "l1=" << source->getDistance(*sample) << " l2=" << det->getDistance(*sample) << std::endl; //std::cerr << "2th=" << theta2/M_PI*180 << " phi=" << phi/M_PI*180 << std::endl; } + +void InstrumentWindowPickTab::showEvent (QShowEvent *) +{ + ProjectionSurface* surface = mInstrumentDisplay->getSurface(); + if (surface) + { + surface->setInteractionModePick(); + } + mInstrumentDisplay->setMouseTracking(true); +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.h index 10a6466cd2fc1549895ece88f4b9ddb9dae481a5..1b44000cb9d285c77f75bf53f9a525499646c568 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.h +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowPickTab.h @@ -36,6 +36,7 @@ private slots: void setSelectionType(); void addPeak(double,double); private: + void showEvent (QShowEvent *); void updatePlot(int detid); void updateSelectionInfo(int detid); void plotSingle(int detid); diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.cpp index 465127f73602eeb8cac9bd12e6190d80c46fa8b7..8ac7679e8429a9808799fe0b244643c2df70a3d0 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.cpp +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.cpp @@ -1,4 +1,5 @@ #include "InstrumentWindowRenderTab.h" +#include "ProjectionSurface.h" #include <QMenu> #include <QVBoxLayout> @@ -207,3 +208,12 @@ void InstrumentWindowRenderTab::showAxes(bool on) m_displayAxes->setChecked(on); m_displayAxes->blockSignals(false); } + +void InstrumentWindowRenderTab::showEvent (QShowEvent *) +{ + ProjectionSurface* surface = m_InstrumentDisplay->getSurface(); + if (surface) + { + surface->setInteractionModeMove(); + } +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.h index 4d1c28db04536e5f9e935845d10f36271ffb9b80..fc679a14e9a001fb7f92342db3281d6fdddfaf19 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.h +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/InstrumentWindowRenderTab.h @@ -41,6 +41,8 @@ private slots: void changeColormap(const QString & filename = ""); void showResetView(int); private: + void showEvent (QShowEvent *); + QFrame * setupAxisFrame(); InstrumentWindow* m_instrWindow; diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.cpp index 7e47953054189ca360a7e98a2b8e57693250c1bc..abf7a41d2bce857db959214d5ba0703e282bd8d3 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.cpp +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.cpp @@ -26,6 +26,7 @@ //1) if the sample buffers are not available then the paint of image on the mdi windows // seems to not work on intel chipset +const Qt::CursorShape cursorShape = Qt::ArrowCursor; MantidGLWidget::MantidGLWidget(QWidget* parent): QGLWidget(QGLFormat(QGL::DepthBuffer|QGL::NoAlphaChannel|QGL::SampleBuffers),parent), @@ -69,7 +70,7 @@ void MantidGLWidget::setSurface(ProjectionSurface* surface) */ void MantidGLWidget::initializeGL() { - setCursor(Qt::PointingHandCursor); // This is to set the initial window mouse cursor to Hand icon + setCursor(cursorShape); // This is to set the initial window mouse cursor to Hand icon // Set the relevant OpenGL rendering options setRenderingOptions(); @@ -260,7 +261,11 @@ void MantidGLWidget::wheelEvent(QWheelEvent* event) */ void MantidGLWidget::keyPressEvent(QKeyEvent *event) { - UNUSED_ARG(event) + if (m_surface) + { + m_surface->keyPressEvent(event); + } + update(); } /** @@ -270,7 +275,7 @@ void MantidGLWidget::keyPressEvent(QKeyEvent *event) void MantidGLWidget::keyReleaseEvent(QKeyEvent *event) { releaseKeyboard(); - setCursor(Qt::PointingHandCursor); + setCursor(cursorShape); m_isKeyPressed=false; if(!event->isAutoRepeat()) { @@ -365,10 +370,15 @@ void MantidGLWidget::componentSelected(Mantid::Geometry::ComponentID id) void MantidGLWidget::refreshView() { - //if( m_interactionMode == PickMode) //This is when in picking mode and the window is resized so update the image - //{ - //mPickingDraw=true; - //} m_surface->updateView(); update(); } + +void MantidGLWidget::leaveEvent (QEvent*) +{ + // Restore possible override cursor + while(QApplication::overrideCursor()) + { + QApplication::restoreOverrideCursor(); + } +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.h index d8757b3a7c6fa07115eee9cdcf5bf16de0504a0d..aac7d666df29e4505fb4e7bbe29b793cd6379a1f 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.h +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/MantidGLWidget.h @@ -48,6 +48,7 @@ protected: void keyReleaseEvent(QKeyEvent *); void draw(); void checkGLError(const QString& funName); + void leaveEvent (QEvent*); private: void setRenderingOptions(); diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.cpp index 389f5edfc2dfae75036acec173befcc7deca8100..022cc12e5a0ab554b00b6140fada6ae6ad243d57 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.cpp +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.cpp @@ -385,3 +385,10 @@ QString Projection3D::getInfoText()const } return text; } + +QRectF Projection3D::getSurfaceBounds()const +{ + double xmin,xmax,ymin,ymax,zmin,zmax; + m_viewport->getInstantProjection(xmin,xmax,ymin,ymax,zmin,zmax); + return QRectF(QPointF(xmin,ymin),QPointF(xmax,ymax)); +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.h index 96ac137ef379d687c54bb76dfc2210aeee9f5a11..02341655d0ed06b9f7f71df2d7218bcef8bcd83c 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.h +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Projection3D.h @@ -58,6 +58,7 @@ class Projection3D : public ProjectionSurface public: Projection3D(const InstrumentActor* rootActor,int winWidth,int winHeight); ~Projection3D(); + virtual QRectF getSurfaceBounds()const; void setViewDirection(const QString& vd); void set3DAxesState(bool on); diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.cpp index 8471122f095e37596fd3a0b9e040b0795a2f7d99..4e91f7c9af2f3711b480f4025b47788cd761108a 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.cpp +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.cpp @@ -36,6 +36,7 @@ ProjectionSurface::ProjectionSurface(const InstrumentActor* rootActor,const Mant m_leftButtonDown(false) { connect(rootActor,SIGNAL(colorMapChanged()),this,SLOT(colorMapChanged())); + connect(&m_maskShapes,SIGNAL(shapeCreated()),this,SLOT(catchShapeCreated())); } ProjectionSurface::~ProjectionSurface() @@ -111,14 +112,24 @@ void ProjectionSurface::draw(MantidGLWidget *widget,bool picking)const if (!picking) { - widget->swapBuffers(); + QPainter painter(widget); + QRectF windowRect = getSurfaceBounds(); + m_maskShapes.setWindow(windowRect,painter.viewport()); + m_maskShapes.draw(painter); + painter.end(); } m_viewChanged = false; + } else if (!picking) { QPainter painter(widget); painter.drawImage(0,0,**image); + + QRectF windowRect = getSurfaceBounds(); + m_maskShapes.setWindow(windowRect,painter.viewport()); + m_maskShapes.draw(painter); + // draw the selection rectangle if (!m_selectRect.isNull()) { @@ -132,49 +143,51 @@ void ProjectionSurface::draw(MantidGLWidget *widget,bool picking)const void ProjectionSurface::mousePressEvent(QMouseEvent* e) { - if (m_interactionMode == MoveMode) - { - this->mousePressEventMove(e); - } - else + switch(m_interactionMode) { - this->mousePressEventPick(e); + case MoveMode: this->mousePressEventMove(e); break; + case PickMode: this->mousePressEventPick(e); break; + case DrawMode: this->mousePressEventDraw(e); break; } } void ProjectionSurface::mouseMoveEvent(QMouseEvent* e) { - if (m_interactionMode == MoveMode) + switch(m_interactionMode) { - this->mouseMoveEventMove(e); - } - else - { - this->mouseMoveEventPick(e); + case MoveMode: this->mouseMoveEventMove(e); break; + case PickMode: this->mouseMoveEventPick(e); break; + case DrawMode: this->mouseMoveEventDraw(e); break; } } void ProjectionSurface::mouseReleaseEvent(QMouseEvent* e) { - if (m_interactionMode == MoveMode) + switch(m_interactionMode) { - this->mouseReleaseEventMove(e); - } - else - { - this->mouseReleaseEventPick(e); + case MoveMode: this->mouseReleaseEventMove(e); break; + case PickMode: this->mouseReleaseEventPick(e); break; + case DrawMode: this->mouseReleaseEventDraw(e); break; } } void ProjectionSurface::wheelEvent(QWheelEvent* e) { - if (m_interactionMode == MoveMode) + switch(m_interactionMode) { - this->wheelEventMove(e); + case MoveMode: this->wheelEventMove(e); break; + case PickMode: this->wheelEventPick(e); break; + case DrawMode: this->wheelEventDraw(e); break; } - else +} + +void ProjectionSurface::keyPressEvent(QKeyEvent* e) +{ + switch(m_interactionMode) { - this->wheelEventPick(e); + case MoveMode: break; + case PickMode: break; + case DrawMode: this->keyPressEventDraw(e); break; } } @@ -218,6 +231,32 @@ void ProjectionSurface::wheelEventPick(QWheelEvent*) { } +void ProjectionSurface::mousePressEventDraw(QMouseEvent* e) +{ + m_maskShapes.mousePressEvent(e); +} + +void ProjectionSurface::mouseMoveEventDraw(QMouseEvent* e) +{ + m_maskShapes.mouseMoveEvent(e); +} + +void ProjectionSurface::mouseReleaseEventDraw(QMouseEvent* e) +{ + m_maskShapes.mouseReleaseEvent(e); +} + +void ProjectionSurface::wheelEventDraw(QWheelEvent* e) +{ + m_maskShapes.wheelEvent(e); +} + +void ProjectionSurface::keyPressEventDraw(QKeyEvent* e) +{ + m_maskShapes.keyPressEvent(e); +} + + void ProjectionSurface::startSelection(int x,int y) { m_selectRect.setRect(x,y,1,1); @@ -365,11 +404,18 @@ void ProjectionSurface::colorMapChanged() void ProjectionSurface::setInteractionModePick() { m_interactionMode = PickMode; + m_maskShapes.deselectAll(); } void ProjectionSurface::setInteractionModeMove() { m_interactionMode = MoveMode; + m_maskShapes.deselectAll(); +} + +void ProjectionSurface::setInteractionModeDraw() +{ + m_interactionMode = DrawMode; } /** @@ -408,3 +454,15 @@ QString ProjectionSurface::getPickInfoText()const return "Move cursor over instrument to see detector information.\n" "Left click and drag to select multiple detectors."; } + + // --- Shape2D manipulation --- // + +void ProjectionSurface::startCreatingShape2D(const QString& type,const QColor& borderColor,const QColor& fillColor) +{ + m_maskShapes.startCreatingShape2D(type,borderColor,fillColor); +} + +void ProjectionSurface::catchShapeCreated() +{ + emit shapeCreated(); +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.h index ce002731611c099cac0eba673902bf22fa9befcf..507b71bc40890d61c886ba2df2ffab59c7512316 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.h +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/ProjectionSurface.h @@ -4,14 +4,14 @@ #include "MantidKernel/V3D.h" #include "MantidKernel/Quat.h" #include "MantidGeometry/IComponent.h" + #include "InstrumentActor.h" -#include <boost/shared_ptr.hpp> +#include "Shape2DCollection.h" #include <QImage> #include <QList> #include <QStack> -#include <QSet> -#include <QMap> +#include <QColor> namespace Mantid{ namespace Geometry{ @@ -39,7 +39,7 @@ class ProjectionSurface: public QObject { Q_OBJECT public: - enum InteractionMode {MoveMode = 0, PickMode = 1}; ///< Move around or select things + enum InteractionMode {MoveMode = 0, PickMode = 1, DrawMode}; ///< Move around or select things ProjectionSurface(const InstrumentActor* rootActor,const Mantid::Kernel::V3D& origin,const Mantid::Kernel::V3D& axis); virtual ~ProjectionSurface(); @@ -51,14 +51,18 @@ public: virtual void updateView(); /// full update and redraw of the surface virtual void updateDetectors(); + /// returns the bounding rectangle in the real coordinates + virtual QRectF getSurfaceBounds()const{return m_viewRect;} - virtual void mousePressEvent(QMouseEvent* event); + virtual void mousePressEvent(QMouseEvent*); virtual void mouseMoveEvent(QMouseEvent*); virtual void mouseReleaseEvent(QMouseEvent*); virtual void wheelEvent(QWheelEvent *); + virtual void keyPressEvent(QKeyEvent*); void setInteractionModeMove(); void setInteractionModePick(); + void setInteractionModeDraw(); InteractionMode getInteractionMode()const{return m_interactionMode;} /// start selection at a point on the screen @@ -83,21 +87,29 @@ public: /// Unzoom view to the previous zoom area or to full view virtual void unzoom(); + // --- Shape2D manipulation --- // + + void startCreatingShape2D(const QString& type,const QColor& borderColor,const QColor& fillColor = QColor()); + signals: void singleDetectorTouched(int); void singleDetectorPicked(int); void multipleDetectorsSelected(QList<int>&); + void shapeCreated(); + protected slots: void colorMapChanged(); + void catchShapeCreated(); protected: virtual void init() = 0; virtual void drawSurface(MantidGLWidget* widget,bool picking = false)const = 0; /// Respond to a change of color map in m_instrActor virtual void changeColorMap() = 0; + virtual void mousePressEventMove(QMouseEvent*){} virtual void mouseMoveEventMove(QMouseEvent*){} virtual void mouseReleaseEventMove(QMouseEvent*){} @@ -108,6 +120,12 @@ protected: virtual void mouseReleaseEventPick(QMouseEvent*); virtual void wheelEventPick(QWheelEvent*); + virtual void mousePressEventDraw(QMouseEvent*); + virtual void mouseMoveEventDraw(QMouseEvent*); + virtual void mouseReleaseEventDraw(QMouseEvent*); + virtual void wheelEventDraw(QWheelEvent*); + virtual void keyPressEventDraw(QKeyEvent*); + void draw(MantidGLWidget* widget,bool picking)const; void clear(); QRect selectionRect()const; @@ -129,6 +147,8 @@ protected: QStack<QRectF> m_zoomStack; InteractionMode m_interactionMode; bool m_leftButtonDown; + + Shape2DCollection m_maskShapes; }; #endif // PROJECTIONSURFACE_H diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2D.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2D.cpp new file mode 100644 index 0000000000000000000000000000000000000000..640da7559a4a75d06b7fcb38cbeeb545cfe9bba7 --- /dev/null +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2D.cpp @@ -0,0 +1,334 @@ +#include "Shape2D.h" + +#include <QPainter> +#include <QPainterPath> +#include <QMouseEvent> +#include <QWheelEvent> + +#include <iostream> +#include <algorithm> + +// number of control points common for all shapes +const size_t Shape2D::NCommonCP = 4; +const qreal Shape2D::sizeCP = 2; + +Shape2D::Shape2D(): +m_color(Qt::red), +m_editing(false) +{ + +} + +void Shape2D::draw(QPainter& painter) const +{ + painter.setPen(m_color); + this->drawShape(painter); + if (m_editing) + { + QColor c(255,255,255,100); + painter.setPen(c); + painter.setCompositionMode(QPainter::CompositionMode_Plus); + painter.drawRect(m_boundingRect); + for(size_t i = 0; i < getNControlPoints(); ++i) + { + QPointF p = painter.transform().map(getControlPoint(i)); + QRectF r(p - QPointF(sizeCP,sizeCP),p + QPointF(sizeCP,sizeCP)); + painter.save(); + painter.resetTransform(); + painter.fillRect(r,c); + painter.restore(); + } + } +} + +size_t Shape2D::getNControlPoints() const +{ + return NCommonCP + this->getShapeNControlPoints(); +} + +QPointF Shape2D::getControlPoint(size_t i) const +{ + if ( i >= getNControlPoints()) + { + throw std::range_error("Control point index is out of range"); + } + + switch(i) + { + case 0: return m_boundingRect.topLeft(); + case 1: return m_boundingRect.topRight(); + case 2: return m_boundingRect.bottomRight(); + case 3: return m_boundingRect.bottomLeft(); + } + return getShapeControlPoint(i - NCommonCP); +} + +void Shape2D::setControlPoint(size_t i,const QPointF& pos) +{ + if ( i >= getNControlPoints()) + { + throw std::range_error("Control point index is out of range"); + } + + switch(i) + { + case 0: m_boundingRect.setTopLeft(pos); correctBoundingRect(); refit(); break; + case 1: m_boundingRect.setTopRight(pos); correctBoundingRect(); refit(); break; + case 2: m_boundingRect.setBottomRight(pos); correctBoundingRect(); refit(); break; + case 3: m_boundingRect.setBottomLeft(pos); correctBoundingRect(); refit(); break; + } + setShapeControlPoint(i - NCommonCP, pos); + adjustBoundingRect(); +} + +void Shape2D::correctBoundingRect() +{ + qreal left = m_boundingRect.left(); + qreal top = m_boundingRect.top(); + qreal width = m_boundingRect.width(); + qreal height = m_boundingRect.height(); + if (m_boundingRect.width() < 0) + { + left = m_boundingRect.right(); + width *= -1; + } + + if (m_boundingRect.height() < 0) + { + top = m_boundingRect.bottom(); + height *= -1; + } + + m_boundingRect = QRectF(left,top,width,height); + +} + +void Shape2D::moveBy(const QPointF& dp) +{ + m_boundingRect.adjust(dp.x(),dp.y(),dp.x(),dp.y()); + refit(); +} + +void Shape2D::adjustBoundingRect(qreal dx1,qreal dy1,qreal dx2,qreal dy2) +{ + qreal dwidth = dx2 - dx1; + if (dwidth <= - m_boundingRect.width()) + { + qreal mu = m_boundingRect.width() / fabs(dwidth); + dx1 *= mu; + dx2 *= mu; + } + qreal dheight = dy2 - dy1; + if (dheight <= - m_boundingRect.height()) + { + qreal mu = m_boundingRect.height() / fabs(dheight); + dy1 *= mu; + dy2 *= mu; + } + m_boundingRect.adjust(dx1,dy1,dx2,dy2); + refit(); +} + +void Shape2D::setBoundingRect(const QRectF& rect) +{ + m_boundingRect = rect; + refit(); +} + +// --- Shape2DEllipse --- // + +Shape2DEllipse::Shape2DEllipse(const QPointF& center,double radius1,double radius2) +:Shape2D() +{ + if (radius2 == 0) + { + radius2 = radius1; + } + QPointF dr(radius1,radius2); + m_boundingRect = QRectF(center - dr, center + dr); +} + +void Shape2DEllipse::drawShape(QPainter& painter) const +{ + painter.drawEllipse(m_boundingRect); + if (m_fill_color != QColor()) + { + QPainterPath path; + path.addEllipse(m_boundingRect); + painter.fillPath(path,m_fill_color); + } +} + +void Shape2DEllipse::addToPath(QPainterPath& path) const +{ + path.addEllipse(m_boundingRect); +} + +bool Shape2DEllipse::selectAt(const QPointF& p)const +{ + if (m_fill_color != QColor()) + {// filled ellipse + return contains(p); + } + + double a = m_boundingRect.width() / 2; + if (a == 0.0) a = 1.0; + double b = m_boundingRect.height() / 2; + if (b == 0.0) b = 1.0; + double xx = m_boundingRect.left() + a - double(p.x()); + double yy = m_boundingRect.top() + b - double(p.y()); + + double f = fabs(xx*xx/(a*a) + yy*yy/(b*b) - 1); + + return f < 0.1; +} + +bool Shape2DEllipse::contains(const QPointF& p)const +{ + QPointF pp = m_boundingRect.center() - p; + double a = m_boundingRect.width() / 2; + if (a == 0.0) a = 1.0; + double b = m_boundingRect.height() / 2; + if (b == 0.0) b = 1.0; + double xx = pp.x(); + double yy = pp.y(); + + double f = xx*xx/(a*a) + yy*yy/(b*b); + + return f <= 1.0; +} + +// --- Shape2DRectangle --- // + +Shape2DRectangle::Shape2DRectangle(const QPointF& leftTop,const QPointF& bottomRight) +{ + m_boundingRect = QRectF(leftTop,bottomRight); +} + +Shape2DRectangle::Shape2DRectangle(const QPointF& leftTop,const QSizeF& size) +{ + m_boundingRect = QRectF(leftTop,size); +} + +bool Shape2DRectangle::selectAt(const QPointF& p)const +{ + if (m_fill_color != QColor()) + {// filled rectangle + return contains(p); + } + + QRectF outer(m_boundingRect); + outer.adjust(-2,-2,2,2); + QRectF inner(m_boundingRect); + inner.adjust(2,2,-2,-2); + return outer.contains(p) && !inner.contains(p); +} + +void Shape2DRectangle::drawShape(QPainter& painter) const +{ + painter.drawRect(m_boundingRect); + if (m_fill_color != QColor()) + { + QPainterPath path; + path.addRect(m_boundingRect); + painter.fillPath(path,m_fill_color); + } +} + +void Shape2DRectangle::addToPath(QPainterPath& path) const +{ + path.addRect(m_boundingRect); +} + +// --- Shape2DRing --- // + +Shape2DRing::Shape2DRing(Shape2D* shape): +m_outer_shape(shape), +m_width(10.0), +m_stored_width(10.0) +{ + m_inner_shape = m_outer_shape->clone(); + QRectF inner_rect = m_inner_shape->getBoundingRect(); + m_inner_shape->adjustBoundingRect(m_width,m_width,-m_width,-m_width); + adjustBoundingRect(); + m_outer_shape->setFillColor(QColor()); + m_inner_shape->setFillColor(QColor()); +} + +Shape2DRing::Shape2DRing(const Shape2DRing& ring): +Shape2D(), +m_outer_shape(ring.m_outer_shape->clone()), +m_inner_shape(ring.m_inner_shape->clone()), +m_width(ring.m_width), +m_stored_width(ring.m_stored_width) +{ + adjustBoundingRect(); +} + +bool Shape2DRing::selectAt(const QPointF& p)const +{ + return contains(p); +} + +bool Shape2DRing::contains(const QPointF& p)const +{ + return m_outer_shape->contains(p) && !m_inner_shape->contains(p); +} + +void Shape2DRing::drawShape(QPainter& painter) const +{ + m_outer_shape->draw(painter); + m_inner_shape->draw(painter); + if (m_fill_color != QColor()) + { + QPainterPath path; + m_outer_shape->addToPath(path); + m_inner_shape->addToPath(path); + painter.fillPath(path,m_fill_color); + } +} + +void Shape2DRing::refit() +{ + if (m_stored_width <= 0) m_stored_width = 1.0; + m_width = m_stored_width; + qreal max_width = std::max(m_boundingRect.width() / 2, m_boundingRect.height() / 2) - 1.0; + if (m_width > max_width) m_width = max_width; + m_outer_shape->setBoundingRect(m_boundingRect); + m_inner_shape->setBoundingRect(m_boundingRect); + m_inner_shape->adjustBoundingRect(m_width,m_width,-m_width,-m_width); +} + +void Shape2DRing::adjustBoundingRect() +{ + m_boundingRect = m_outer_shape->getBoundingRect(); +} + +QPointF Shape2DRing::getShapeControlPoint(size_t i) const +{ + QRectF rect = m_inner_shape->getBoundingRect(); + switch(i) + { + case 0: return QPointF(m_boundingRect.center().x(), m_boundingRect.top()); + case 1: return QPointF(m_boundingRect.center().x(), m_boundingRect.bottom()); + case 2: return QPointF(rect.left(),rect.center().y()); + case 3: return QPointF(rect.right(),rect.center().y()); + } + return QPointF(); +} + +void Shape2DRing::setShapeControlPoint(size_t i,const QPointF& pos) +{ + QPointF dp = pos - getShapeControlPoint(i); + + switch(i) + { + case 0: Shape2D::adjustBoundingRect(dp.y(),dp.y(),-dp.y(),-dp.y()); break; + case 1: Shape2D::adjustBoundingRect(-dp.y(),-dp.y(),dp.y(),dp.y()); break; + case 2: m_stored_width += dp.x(); + refit(); break; + case 3: m_stored_width -= dp.x(); + refit(); break; + } + +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2D.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2D.h new file mode 100644 index 0000000000000000000000000000000000000000..bda92313c655871d9ba48df2a210f2c40c87314a --- /dev/null +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2D.h @@ -0,0 +1,120 @@ +#ifndef MANTIDPLOT_SHAPE2D_H_ +#define MANTIDPLOT_SHAPE2D_H_ + +#include <QColor> +#include <QPointF> +#include <QRectF> +#include <QPolygonF> +#include <QList> +#include <QTransform> + +class QPainter; +class QPainterPath; +class QMouseEvent; +class QWheelEvent; + +/** + * Base class for an editable 2D shape, which can be drawn on ProjectionSurface. + */ +class Shape2D +{ +public: + Shape2D(); + virtual ~Shape2D(){} + + virtual Shape2D* clone()const = 0; + virtual void addToPath(QPainterPath& path) const = 0; + // make sure the shape is withing the bounding box + virtual void refit() = 0; + + virtual void draw(QPainter& painter) const; + virtual void moveBy(const QPointF& pos); + virtual size_t getNControlPoints() const; + virtual QPointF getControlPoint(size_t i) const; + virtual void setControlPoint(size_t i,const QPointF& pos); + virtual QRectF getBoundingRect() const {return m_boundingRect;} + // move the left, top, right and bottom sides of the bounding rect + // by dx1, dy1, dx2, and dy2 correspondingly + virtual void adjustBoundingRect(qreal dx1,qreal dy1,qreal dx2,qreal dy2); + virtual void setBoundingRect(const QRectF& rect); + + void setColor(const QColor& color){m_color = color;} + void setFillColor(const QColor& color){m_fill_color = color;} + void edit(bool on){m_editing = on;} + bool isEditing()const{return m_editing;} + virtual bool selectAt(const QPointF& p)const{return false;} + virtual bool contains(const QPointF& p)const{return false;} + +protected: + virtual void drawShape(QPainter& painter) const = 0; + + // return number of control points specific to this shape + virtual size_t getShapeNControlPoints() const{return 0;} + // returns position of a shape specific control point, 0 < i < getShapeNControlPoints() + virtual QPointF getShapeControlPoint(size_t i) const{return QPointF();} + // sets position of a shape specific control point, 0 < i < getShapeNControlPoints() + virtual void setShapeControlPoint(size_t i,const QPointF& pos){} + // make sure the bounding box is correct + virtual void adjustBoundingRect() {} + + void correctBoundingRect(); + + static const size_t NCommonCP; + static const qreal sizeCP; + QRectF m_boundingRect; + QColor m_color; + QColor m_fill_color; + bool m_editing; +}; + +class Shape2DEllipse: public Shape2D +{ +public: + Shape2DEllipse(const QPointF& center,double radius1,double radius2 = 0); + virtual Shape2D* clone()const{return new Shape2DEllipse(*this);} + virtual bool selectAt(const QPointF& p)const; + virtual bool contains(const QPointF& p)const; + virtual void addToPath(QPainterPath& path) const; +protected: + virtual void drawShape(QPainter& painter) const; + virtual void refit(){} +}; + +class Shape2DRectangle: public Shape2D +{ +public: + Shape2DRectangle(const QPointF& leftTop,const QPointF& bottomRight); + Shape2DRectangle(const QPointF& leftTop,const QSizeF& size); + virtual Shape2D* clone()const{return new Shape2DRectangle(*this);} + virtual bool selectAt(const QPointF& p)const; + virtual bool contains(const QPointF& p)const{return m_boundingRect.contains(p);} + virtual void addToPath(QPainterPath& path) const; +protected: + virtual void drawShape(QPainter& painter) const; + virtual void refit(){} +}; + +class Shape2DRing: public Shape2D +{ +public: + Shape2DRing(Shape2D* shape); + Shape2DRing(const Shape2DRing& ring); + virtual Shape2D* clone()const{return new Shape2DRing(*this);} + virtual bool selectAt(const QPointF& p)const; + virtual bool contains(const QPointF& p)const; +protected: + virtual void drawShape(QPainter& painter) const; + virtual void addToPath(QPainterPath& path) const {} + virtual void refit(); + virtual void adjustBoundingRect(); + virtual size_t getShapeNControlPoints() const{return 4;} + virtual QPointF getShapeControlPoint(size_t i) const; + virtual void setShapeControlPoint(size_t i,const QPointF& pos); + Shape2D* m_outer_shape; + Shape2D* m_inner_shape; + qreal m_width; + qreal m_stored_width; +}; + + +#endif /*MANTIDPLOT_SHAPE2D_H_*/ diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2DCollection.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2DCollection.cpp new file mode 100644 index 0000000000000000000000000000000000000000..243bc520c8b3b14f1a233bd67034b98e1de0d0e3 --- /dev/null +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2DCollection.cpp @@ -0,0 +1,280 @@ +#include "Shape2DCollection.h" + +#include <QPainter> +#include <QMouseEvent> +#include <QWheelEvent> +#include <QKeyEvent> +#include <QApplication> +#include <QStringList> + +#include <iostream> +#include <stdexcept> + +Shape2DCollection::Shape2DCollection(): +Shape2D(), +m_creating(false), +m_editing(false), +m_moving(false), +m_x(0), +m_y(0), +m_currentShape(NULL), +m_leftButtonPressed(false), +m_overridingCursor(false) +{ +} + +Shape2DCollection::~Shape2DCollection() +{ + foreach(Shape2D* shape,m_shapes) + { + delete shape; + } +} + +void Shape2DCollection::draw(QPainter& painter) const +{ + painter.save(); + painter.setTransform(m_transform); + foreach(const Shape2D* shape,m_shapes) + { + shape->draw(painter); + } + painter.restore(); +} + +void Shape2DCollection::addShape(Shape2D* shape) +{ + m_shapes.push_back(shape); + m_boundingRect |= shape->getBoundingRect(); + emit shapeCreated(); +} + +void Shape2DCollection::setWindow(const QRectF& rect,const QRect& viewport) const +{ + m_transform.reset(); + if ( m_windowRect.isNull() ) + { + m_windowRect = rect; + m_h = viewport.height(); + m_wx = viewport.width() / rect.width(); + m_wy = m_h / rect.height(); + } + else + { + double wx = viewport.width() / rect.width(); + double wy = viewport.height() / rect.height(); + double rx = m_windowRect.left() - rect.left(); + double ry = m_windowRect.top() - rect.top(); + qreal sx = wx / m_wx; + qreal sy = wy / m_wy; + qreal dx = rx * wx; + qreal dy = viewport.height() - sy * m_h - ry * wy; + + m_transform.translate(dx,dy); + m_transform.scale(sx,sy); + } +} + +void Shape2DCollection::refit() +{ +} + +void Shape2DCollection::adjustBoundingRect() +{ + m_boundingRect = QRectF(); + foreach(const Shape2D* shape,m_shapes) + { + m_boundingRect |= shape->getBoundingRect(); + } +} + +void Shape2DCollection::mousePressEvent(QMouseEvent* e) +{ + if (e->button() == Qt::LeftButton) + { + m_leftButtonPressed = true; + if (m_creating && !m_shapeType.isEmpty()) + { + deselectAll(); + addShape(m_shapeType,e->x(),e->y()); + if (!m_currentShape) return; + m_currentShape->edit(true); + m_currentCP = 2; + m_editing = true; + } + else if (selectControlPointAt(e->x(),e->y())) + { + m_editing = true; + } + else if (selectAt(e->x(),e->y())) + { + m_x = e->x(); + m_y = e->y(); + m_moving = true; + } + else + { + deselectAll(); + } + } +} + +void Shape2DCollection::mouseMoveEvent(QMouseEvent* e) +{ + if (m_editing) + { + if (m_leftButtonPressed && m_currentShape && m_currentShape->isEditing() && m_currentCP < m_currentShape->getNControlPoints()) + { + QPointF p = m_transform.inverted().map(QPointF(e->x(),e->y())); + m_currentShape->setControlPoint(m_currentCP,p); + } + } + else if (m_moving && m_leftButtonPressed && m_currentShape) + { + QPointF p1 = m_transform.inverted().map(QPointF( e->x(),e->y() )); + QPointF p2 = m_transform.inverted().map(QPointF( m_x, m_y )); + m_currentShape->moveBy(p1 - p2); + m_x = e->x(); + m_y = e->y(); + } + else if (selectControlPointAt(e->x(),e->y()) || isOverCurrentAt(e->x(),e->y())) + { + m_overridingCursor = true; + QApplication::setOverrideCursor(Qt::SizeAllCursor); + } + else if (m_overridingCursor) + { + QApplication::restoreOverrideCursor(); + } +} + +void Shape2DCollection::mouseReleaseEvent(QMouseEvent* e) +{ + if (e->button() == Qt::LeftButton) + { + m_leftButtonPressed = false; + } + m_creating = false; + m_editing = false; + m_moving = false; +} + +void Shape2DCollection::wheelEvent(QWheelEvent* e) +{ +} + +void Shape2DCollection::keyPressEvent(QKeyEvent* e) +{ + switch(e->key()) + { + case Qt::Key_Delete: + case Qt::Key_Backspace: removeCurrentShape(); break; + } +} + +void Shape2DCollection::addShape(const QString& type,int x,int y) +{ + m_currentShape = createShape(type,x,y); + if ( ! m_currentShape ) return; + m_currentShape->setColor(m_borderColor); + m_currentShape->setFillColor(m_fillColor); + m_creating = true; + addShape( m_currentShape); +} + +Shape2D* Shape2DCollection::createShape(const QString& type,int x,int y) const +{ + QPointF p = m_transform.inverted().map(QPointF(x,y)); + + if (type.toLower() == "ellipse") + { + return new Shape2DEllipse(p,1.0); + } + else if (type.toLower() == "rectangle") + { + return new Shape2DRectangle(p,QSizeF(1,1)); + } + + QStringList complexType = type.split(' ',QString::SkipEmptyParts); + + if (complexType.size() < 2) return NULL; + + QString mainType = complexType[0]; + + if (mainType.toLower() == "ring") + { + Shape2D* child = createShape(complexType[1],x,y); + return new Shape2DRing(child); + } + + throw std::invalid_argument("Shape " + type + " cannot be created"); + +} + +void Shape2DCollection::startCreatingShape2D(const QString& type,const QColor& borderColor,const QColor& fillColor) +{ + m_creating = true; + m_shapeType = type; + m_borderColor = borderColor; + m_fillColor = fillColor; +} + +void Shape2DCollection::deselectAll() +{ + foreach(Shape2D* shape,m_shapes) + { + shape->edit(false); + } + m_currentShape = NULL; +} + +bool Shape2DCollection::selectAt(int x,int y) +{ + QPointF p = m_transform.inverted().map(QPointF(x,y)); + foreach(Shape2D* shape,m_shapes) + { + bool picked = shape->selectAt(p); + if (picked) + { + if (m_currentShape) + { + m_currentShape->edit(false); + } + m_currentShape = shape; + m_currentShape->edit(true); + return true; + } + } + return false; +} + +bool Shape2DCollection::isOverCurrentAt(int x,int y) +{ + if (!m_currentShape) return false; + QPointF p = m_transform.inverted().map(QPointF(x,y)); + return m_currentShape->selectAt(p); +} + +bool Shape2DCollection::selectControlPointAt(int x,int y) +{ + QPointF p = m_transform.inverted().map(QPointF(x,y)); + if (!m_currentShape) return false; + for(size_t i = 0; i < m_currentShape->getNControlPoints(); ++i) + { + QPointF cp = m_currentShape->getControlPoint(i) - p; + if (cp.manhattanLength() <= sizeCP + 2) + { + m_currentCP = i; + return true; + } + } + return false; +} + +void Shape2DCollection::removeCurrentShape() +{ + if (m_currentShape) + { + m_shapes.removeOne(m_currentShape); + } +} diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2DCollection.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2DCollection.h new file mode 100644 index 0000000000000000000000000000000000000000..de8605919f9b0032920a2ef53f4e10fe7808f234 --- /dev/null +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/Shape2DCollection.h @@ -0,0 +1,69 @@ +#ifndef MANTIDPLOT_SHAPE2DCOLLECTION_H_ +#define MANTIDPLOT_SHAPE2DCOLLECTION_H_ + +#include "Shape2D.h" + +#include <QList> +#include <QTransform> + +class QPainter; +class QMouseEvent; +class QWheelEvent; +class QKeyEvent; + +class Shape2DCollection: public QObject, public Shape2D +{ + Q_OBJECT +public: + Shape2DCollection(); + ~Shape2DCollection(); + Shape2D* clone()const{return NULL;} + void setWindow(const QRectF& rect,const QRect& viewport) const; + virtual void draw(QPainter& painter) const; + virtual void addShape(Shape2D*); + + void mousePressEvent(QMouseEvent*); + void mouseMoveEvent(QMouseEvent*); + void mouseReleaseEvent(QMouseEvent*); + void wheelEvent(QWheelEvent*); + void keyPressEvent(QKeyEvent*); + + void addShape(const QString& type,int x,int y); + void startCreatingShape2D(const QString& type,const QColor& borderColor = Qt::red,const QColor& fillColor = QColor()); + void deselectAll(); + bool selectAt(int x,int y); + void removeCurrentShape(); + +signals: + + void shapeCreated(); + +protected: + virtual void drawShape(QPainter& painter) const{} // never called + virtual void addToPath(QPainterPath& path) const{} + virtual void refit(); + virtual void adjustBoundingRect(); + + Shape2D* createShape(const QString& type,int x,int y)const; + bool selectControlPointAt(int x,int y); + bool isOverCurrentAt(int x,int y); + + QList<Shape2D*> m_shapes; + mutable QRectF m_windowRect; + mutable double m_wx,m_wy; + mutable int m_h; + mutable QTransform m_transform; + + bool m_creating; + bool m_editing; + bool m_moving; + int m_x,m_y; + QString m_shapeType; + QColor m_borderColor, m_fillColor; + Shape2D* m_currentShape; + size_t m_currentCP; + bool m_leftButtonPressed; + bool m_overridingCursor; +}; + +#endif /*MANTIDPLOT_SHAPE2DCOLLECTION_H_*/ diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.cpp b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.cpp index 072aec5e1173b348f8f889b5701441fa8059a9d0..bff059967cec32a9a7fc4a7dc318984c80120864 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.cpp +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.cpp @@ -571,3 +571,8 @@ QString UnwrappedSurface::getInfoText()const QString text = "Left mouse click and drag to zoom in.\nRight mouse click to zoom out."; return text; } + +QRectF UnwrappedSurface::getSurfaceBounds()const +{ + return QRectF(m_viewRect.left(),m_viewRect.bottom(),m_viewRect.width(),-m_viewRect.height()); +} \ No newline at end of file diff --git a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.h b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.h index cd24caa2aa99b7de1795c954343e8900a562842c..4a8c48d817b0ad143de447d5d0cfe211897d64ec 100644 --- a/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.h +++ b/Code/Mantid/MantidPlot/src/Mantid/InstrumentWidget/UnwrappedSurface.h @@ -67,6 +67,7 @@ public: void componentSelected(Mantid::Geometry::ComponentID = NULL); void getSelectedDetectors(QList<int>& dets); virtual QString getInfoText()const; + virtual QRectF getSurfaceBounds()const; protected: virtual void drawSurface(MantidGLWidget* widget,bool picking = false)const;