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;