Exemplo n.º 1
0
  def run(self):
      self.checkRequirments()
      if not self.isSupported:
            QMessageBox.critical(QWidget(), "Version - Error","system configuration insufficient" )
            return
      # Run the plug-in when clicked on the icon
      #Create an initialization window, where is given :
      # the landscape picture
      # the DEM
      # the ortho-image if used
      # the approach chosen for georeferencing
      self.ini = Initialization_dialog()
      self.ini.setWindowModality(Qt.ApplicationModal)
      result = self.ini.exec_()
      
      # The data are all given at the beginning
      self.picture_name = self.ini.ui.lineEdit.text()
      self.DEM_name = self.ini.ui.lineEditDEM.text()
      self.useOrtho = self.ini.ui.checkBox.isChecked()
      self.pathToData = self.ini.currentPath

      # See if OK was pressed

      if result == 1:
          try:
              img = QImage(self.picture_name)
              if img.isNull():
                  raise IOError
              a = self.DEM_name.split('.')
              if len(a) == 1:
                  raise IOError
          except IOError:
              QMessageBox.warning(QWidget(), "I/O - Error","Failed to initialize picture or DEM" )
              return
          if self.DEM_name.split('.')[1] != 'tiff' and self.DEM_name.split('.')[1] != 'tif':
              QMessageBox.warning(QWidget(), "I/O - Error","Unable to load DEM. \nDEM must be in geotiff format.\nIf you use the test data set, copy it outside the plugin folder." )
              return
          #see if orthoimage is used and load it in case it's used
          if self.useOrtho:
              #get name entered
              self.ortho_name = self.ini.ui.lineEdit_2.text()
              try:
                    #load as image for checking existence
                    img = QImage(self.ortho_name)
                    if img.isNull():
                        raise IOError
                    img = None
                    #load as raster for getting bounding box
                    fileInfo = QFileInfo(self.ortho_name)
                    baseName = fileInfo.baseName()
                    rlayer = QgsRasterLayer(self.ortho_name, baseName)
                    self.ortho_box =  [int(-rlayer.extent().xMinimum()), int(rlayer.extent().yMinimum()), int(-rlayer.extent().xMaximum()), int(rlayer.extent().yMaximum())]
              except IOError:
                   QMessageBox.warning(QWidget(), "I/O - Error","Failed to load ortho-image" )
                   return
               
          #see if orthoimage is not used and initialize
          else:
              self.ortho_name = None
              self.ortho_box = None
              
          result = self.load_dem()
          #result is 1 if dem is load correctly
          if result == 1:
              # the buffer object contains arrays used for openGL
              self.load_buffer()
              
              #Check if the GCP approach is chosen
              if self.ini.ui.radioButton.isChecked():
                  self.runGCPMainWindow(False)
                  
              #Check if the virtual 3D approach is chosen
              if self.ini.ui.radioButton_2.isChecked():
                  self.virtual3DMainWindow = Virtual3DMainWindow(self.buffers,self.picture_name, self.crs, self.pathToData)
                  self.virtual3DMainWindow.show()
                  self.virtual3DMainWindow.ui.goToMonoplotterButton.clicked.connect(self.goToMonoplotter)
                  self.virtual3DMainWindow.ui.GoToGCP.clicked.connect(self.runGCPMainWindowFromVirtual3D)
Exemplo n.º 2
0
  def run(self):
      self.checkRequirments()
      if not self.isSupported:
            QMessageBox.critical(QWidget(), "Version - Error","system configuration insufficient" )
            return
      # Run the plug-in when clicked on the icon
      #Create an initialization window, where is given :
      # the landscape picture
      # the DEM
      # the ortho-image if used
      # the approach chosen for georeferencing
      self.ini = Initialization_dialog()
      self.ini.setWindowModality(Qt.ApplicationModal)
      result = self.ini.exec_()
      
      # The data are all given at the beginning
      self.picture_name = self.ini.ui.lineEdit.text()
      self.DEM_name = self.ini.ui.lineEditDEM.text()
      self.useOrtho = self.ini.ui.checkBox.isChecked()
      self.pathToData = self.ini.currentPath

      # See if OK was pressed

      if result == 1:
          try:
              img = QImage(self.picture_name)
              if img.isNull():
                  raise IOError
              a = self.DEM_name.split('.')
              if len(a) == 1:
                  raise IOError
          except IOError:
              QMessageBox.warning(QWidget(), "I/O - Error","Failed to initialize picture or DEM" )
              return
          if self.DEM_name.split('.')[1] != 'tiff' and self.DEM_name.split('.')[1] != 'tif':
              QMessageBox.warning(QWidget(), "I/O - Error","Unable to load DEM. \nDEM must be in geotiff format.\nIf you use the test data set, copy it outside the plugin folder." )
              return
          #see if orthoimage is used and load it in case it's used
          if self.useOrtho:
              #get name entered
              self.ortho_name = self.ini.ui.lineEdit_2.text()
              try:
                    #load as image for checking existence
                    img = QImage(self.ortho_name)
                    if img.isNull():
                        raise IOError
                    img = None
                    #load as raster for getting bounding box
                    fileInfo = QFileInfo(self.ortho_name)
                    baseName = fileInfo.baseName()
                    rlayer = QgsRasterLayer(self.ortho_name, baseName)
                    self.ortho_box =  [int(-rlayer.extent().xMinimum()), int(rlayer.extent().yMinimum()), int(-rlayer.extent().xMaximum()), int(rlayer.extent().yMaximum())]
              except IOError:
                   QMessageBox.warning(QWidget(), "I/O - Error","Failed to load ortho-image" )
                   return
               
          #see if orthoimage is not used and initialize
          else:
              self.ortho_name = None
              self.ortho_box = None
              
          result = self.load_dem()
          #result is 1 if dem is load correctly
          if result == 1:
              # the buffer object contains arrays used for openGL
              self.load_buffer()
              
              #Check if the GCP approach is chosen
              if self.ini.ui.radioButton.isChecked():
                  self.runGCPMainWindow(False)
                  
              #Check if the virtual 3D approach is chosen
              if self.ini.ui.radioButton_2.isChecked():
                  self.virtual3DMainWindow = Virtual3DMainWindow(self.buffers,self.picture_name, self.crs, self.pathToData)
                  self.virtual3DMainWindow.show()
                  self.virtual3DMainWindow.ui.goToMonoplotterButton.clicked.connect(self.goToMonoplotter)
                  self.virtual3DMainWindow.ui.GoToGCP.clicked.connect(self.runGCPMainWindowFromVirtual3D)
Exemplo n.º 3
0
class Pic2Map:
  def __init__(self, iface):
    # save reference to the QGIS interface
    self.iface = iface
    self.canvas = self.iface.mapCanvas()
    
    self.u = 0
    self.v = 0
          
  def checkRequirments(self):
      #Check the openGL version
      ##ex = CheckVersion()
      ##ex.updateGL()
      ##self.isSupported = ex.isSupported
      ex = CheckVersion()
      ex.updateGL()
      # Opengl 3.0 is required for Framebuffer
      self.isFrameBufferSupported =ex.isSupported
    
      # Until now, no hard or software, apart from QGis 2.0 version, has showed problem with the plugin
      self.isSupported = 1
                            
  def initGui(self):

    # create action that will start plugin configuration
    self.action = QAction(QIcon(":/plugins/Pic2Map/icon.png"), "Pic2Map",self.iface.mainWindow())
    QObject.connect(self.action, SIGNAL("triggered()"), self.run)

    # add toolbar button and menu item
    self.iface.addToolBarIcon(self.action)
    self.iface.addPluginToMenu("&Pic2Map", self.action)

    
  def unload(self):
    # remove the plugin menu item and icon
    self.iface.removePluginMenu("&Pic2Map",self.action)
    self.iface.removeToolBarIcon(self.action)
    # disconnect form signal of the canvas

  def run(self):
      self.checkRequirments()
      if not self.isSupported:
            QMessageBox.critical(QWidget(), "Version - Error","system configuration insufficient" )
            return
      # Run the plug-in when clicked on the icon
      #Create an initialization window, where is given :
      # the landscape picture
      # the DEM
      # the ortho-image if used
      # the approach chosen for georeferencing
      self.ini = Initialization_dialog()
      self.ini.setWindowModality(Qt.ApplicationModal)
      result = self.ini.exec_()
      
      # The data are all given at the beginning
      self.picture_name = self.ini.ui.lineEdit.text()
      self.DEM_name = self.ini.ui.lineEditDEM.text()
      self.useOrtho = self.ini.ui.checkBox.isChecked()
      self.pathToData = self.ini.currentPath

      # See if OK was pressed

      if result == 1:
          try:
              img = QImage(self.picture_name)
              if img.isNull():
                  raise IOError
              a = self.DEM_name.split('.')
              if len(a) == 1:
                  raise IOError
          except IOError:
              QMessageBox.warning(QWidget(), "I/O - Error","Failed to initialize picture or DEM" )
              return
          if self.DEM_name.split('.')[1] != 'tiff' and self.DEM_name.split('.')[1] != 'tif':
              QMessageBox.warning(QWidget(), "I/O - Error","Unable to load DEM. \nDEM must be in geotiff format.\nIf you use the test data set, copy it outside the plugin folder." )
              return
          #see if orthoimage is used and load it in case it's used
          if self.useOrtho:
              #get name entered
              self.ortho_name = self.ini.ui.lineEdit_2.text()
              try:
                    #load as image for checking existence
                    img = QImage(self.ortho_name)
                    if img.isNull():
                        raise IOError
                    img = None
                    #load as raster for getting bounding box
                    fileInfo = QFileInfo(self.ortho_name)
                    baseName = fileInfo.baseName()
                    rlayer = QgsRasterLayer(self.ortho_name, baseName)
                    self.ortho_box =  [int(-rlayer.extent().xMinimum()), int(rlayer.extent().yMinimum()), int(-rlayer.extent().xMaximum()), int(rlayer.extent().yMaximum())]
              except IOError:
                   QMessageBox.warning(QWidget(), "I/O - Error","Failed to load ortho-image" )
                   return
               
          #see if orthoimage is not used and initialize
          else:
              self.ortho_name = None
              self.ortho_box = None
              
          result = self.load_dem()
          #result is 1 if dem is load correctly
          if result == 1:
              # the buffer object contains arrays used for openGL
              self.load_buffer()
              
              #Check if the GCP approach is chosen
              if self.ini.ui.radioButton.isChecked():
                  self.runGCPMainWindow(False)
                  
              #Check if the virtual 3D approach is chosen
              if self.ini.ui.radioButton_2.isChecked():
                  self.virtual3DMainWindow = Virtual3DMainWindow(self.buffers,self.picture_name, self.crs, self.pathToData)
                  self.virtual3DMainWindow.show()
                  self.virtual3DMainWindow.ui.goToMonoplotterButton.clicked.connect(self.goToMonoplotter)
                  self.virtual3DMainWindow.ui.GoToGCP.clicked.connect(self.runGCPMainWindowFromVirtual3D)

    
  
  def openOrthoWidget(self):
        # This function is called from the monoplotter for ortho-rectification.
        # It is located here because some strange behavior were observed 
        # probably because the synchronous work on 2 opengl window. 
        # They may be a more elegant solution, but it works like this...
        
        # Get the view parameters from the monoplotter
        modelview = self.monoplotter.qgl_window.modelview
        projection = self.monoplotter.qgl_window.projection
        viewport = self.monoplotter.qgl_window.viewport
        texture = self.monoplotter.texture
        
        # Create the window for ortho-rectification
        self.drappingInstance = drappingMain(self.buffers, self.picture_name,
                                          modelview,
                                          projection,
                                          viewport,
                                          texture,
                                          self.crs,
                                          self.DEM_name,
                                          self.isFrameBufferSupported)
        self.drappingInstance.setWindowModality(Qt.ApplicationModal)
        self.drappingInstance.show()
        
  def goToMonoplotter(self):
      # This function is called from both approach (CCP and virtual 3D)
      # It close the pose estimation window (GetGCPMainWindow or Virtual3DMainWindow)
      # and launch the monoplotter. It is not possible anymore to change the position of the view
      
      # Check if the virtual 3D approach has been used
      if hasattr(self, 'virtual3DMainWindow'):
          self.virtual3DMainWindow.close()
          self.ParamPose = self.virtual3DMainWindow.ParamPose
          
      #Check if the GCP approach has been used
      elif hasattr(self, 'gcpMainWindow'):
          self.gcpMainWindow.goToMonoplot = True

          self.gcpMainWindow.close()
          self.ParamPose = [self.gcpMainWindow.pos, self.gcpMainWindow.lookat, self.gcpMainWindow.FOV, self.gcpMainWindow.roll]
      else:
          raise  IOError
          
      # launch monoplotter
      self.monoplotter = MonoplotterMainWindow(self.iface,
                                               self.buffers,
                                               self.picture_name,
                                               self.ParamPose,
                                               self.dem_box,
                                               self.cLayer,
                                               self.pathToData,
                                               self.crs,
                                               self.DEM_name,
                                               self.isFrameBufferSupported)
      self.monoplotter.show()
      self.monoplotter.qgl_window.updateGL()
      self.monoplotter.openOrtho.connect(self.openOrthoWidget)
      self.monoplotter.qgl_window.blow.connect(self.clickOnMonoplotter)
      self.monoplotter.ui.measureButton.clicked.connect(self.activateMeasurement)
      self.monoplotter.clearMapTool.connect(self.clearMapTool)
      self.activeFlackOnMonoplotter()

  def drawPinkCross(self, pos):
        if hasattr(self, 'pinkCross'):
            self.canvas.scene().removeItem(self.pinkCross)
        if pos[2] == 1:
            self.pinkCross = QgsVertexMarker(self.canvas)
            posi = QgsPoint(pos[0],pos[1])
            self.pinkCross.setCenter(posi)
            self.pinkCross.setColor(QColor(255, 122, 255))
            self.pinkCross.setIconSize(10)
            self.pinkCross.setIconType(QgsVertexMarker.ICON_CROSS)
            self.pinkCross.setPenWidth(10)
      

  def clearMapTool(self):
        #Call when GCP window is closed
        #Call when the tool "Measure 3D" from monoplotter is activated
        self.cTool = self.canvas.mapTool()
        self.canvas.unsetMapTool(self.cTool)
        if hasattr(self, 'monoplotter'):
            if hasattr(self.monoplotter.qgl_window, 'lineEditBuffer'):                     
                      self.monoplotter.qgl_window.lineEditBuffer = []
                      self.monoplotter.qgl_window.updateGL()

  def runGCPMainWindowFromVirtual3D(self):
        self.virtual3DMainWindow.close()
        self.ParamPose = self.virtual3DMainWindow.ParamPose
        self.runGCPMainWindow(True)
        
  def runGCPMainWindow(self, fromVirtual3D):
        # this QGIS tool emits as QgsPoint after each click on the map canvas
        self.clickTool = QgsMapToolEmitPoint(self.canvas)
        self.canvas.setMapTool(self.clickTool)
        # create main window
        self.gcpMainWindow = GetGCPMainWindow(self.iface, self.buffers, self.picture_name,self.pathToData,self.isFrameBufferSupported, self.crs)
        # show the Main Window
        self.gcpMainWindow.show()
        # load the picture in the central widget
        self.load_picture(self.picture_name)
        # get DEM reference layer
        self.gcpMainWindow.cLayer = self.cLayer
        # connect events between getGCPmainWindow and Qgis canvas
        self.gcpMainWindow.ui.tableView.selectionModel().currentRowChanged.connect(self.resetClickTool) 
        self.gcpMainWindow.resetToolSignal.connect(self.resetClickTool)
        self.gcpMainWindow.setCanvasExtentSignal.connect(self.setCanvasExtent)
        self.clickTool.canvasClicked.connect(self.newCanvasGCP)
        self.gcpMainWindow.GoToMonoplotterButton.triggered.connect(self.goToMonoplotter)
        self.gcpMainWindow.clearMapTool2.connect(self.clearMapTool)
        
        if fromVirtual3D:
            self.gcpMainWindow.pos = self.ParamPose[0]
            self.gcpMainWindow.lookat = self.ParamPose[1]
            self.gcpMainWindow.FOV = self.ParamPose[2]
            self.gcpMainWindow.roll = self.ParamPose[3] 
        
  def load_dem(self):
      try:
          #load dem for info and extent
          fileName = self.DEM_name
          fileInfo = QFileInfo(fileName)
          baseName = fileInfo.baseName()
          rlayer2 = QgsRasterLayer(fileName, baseName)
          self.dem_box = rlayer2.extent()
         
          # Get coordinate system
          self.crs = rlayer2.crs()
          # check if the map units are meter
          if self.crs.mapUnits() != 0:
              raise CRSERROR
          self.iface.addRasterLayer(self.DEM_name)
          
          if not rlayer2.isValid():
              QMessageBox.warning(QWidget(), "IO - Error",
                "Raster failed to load")
              return 0
          # DEM is set such that altitude request will be done on it
          self.cLayer = rlayer2
          return 1
      except CRSERROR:
          QMessageBox.warning(QWidget(), "CRS - Error",
                "DEM must be in projected coordinates, meter units")
          return 0

  def load_picture(self, picName):
      ##s.setValue( "/Projections/defaultBehaviour", "UseProject" )
      fileName = picName
      fileInfo = QFileInfo(fileName)
      baseName = fileInfo.baseName()
      rlayer = QgsRasterLayer(fileName, baseName)
      resolution = QDesktopWidget().screenGeometry()
      size = [0,0]
      size[1] = resolution.height()/2
      size[0] = int(float(float(rlayer.width())/float(rlayer.height()))*size[1])

      if not rlayer.isValid():
            QMessageBox.warning(QWidget(), "IO - Error",
                "Picture failed to load!")
            return
      # Load the image inside the graphicsview
      self.gcpMainWindow.setImage(fileName, size)
      # Set the zoom such the picture is almost in full view in the height directon
      factor = float(self.gcpMainWindow.ui.graphicsView.size().height())/rlayer.height()
      self.gcpMainWindow.ui.graphicsView.scale(factor, factor)
      
  def resetClickTool(self):
      # This function is used for GCP digitalization.
      # It resets the click tool for getting altitude on the DEm layer (self.clayer)
      self.canvas.setMapTool(self.clickTool)
      QObject.connect(self.clickTool, SIGNAL("canvasClicked(const QgsPoint &, Qt::MouseButton)"),self.newCanvasGCP)
      
  def newCanvasGCP(self,point):
      # This function is used for GCP digitalization.
      # a new GCP is created on canvas
      self.gcpMainWindow.selectCanvasPoint(point)
      self.gcpMainWindow.refreshCanvasGCP()
        
  def load_buffer(self):
        # This function is used at initialization.
        # Arrays are created here for openGL. The normal calculation can take some time (ca. 1sec for 100'000 pixels)
        self.buffers = Buffers(self.DEM_name, self.dem_box, self.useOrtho, self.ortho_name, self.ortho_box)
        self.buffers.getBuffer()
        self.buffers.close()
        
  def clickOnMonoplotter(self,customEvent):
      # This function is called when digitalization or measurement on monoplotter
      x = customEvent[0]
      y = customEvent[1]
      button = customEvent[2]
      self.monoplotter.qgl_window.notUpdate = False
      self.cTool = self.canvas.mapTool()
      # check if a layer is edited
      if self.cTool != None: 
          currentLayer = self.iface.activeLayer()   
          if currentLayer.type() == 0 and currentLayer.isEditable() and currentLayer.geometryType() < 3:
              if self.cTool.isEditTool():
                  if self.iface.actionAddFeature().isEnabled():
                      u,v = self.WorldToPixelOfCanvasCoordinates(x,y)
                      p = QPoint()
                      p.setX(u)
                      p.setY(v)
                      event = QMouseEvent(QEvent.MouseButtonRelease,p,button,button,Qt.NoModifier)
                      self.cTool.canvasReleaseEvent(event)
                  # check if the edited layer is a point layer
                  if currentLayer.geometryType() == 0:
                      # Since the feature has been added in the layer, it is sufficient to refresh
                      #the monoplotter for displaying the new point
                      self.monoplotter.refreshLayers(boolSymbology = True)
                  # check if the edited layer is a line layer    
                  if currentLayer.geometryType() == 1:
                      # a new point is added if the left button is clicked
                      if button == Qt.LeftButton:
                          self.monoplotter.qgl_window.lineEditBufferAppend()
                          self.monoplotter.qgl_window.updateGL()
                      # The line edition ends if the right button is clicked
                      elif button == Qt.RightButton:
                          self.monoplotter.qgl_window.lineEditBuffer = []
                          self.monoplotter.refreshLayers(boolSymbology = False)
                          
                          
                  if currentLayer.geometryType() == 2:
                      # a new point is added if the left button is clicked
                      if button == Qt.LeftButton:
                          self.monoplotter.qgl_window.lineEditBufferAppend()
                          self.monoplotter.qgl_window.updateGL()
                      # The line edition ends if the right button is clicked
                      elif button == Qt.RightButton:
                          self.monoplotter.qgl_window.lineEditBuffer = []
                          self.monoplotter.refreshLayers(boolSymbology = False)
          # check if the measuring tool is being used             
          if self.cTool.action() == self.iface.actionMeasure():
              # measurement is done if left button is clicked
              if button == Qt.LeftButton:
                  #self.cTool.activate()
                  # get the position in canvas coordinates
                  u_previous = self.u
                  v_previous = self.v
                  self.u,self.v = self.WorldToPixelOfCanvasCoordinates(x,y)
    
                  # create a mouse click at the projected location on canvas
    
                  # create a line on the monoplotter
                  
                  if u_previous != self.u or v_previous != self.v:
                     p = QPoint()
                     p.setX(self.u)
                     p.setY(self.v)
                     event = QMouseEvent(QEvent.MouseButtonRelease,p,button,button,Qt.NoModifier)
                     self.cTool.canvasReleaseEvent(event)
                     self.monoplotter.qgl_window.lineEditBufferAppend()
                     self.monoplotter.qgl_window.updateGL()
                     self.cTool.activate()
                     
                  # update window
              # the line is cleared if the right button is clicked
              elif button == Qt.RightButton:
                  self.monoplotter.qgl_window.lineEditBuffer = []
                  self.monoplotter.qgl_window.updateGL()
      else:
         if self.monoplotter.isMeasuring3D:
             if button == Qt.LeftButton:
                 self.monoplotter.qgl_window.lineEditBufferAppend()
                 self.monoplotter.qgl_window.updateGL()
                 self.monoplotter.dlgMeasure3D.addPoint(x,y)
             if button == Qt.RightButton:
                 self.monoplotter.qgl_window.lineEditBuffer = []
                 self.monoplotter.qgl_window.updateGL()
                 self.monoplotter.dlgMeasure3D.removePath()
  
  def activateMeasurement(self):
      # This function is used for calling the measuring tool from the monoplotter
      self.iface.actionMeasure().trigger()
      self.monoplotter.stopMeasure3D()
      
  def disactiveFlackOnMonoplotter(self):
      self.canvas.mapTool().canvasMoveEvent = None
      self.canvas.mapTool().deactivate()
      self.canvas.unsetMapTool(self.canvas.mapTool())


  def activeFlackOnMonoplotter(self):
      self.clickTool2 = QgsMapToolEmitPoint(self.canvas)
      self.canvas.setMapTool(self.clickTool2)
      self.canvas.mapTool().canvasMoveEvent = self.test3
      self.monoplotter.qgl_window.pinkCrossSignal.connect(self.drawPinkCross)
      self.monoplotter.closingMonoplot.connect(self.disactiveFlackOnMonoplotter)
    
  def test3(self, ev):
      x,y = self.CanvastoWorldCoordinates(ev.x(), ev.y())
      self.monoplotter.preparePurpleCross(x,y)


  def WorldToPixelOfCanvasCoordinates(self,x,y):
      # This function is called in "clickOnMonoplotter".
      # It convert CSR coordinates (world) into the screen coordinates for a given point in the canvas.
      canvasExtent =  self.canvas.extent()
      canvasBox =  [canvasExtent.xMinimum(), canvasExtent.yMinimum(), canvasExtent.xMaximum(), canvasExtent.yMaximum()]
      UPP = self.canvas.mapUnitsPerPixel()
      u = int((x-canvasBox[0])/UPP)
      v = int((canvasBox[3]-y)/UPP)
      return (u,v)
  
  def CanvastoWorldCoordinates(self,u,v):
      # This function is called in "clickOnMonoplotter".
      # It convert CSR coordinates (world) into the screen coordinates for a given point in the canvas.
      canvasExtent =  self.canvas.extent()
      canvasBox =  [canvasExtent.xMinimum(), canvasExtent.yMinimum(), canvasExtent.xMaximum(), canvasExtent.yMaximum()]
      UPP = self.canvas.mapUnitsPerPixel()
      x = int(u*UPP+canvasBox[0])
      y = int(-v*UPP+canvasBox[3])
      return (x,y)
  
  def setCanvasExtent(self, pos):
      # This function is used for GCP digitalization.
      # It zoom on the canvas depending on which GCp is selected. 
      # The zoom extent is set with referenced to the size of the DEM
      offset = (self.dem_box.xMinimum()-self.dem_box.xMaximum())/50
      zoomRectangle = QgsRectangle(pos[0]-offset, pos[1]-offset,pos[0]+offset,pos[1]+offset)
      self.canvas.setExtent(zoomRectangle)
      self.canvas.refresh()
Exemplo n.º 4
0
class Pic2Map:
  def __init__(self, iface):
    # save reference to the QGIS interface
    self.iface = iface
    self.canvas = self.iface.mapCanvas()
    
    self.u = 0
    self.v = 0
          
  def checkRequirments(self):
      #Check the openGL version
      ##ex = CheckVersion()
      ##ex.updateGL()
      ##self.isSupported = ex.isSupported
      ex = CheckVersion()
      ex.updateGL()
      # Opengl 3.0 is required for Framebuffer
      self.isFrameBufferSupported =ex.isSupported
    
      # Until now, no hard or software, apart from QGis 2.0 version, has showed problem with the plugin
      self.isSupported = 1
                            
  def initGui(self):

    # create action that will start plugin configuration
    self.action = QAction(QIcon(":/plugins/Pic2Map/icon.png"), "Pic2Map",self.iface.mainWindow())
    QObject.connect(self.action, SIGNAL("triggered()"), self.run)

    # add toolbar button and menu item
    self.iface.addToolBarIcon(self.action)
    self.iface.addPluginToMenu("&Pic2Map", self.action)

    
  def unload(self):
    # remove the plugin menu item and icon
    self.iface.removePluginMenu("&Pic2Map",self.action)
    self.iface.removeToolBarIcon(self.action)
    # disconnect form signal of the canvas

  def run(self):
      self.checkRequirments()
      if not self.isSupported:
            QMessageBox.critical(QWidget(), "Version - Error","system configuration insufficient" )
            return
      # Run the plug-in when clicked on the icon
      #Create an initialization window, where is given :
      # the landscape picture
      # the DEM
      # the ortho-image if used
      # the approach chosen for georeferencing
      self.ini = Initialization_dialog()
      self.ini.setWindowModality(Qt.ApplicationModal)
      result = self.ini.exec_()
      
      # The data are all given at the beginning
      self.picture_name = self.ini.ui.lineEdit.text()
      self.DEM_name = self.ini.ui.lineEditDEM.text()
      self.useOrtho = self.ini.ui.checkBox.isChecked()
      self.pathToData = self.ini.currentPath

      # See if OK was pressed

      if result == 1:
          try:
              img = QImage(self.picture_name)
              if img.isNull():
                  raise IOError
              a = self.DEM_name.split('.')
              if len(a) == 1:
                  raise IOError
          except IOError:
              QMessageBox.warning(QWidget(), "I/O - Error","Failed to initialize picture or DEM" )
              return
          if self.DEM_name.split('.')[1] != 'tiff' and self.DEM_name.split('.')[1] != 'tif':
              QMessageBox.warning(QWidget(), "I/O - Error","Unable to load DEM. \nDEM must be in geotiff format.\nIf you use the test data set, copy it outside the plugin folder." )
              return
          #see if orthoimage is used and load it in case it's used
          if self.useOrtho:
              #get name entered
              self.ortho_name = self.ini.ui.lineEdit_2.text()
              try:
                    #load as image for checking existence
                    img = QImage(self.ortho_name)
                    if img.isNull():
                        raise IOError
                    img = None
                    #load as raster for getting bounding box
                    fileInfo = QFileInfo(self.ortho_name)
                    baseName = fileInfo.baseName()
                    rlayer = QgsRasterLayer(self.ortho_name, baseName)
                    self.ortho_box =  [int(-rlayer.extent().xMinimum()), int(rlayer.extent().yMinimum()), int(-rlayer.extent().xMaximum()), int(rlayer.extent().yMaximum())]
              except IOError:
                   QMessageBox.warning(QWidget(), "I/O - Error","Failed to load ortho-image" )
                   return
               
          #see if orthoimage is not used and initialize
          else:
              self.ortho_name = None
              self.ortho_box = None
              
          result = self.load_dem()
          #result is 1 if dem is load correctly
          if result == 1:
              # the buffer object contains arrays used for openGL
              self.load_buffer()
              
              #Check if the GCP approach is chosen
              if self.ini.ui.radioButton.isChecked():
                  self.runGCPMainWindow(False)
                  
              #Check if the virtual 3D approach is chosen
              if self.ini.ui.radioButton_2.isChecked():
                  self.virtual3DMainWindow = Virtual3DMainWindow(self.buffers,self.picture_name, self.crs, self.pathToData)
                  self.virtual3DMainWindow.show()
                  self.virtual3DMainWindow.ui.goToMonoplotterButton.clicked.connect(self.goToMonoplotter)
                  self.virtual3DMainWindow.ui.GoToGCP.clicked.connect(self.runGCPMainWindowFromVirtual3D)

    
  
  def openOrthoWidget(self):
        # This function is called from the monoplotter for ortho-rectification.
        # It is located here because some strange behavior were observed 
        # probably because the synchronous work on 2 opengl window. 
        # They may be a more elegant solution, but it works like this...
        
        # Get the view parameters from the monoplotter
        modelview = self.monoplotter.qgl_window.modelview
        projection = self.monoplotter.qgl_window.projection
        viewport = self.monoplotter.qgl_window.viewport
        texture = self.monoplotter.texture
        
        Xmat = self.monoplotter.Xmat
        Ymat = self.monoplotter.Ymat #20150823
        
        # Create the window for ortho-rectification
        self.drappingInstance = drappingMain(self.buffers, self.picture_name,
                                          modelview,
                                          projection,
                                          viewport,
                                          texture,
                                          self.crs,
                                          self.DEM_name,
                                          self.isFrameBufferSupported,
                                          Xmat,
                                          Ymat)#20150923
        self.drappingInstance.setWindowModality(Qt.ApplicationModal)
        self.drappingInstance.show()
        
  def goToMonoplotter(self):
      # This function is called from both approach (CCP and virtual 3D)
      # It close the pose estimation window (GetGCPMainWindow or Virtual3DMainWindow)
      # and launch the monoplotter. It is not possible anymore to change the position of the view
      
      # Check if the virtual 3D approach has been used
      if hasattr(self, 'virtual3DMainWindow'):
          self.virtual3DMainWindow.close()
          self.ParamPose = self.virtual3DMainWindow.ParamPose
          
      #Check if the GCP approach has been used
      elif hasattr(self, 'gcpMainWindow'):
          self.gcpMainWindow.goToMonoplot = True

          self.gcpMainWindow.close()
          self.ParamPose = [self.gcpMainWindow.pos, self.gcpMainWindow.lookat, self.gcpMainWindow.FOV, self.gcpMainWindow.roll]
      else:
          raise  IOError
      
      # launch monoplotter
      self.monoplotter = MonoplotterMainWindow(self.iface,
                                               self.buffers,
                                               self.picture_name,
                                               self.ParamPose,
                                               self.dem_box,
                                               self.cLayer,
                                               self.pathToData,
                                               self.crs,
                                               self.DEM_name,
                                               self.isFrameBufferSupported,
                                               self.demMax, self.demMin)#
      self.monoplotter.show()
      self.monoplotter.qgl_window.updateGL()
      self.monoplotter.openOrtho.connect(self.openOrthoWidget)
      self.monoplotter.qgl_window.blow.connect(self.clickOnMonoplotter)
      self.monoplotter.ui.measureButton.clicked.connect(self.activateMeasurement)
      self.monoplotter.clearMapTool.connect(self.clearMapTool)
      self.activeFlackOnMonoplotter()

  def drawPinkCross(self, pos):
        if hasattr(self, 'pinkCross'):
            self.canvas.scene().removeItem(self.pinkCross)
        if pos[2] == 1:
            self.pinkCross = QgsVertexMarker(self.canvas)
            posi = QgsPoint(pos[0],pos[1])
            self.pinkCross.setCenter(posi)
            self.pinkCross.setColor(QColor(255, 122, 255))
            self.pinkCross.setIconSize(10)
            self.pinkCross.setIconType(QgsVertexMarker.ICON_CROSS)
            self.pinkCross.setPenWidth(10)
      

  def clearMapTool(self):
        #Call when GCP window is closed
        #Call when the tool "Measure 3D" from monoplotter is activated
        self.cTool = self.canvas.mapTool()
        self.canvas.unsetMapTool(self.cTool)
        if hasattr(self, 'monoplotter'):
            if hasattr(self.monoplotter.qgl_window, 'lineEditBuffer'):                     
                      self.monoplotter.qgl_window.lineEditBuffer = []
                      self.monoplotter.qgl_window.updateGL()

  def runGCPMainWindowFromVirtual3D(self):
        self.virtual3DMainWindow.close()
        self.ParamPose = self.virtual3DMainWindow.ParamPose
        self.runGCPMainWindow(True)
        
  def runGCPMainWindow(self, fromVirtual3D):
        # this QGIS tool emits as QgsPoint after each click on the map canvas
        self.clickTool = QgsMapToolEmitPoint(self.canvas)
        self.canvas.setMapTool(self.clickTool)
        # create main window
        self.gcpMainWindow = GetGCPMainWindow(self.iface, self.buffers, self.picture_name,self.pathToData,self.isFrameBufferSupported, self.crs)
        # show the Main Window
        self.gcpMainWindow.show()
        # load the picture in the central widget
        self.load_picture(self.picture_name)
        # get DEM reference layer
        self.gcpMainWindow.cLayer = self.cLayer
        # connect events between getGCPmainWindow and Qgis canvas
        self.gcpMainWindow.ui.tableView.selectionModel().currentRowChanged.connect(self.resetClickTool) 
        self.gcpMainWindow.resetToolSignal.connect(self.resetClickTool)
        self.gcpMainWindow.setCanvasExtentSignal.connect(self.setCanvasExtent)
        self.clickTool.canvasClicked.connect(self.newCanvasGCP)
        self.gcpMainWindow.GoToMonoplotterButton.triggered.connect(self.goToMonoplotter)
        self.gcpMainWindow.clearMapTool2.connect(self.clearMapTool)
        
        if fromVirtual3D:
            self.gcpMainWindow.pos = self.ParamPose[0]
            self.gcpMainWindow.lookat = self.ParamPose[1]
            self.gcpMainWindow.FOV = self.ParamPose[2]
            self.gcpMainWindow.roll = self.ParamPose[3] 
        
  def load_dem(self):
      try:
          #load dem for info and extent
          fileName = self.DEM_name
          fileInfo = QFileInfo(fileName)
          baseName = fileInfo.baseName()
          rlayer2 = QgsRasterLayer(fileName, baseName)
          self.dem_box = rlayer2.extent()
          
          provider = rlayer2.dataProvider()
          stats = provider.bandStatistics(1, QgsRasterBandStats.All,self.dem_box, 0)
          self.demMin = stats.minimumValue
          self.demMax = stats.maximumValue
          
         
          # Get coordinate system
          self.crs = rlayer2.crs()
         
          # check if the map units are meter
          if self.crs.mapUnits() != 0:
              raise CRSERROR
          self.iface.addRasterLayer(self.DEM_name)
          
          if not rlayer2.isValid():
              QMessageBox.warning(QWidget(), "IO - Error",
                "Raster failed to load")
              return 0
          # DEM is set such that altitude request will be done on it
          self.cLayer = rlayer2
          return 1
      except CRSERROR:
          QMessageBox.warning(QWidget(), "CRS - Error",
                "DEM must be in projected coordinates, meter units")
          return 0

  def load_picture(self, picName):
      ##s.setValue( "/Projections/defaultBehaviour", "UseProject" )
      fileName = picName
      fileInfo = QFileInfo(fileName)
      baseName = fileInfo.baseName()
      rlayer = QgsRasterLayer(fileName, baseName)
      resolution = QDesktopWidget().screenGeometry()
      size = [0,0]
      size[1] = resolution.height()/2
      size[0] = int(float(float(rlayer.width())/float(rlayer.height()))*size[1])

      if not rlayer.isValid():
            QMessageBox.warning(QWidget(), "IO - Error",
                "Picture failed to load!")
            return
      # Load the image inside the graphicsview
      self.gcpMainWindow.setImage(fileName, size)
      # Set the zoom such the picture is almost in full view in the height directon
      factor = float(self.gcpMainWindow.ui.graphicsView.size().height())/rlayer.height()
      self.gcpMainWindow.ui.graphicsView.scale(factor, factor)
      
  def resetClickTool(self):
      # This function is used for GCP digitalization.
      # It resets the click tool for getting altitude on the DEm layer (self.clayer)
      self.canvas.setMapTool(self.clickTool)
      QObject.connect(self.clickTool, SIGNAL("canvasClicked(const QgsPoint &, Qt::MouseButton)"),self.newCanvasGCP)
      
  def newCanvasGCP(self,point):
      # This function is used for GCP digitalization.
      # a new GCP is created on canvas
      self.gcpMainWindow.selectCanvasPoint(point)
      self.gcpMainWindow.refreshCanvasGCP()
        
  def load_buffer(self):
        # This function is used at initialization.
        # Arrays are created here for openGL. The normal calculation can take some time (ca. 1sec for 100'000 pixels)
        self.buffers = Buffers(self.DEM_name, self.dem_box, self.useOrtho, self.ortho_name, self.ortho_box)
        self.buffers.getBuffer()
        self.buffers.close()
        
  def clickOnMonoplotter(self,customEvent):
      # This function is called when digitalization or measurement on monoplotter
      x = customEvent[0]
      y = customEvent[1]
      button = customEvent[2]
      self.monoplotter.qgl_window.notUpdate = False
      self.cTool = self.canvas.mapTool()
      
      # check if a layer is edited
      if self.cTool != None: 
          currentLayer = self.iface.activeLayer()   
          if currentLayer.type() == 0 and currentLayer.isEditable() and currentLayer.geometryType() < 3:
              if self.cTool.isEditTool():
                  if self.iface.actionAddFeature().isEnabled():
                      u,v = self.WorldToPixelOfCanvasCoordinates(x,y)
                      p = QPoint()
                      p.setX(u)
                      p.setY(v)
                      event = QMouseEvent(QEvent.MouseButtonRelease,p,button,button,Qt.NoModifier)
                      event2 = QgsMapMouseEvent(self.canvas, event)####
                      self.cTool.canvasReleaseEvent(event2)
                  # check if the edited layer is a point layer
                  if currentLayer.geometryType() == 0:
                      # Since the feature has been added in the layer, it is sufficient to refresh
                      #the monoplotter for displaying the new point
                      self.monoplotter.refreshLayers(boolSymbology = True)#True)
                  # check if the edited layer is a line layer    
                  if currentLayer.geometryType() == 1:
                      # a new point is added if the left button is clicked
                      if button == Qt.LeftButton:
                          self.monoplotter.qgl_window.lineEditBufferAppend()
                          self.monoplotter.qgl_window.updateGL()
                      # The line edition ends if the right button is clicked
                      elif button == Qt.RightButton:
                          self.monoplotter.qgl_window.lineEditBuffer = []
                          self.monoplotter.refreshLayers(boolSymbology = False)
                          
                          
                  if currentLayer.geometryType() == 2:
                      # a new point is added if the left button is clicked
                      if button == Qt.LeftButton:
                          self.monoplotter.qgl_window.lineEditBufferAppend()
                          self.monoplotter.qgl_window.updateGL()
                      # The line edition ends if the right button is clicked
                      elif button == Qt.RightButton:
                          self.monoplotter.qgl_window.lineEditBuffer = []
                          self.monoplotter.refreshLayers(boolSymbology = False)
          # check if the measuring tool is being used             
          if self.cTool.action() == self.iface.actionMeasure():
              # measurement is done if left button is clicked
              if button == Qt.LeftButton:
                  #self.cTool.activate()
                  # get the position in canvas coordinates
                  u_previous = self.u
                  v_previous = self.v
                  self.u,self.v = self.WorldToPixelOfCanvasCoordinates(x,y)
    
                  # create a mouse click at the projected location on canvas
    
                  # create a line on the monoplotter
                  
                  if u_previous != self.u or v_previous != self.v:
                     p = QPoint()
                     p.setX(self.u)
                     p.setY(self.v)
                     event = QMouseEvent(QEvent.MouseButtonRelease,p,button,button,Qt.NoModifier)
                     event2 = QgsMapMouseEvent(self.canvas, event)
                     
                     self.cTool.canvasReleaseEvent(event2)
                     self.monoplotter.qgl_window.lineEditBufferAppend()
                     self.monoplotter.qgl_window.updateGL()
                     self.cTool.activate()
                     
                  # update window
              # the line is cleared if the right button is clicked
              elif button == Qt.RightButton:
                  self.monoplotter.qgl_window.lineEditBuffer = []
                  self.monoplotter.qgl_window.updateGL()
      else:
         if self.monoplotter.isMeasuring3D:
             if button == Qt.LeftButton:
                 self.monoplotter.qgl_window.lineEditBufferAppend()
                 self.monoplotter.qgl_window.updateGL()
                 self.monoplotter.dlgMeasure3D.addPoint(x,y)
             if button == Qt.RightButton:
                 self.monoplotter.qgl_window.lineEditBuffer = []
                 self.monoplotter.qgl_window.updateGL()
                 self.monoplotter.dlgMeasure3D.removePath()
  
  def activateMeasurement(self):
      # This function is used for calling the measuring tool from the monoplotter
      self.iface.actionMeasure().trigger()
      self.monoplotter.stopMeasure3D()
      
  def disactiveFlackOnMonoplotter(self):
      self.canvas.mapTool().canvasMoveEvent = None
      self.canvas.mapTool().deactivate()
      self.canvas.unsetMapTool(self.canvas.mapTool())


  def activeFlackOnMonoplotter(self):
      self.clickTool2 = QgsMapToolEmitPoint(self.canvas)
      self.canvas.setMapTool(self.clickTool2)
      self.canvas.mapTool().canvasMoveEvent = self.test3
      self.monoplotter.qgl_window.pinkCrossSignal.connect(self.drawPinkCross)
      self.monoplotter.closingMonoplot.connect(self.disactiveFlackOnMonoplotter)
    
  def test3(self, ev):
      x,y = self.CanvastoWorldCoordinates(ev.x(), ev.y())
      self.monoplotter.preparePurpleCross(x,y)


  def WorldToPixelOfCanvasCoordinates(self,x,y):
      # This function is called in "clickOnMonoplotter".
      # It convert CSR coordinates (world) into the screen coordinates for a given point in the canvas.
      canvasExtent =  self.canvas.extent()
      canvasBox =  [canvasExtent.xMinimum(), canvasExtent.yMinimum(), canvasExtent.xMaximum(), canvasExtent.yMaximum()]
      UPP = self.canvas.mapUnitsPerPixel()
      u = int((x-canvasBox[0])/UPP)
      v = int((canvasBox[3]-y)/UPP)
      return (u,v)
  
  def CanvastoWorldCoordinates(self,u,v):
      # This function is called in "clickOnMonoplotter".
      # It convert CSR coordinates (world) into the screen coordinates for a given point in the canvas.
      canvasExtent =  self.canvas.extent()
      canvasBox =  [canvasExtent.xMinimum(), canvasExtent.yMinimum(), canvasExtent.xMaximum(), canvasExtent.yMaximum()]
      UPP = self.canvas.mapUnitsPerPixel()
      x = int(u*UPP+canvasBox[0])
      y = int(-v*UPP+canvasBox[3])
      return (x,y)
  
  def setCanvasExtent(self, pos):
      # This function is used for GCP digitalization.
      # It zoom on the canvas depending on which GCp is selected. 
      # The zoom extent is set with referenced to the size of the DEM
      offset = (self.dem_box.xMinimum()-self.dem_box.xMaximum())/50
      zoomRectangle = QgsRectangle(pos[0]-offset, pos[1]-offset,pos[0]+offset,pos[1]+offset)
      self.canvas.setExtent(zoomRectangle)
      self.canvas.refresh()