mirror of
				https://github.com/qgis/QGIS.git
				synced 2025-10-31 00:06:02 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			400 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			400 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # -*- coding: utf-8 -*-
 | |
| 
 | |
| """
 | |
| ***************************************************************************
 | |
|     ModelerGraphicItem.py
 | |
|     ---------------------
 | |
|     Date                 : August 2012
 | |
|     Copyright            : (C) 2012 by Victor Olaya
 | |
|     Email                : volayaf at gmail dot com
 | |
| ***************************************************************************
 | |
| *                                                                         *
 | |
| *   This program is free software; you can redistribute it and/or modify  *
 | |
| *   it under the terms of the GNU General Public License as published by  *
 | |
| *   the Free Software Foundation; either version 2 of the License, or     *
 | |
| *   (at your option) any later version.                                   *
 | |
| *                                                                         *
 | |
| ***************************************************************************
 | |
| """
 | |
| 
 | |
| 
 | |
| __author__ = 'Victor Olaya'
 | |
| __date__ = 'August 2012'
 | |
| __copyright__ = '(C) 2012, Victor Olaya'
 | |
| 
 | |
| # This will get replaced with a git SHA1 when you do a git archive
 | |
| 
 | |
| __revision__ = '$Format:%H$'
 | |
| 
 | |
| import os
 | |
| from PyQt4.QtCore import Qt, QPointF, QRectF
 | |
| from PyQt4.QtGui import QIcon, QGraphicsItem, QFont, QFontMetricsF, QMessageBox, QMenu, QPen, QBrush, QColor, QPolygonF
 | |
| from processing.modeler.ModelerAlgorithm import ModelerParameter, Algorithm, ModelerOutput
 | |
| from processing.modeler.ModelerParameterDefinitionDialog import ModelerParameterDefinitionDialog
 | |
| from processing.modeler.ModelerParametersDialog import ModelerParametersDialog
 | |
| 
 | |
| pluginPath = os.path.split(os.path.dirname(__file__))[0]
 | |
| 
 | |
| 
 | |
| class ModelerGraphicItem(QGraphicsItem):
 | |
| 
 | |
|     BOX_HEIGHT = 30
 | |
|     BOX_WIDTH = 200
 | |
| 
 | |
|     def __init__(self, element, model):
 | |
|         super(ModelerGraphicItem, self).__init__(None, None)
 | |
|         self.model = model
 | |
|         self.element = element
 | |
|         if isinstance(element, ModelerParameter):
 | |
|             icon = QIcon(os.path.join(pluginPath, 'images', 'input.png'))
 | |
|             self.pixmap = icon.pixmap(20, 20, state=QIcon.On)
 | |
|             self.text = element.param.description
 | |
|         elif isinstance(element, ModelerOutput):
 | |
|             # Output name
 | |
|             icon = QIcon(os.path.join(pluginPath, 'images', 'output.png'))
 | |
|             self.pixmap = icon.pixmap(20, 20, state=QIcon.On)
 | |
|             self.text = element.description
 | |
|         else:
 | |
|             self.text = element.description
 | |
|             self.pixmap = element.algorithm.getIcon().pixmap(15, 15)
 | |
|         self.arrows = []
 | |
|         self.setFlag(QGraphicsItem.ItemIsMovable, True)
 | |
|         self.setFlag(QGraphicsItem.ItemIsSelectable, True)
 | |
|         self.setFlag(QGraphicsItem.ItemSendsGeometryChanges, True)
 | |
|         self.setZValue(1000)
 | |
| 
 | |
|         if not isinstance(element, ModelerOutput):
 | |
|             icon = QIcon(os.path.join(pluginPath, 'images', 'edit.png'))
 | |
|             pt = QPointF(ModelerGraphicItem.BOX_WIDTH / 2
 | |
|                          - FlatButtonGraphicItem.WIDTH / 2,
 | |
|                          ModelerGraphicItem.BOX_HEIGHT / 2
 | |
|                          - FlatButtonGraphicItem.HEIGHT / 2 + 1)
 | |
|             self.editButton = FlatButtonGraphicItem(icon, pt, self.editElement)
 | |
|             self.editButton.setParentItem(self)
 | |
|             icon = QIcon(os.path.join(pluginPath, 'images', 'delete.png'))
 | |
|             pt = QPointF(ModelerGraphicItem.BOX_WIDTH / 2
 | |
|                          - FlatButtonGraphicItem.WIDTH / 2,
 | |
|                          - ModelerGraphicItem.BOX_HEIGHT / 2
 | |
|                          + FlatButtonGraphicItem.HEIGHT / 2 + 1)
 | |
|             self.deleteButton = FlatButtonGraphicItem(icon, pt,
 | |
|                                                       self.removeElement)
 | |
|             self.deleteButton.setParentItem(self)
 | |
| 
 | |
|         if isinstance(element, Algorithm):
 | |
|             alg = element.algorithm
 | |
|             if alg.parameters:
 | |
|                 pt = self.getLinkPointForParameter(-1)
 | |
|                 pt = QPointF(0, pt.y() + 2)
 | |
|                 self.inButton = FoldButtonGraphicItem(pt, self.foldInput, self.element.paramsFolded)
 | |
|                 self.inButton.setParentItem(self)
 | |
|             if alg.outputs:
 | |
|                 pt = self.getLinkPointForOutput(-1)
 | |
|                 pt = QPointF(0, pt.y() + 2)
 | |
|                 self.outButton = FoldButtonGraphicItem(pt, self.foldOutput, self.element.outputsFolded)
 | |
|                 self.outButton.setParentItem(self)
 | |
| 
 | |
|     def foldInput(self, folded):
 | |
|         self.element.paramsFolded = folded
 | |
|         self.prepareGeometryChange()
 | |
|         if self.element.algorithm.outputs:
 | |
|             pt = self.getLinkPointForOutput(-1)
 | |
|             pt = QPointF(0, pt.y())
 | |
|             self.outButton.position = pt
 | |
|         for arrow in self.arrows:
 | |
|             arrow.updatePath()
 | |
|         self.update()
 | |
| 
 | |
|     def foldOutput(self, folded):
 | |
|         self.element.outputsFolded = folded
 | |
|         self.prepareGeometryChange()
 | |
|         for arrow in self.arrows:
 | |
|             arrow.updatePath()
 | |
|         self.update()
 | |
| 
 | |
|     def addArrow(self, arrow):
 | |
|         self.arrows.append(arrow)
 | |
| 
 | |
|     def boundingRect(self):
 | |
|         font = QFont('Verdana', 8)
 | |
|         fm = QFontMetricsF(font)
 | |
|         unfolded = isinstance(self.element, Algorithm) and not self.element.paramsFolded
 | |
|         numParams = len(self.element.algorithm.parameters) if unfolded else 0
 | |
|         unfolded = isinstance(self.element, Algorithm) and not self.element.outputsFolded
 | |
|         numOutputs = len(self.element.algorithm.outputs) if unfolded else 0
 | |
| 
 | |
|         hUp = fm.height() * 1.2 * (numParams + 2)
 | |
|         hDown = fm.height() * 1.2 * (numOutputs + 2)
 | |
|         rect = QRectF(-(ModelerGraphicItem.BOX_WIDTH + 2) / 2,
 | |
|                       -(ModelerGraphicItem.BOX_HEIGHT + 2) / 2 - hUp,
 | |
|                       ModelerGraphicItem.BOX_WIDTH + 2,
 | |
|                       ModelerGraphicItem.BOX_HEIGHT + hDown + hUp)
 | |
|         return rect
 | |
| 
 | |
|     def mouseDoubleClickEvent(self, event):
 | |
|         pass
 | |
|         #self.editElement()
 | |
| 
 | |
|     def contextMenuEvent(self, event):
 | |
|         if isinstance(self.element, ModelerOutput):
 | |
|             return
 | |
|         popupmenu = QMenu()
 | |
|         removeAction = popupmenu.addAction('Remove')
 | |
|         removeAction.triggered.connect(self.removeElement)
 | |
|         editAction = popupmenu.addAction('Edit')
 | |
|         editAction.triggered.connect(self.editElement)
 | |
|         if isinstance(self.element, Algorithm):
 | |
|             if not self.element.active:
 | |
|                 removeAction = popupmenu.addAction('Activate')
 | |
|                 removeAction.triggered.connect(self.activateAlgorithm)
 | |
|             else:
 | |
|                 deactivateAction = popupmenu.addAction('Deactivate')
 | |
|                 deactivateAction.triggered.connect(self.deactivateAlgorithm)
 | |
|         popupmenu.exec_(event.screenPos())
 | |
| 
 | |
|     def deactivateAlgorithm(self):
 | |
|         self.model.deactivateAlgorithm(self.element.name)
 | |
|         self.model.updateModelerView()
 | |
| 
 | |
|     def activateAlgorithm(self):
 | |
|         if self.model.activateAlgorithm(self.element.name):
 | |
|             self.model.updateModelerView()
 | |
|         else:
 | |
|             QMessageBox.warning(None, 'Could not activate Algorithm',
 | |
|                                 'The selected algorithm depends on other currently non-active algorithms.\n'
 | |
|                                 'Activate them them before trying to activate it.')
 | |
| 
 | |
|     def editElement(self):
 | |
|         if isinstance(self.element, ModelerParameter):
 | |
|             dlg = ModelerParameterDefinitionDialog(self.model,
 | |
|                                                    param=self.element.param)
 | |
|             dlg.exec_()
 | |
|             if dlg.param is not None:
 | |
|                 self.model.updateParameter(dlg.param)
 | |
|                 self.element.param = dlg.param
 | |
|                 self.text = dlg.param.description
 | |
|                 self.update()
 | |
|         elif isinstance(self.element, Algorithm):
 | |
|             dlg = self.element.algorithm.getCustomModelerParametersDialog(self.model, self.element.name)
 | |
|             if not dlg:
 | |
|                 dlg = ModelerParametersDialog(self.element.algorithm, self.model, self.element.name)
 | |
|             dlg.exec_()
 | |
|             if dlg.alg is not None:
 | |
|                 dlg.alg.name = self.element.name
 | |
|                 self.model.updateAlgorithm(dlg.alg)
 | |
|                 self.model.updateModelerView()
 | |
| 
 | |
|     def removeElement(self):
 | |
|         if isinstance(self.element, ModelerParameter):
 | |
|             if not self.model.removeParameter(self.element.param.name):
 | |
|                 QMessageBox.warning(None, 'Could not remove element',
 | |
|                                     'Other elements depend on the selected one.\n'
 | |
|                                     'Remove them before trying to remove it.')
 | |
|             else:
 | |
|                 self.model.updateModelerView()
 | |
|         elif isinstance(self.element, Algorithm):
 | |
|             if not self.model.removeAlgorithm(self.element.name):
 | |
|                 QMessageBox.warning(None, 'Could not remove element',
 | |
|                                     'Other elements depend on the selected one.\n'
 | |
|                                     'Remove them before trying to remove it.')
 | |
|             else:
 | |
|                 self.model.updateModelerView()
 | |
| 
 | |
|     def getAdjustedText(self, text):
 | |
|         font = QFont('Verdana', 8)
 | |
|         fm = QFontMetricsF(font)
 | |
|         w = fm.width(text)
 | |
|         if w < self.BOX_WIDTH - 25 - FlatButtonGraphicItem.WIDTH:
 | |
|             return text
 | |
| 
 | |
|         text = text[0:-3] + '...'
 | |
|         w = fm.width(text)
 | |
|         while w > self.BOX_WIDTH - 25 - FlatButtonGraphicItem.WIDTH:
 | |
|             text = text[0:-4] + '...'
 | |
|             w = fm.width(text)
 | |
|         return text
 | |
| 
 | |
|     def paint(self, painter, option, widget=None):
 | |
|         rect = QRectF(-(ModelerGraphicItem.BOX_WIDTH + 2) / 2.0,
 | |
|                       -(ModelerGraphicItem.BOX_HEIGHT + 2) / 2.0,
 | |
|                       ModelerGraphicItem.BOX_WIDTH + 2,
 | |
|                       ModelerGraphicItem.BOX_HEIGHT + 2)
 | |
|         painter.setPen(QPen(Qt.gray, 1))
 | |
|         color = QColor(125, 232, 232)
 | |
|         if isinstance(self.element, ModelerParameter):
 | |
|             color = QColor(179, 179, 255)
 | |
|         elif isinstance(self.element, Algorithm):
 | |
|             color = Qt.white
 | |
|         painter.setBrush(QBrush(color, Qt.SolidPattern))
 | |
|         painter.drawRect(rect)
 | |
|         font = QFont('Verdana', 8)
 | |
|         painter.setFont(font)
 | |
|         painter.setPen(QPen(Qt.black))
 | |
|         text = self.getAdjustedText(self.text)
 | |
|         if isinstance(self.element, Algorithm) and not self.element.active:
 | |
|             painter.setPen(QPen(Qt.gray))
 | |
|             text = text + "\n(deactivated)"
 | |
|         elif self.isSelected():
 | |
|             painter.setPen(QPen(Qt.blue))
 | |
|         fm = QFontMetricsF(font)
 | |
|         text = self.getAdjustedText(self.text)
 | |
|         h = fm.height()
 | |
|         pt = QPointF(-ModelerGraphicItem.BOX_WIDTH / 2 + 25, h / 2.0)
 | |
|         painter.drawText(pt, text)
 | |
|         painter.setPen(QPen(Qt.black))
 | |
|         if isinstance(self.element, Algorithm):
 | |
|             h = -(fm.height() * 1.2)
 | |
|             h = h - ModelerGraphicItem.BOX_HEIGHT / 2.0 + 5
 | |
|             pt = QPointF(-ModelerGraphicItem.BOX_WIDTH / 2 + 25, h)
 | |
|             painter.drawText(pt, 'In')
 | |
|             i = 1
 | |
|             if not self.element.paramsFolded:
 | |
|                 for param in self.element.algorithm.parameters:
 | |
|                     if not param.hidden:
 | |
|                         text = self.getAdjustedText(param.description)
 | |
|                         h = -(fm.height() * 1.2) * (i + 1)
 | |
|                         h = h - ModelerGraphicItem.BOX_HEIGHT / 2.0 + 5
 | |
|                         pt = QPointF(-ModelerGraphicItem.BOX_WIDTH / 2
 | |
|                                      + 33, h)
 | |
|                         painter.drawText(pt, text)
 | |
|                         i += 1
 | |
|             h = fm.height() * 1.2
 | |
|             h = h + ModelerGraphicItem.BOX_HEIGHT / 2.0
 | |
|             pt = QPointF(-ModelerGraphicItem.BOX_WIDTH / 2 + 25, h)
 | |
|             painter.drawText(pt, 'Out')
 | |
|             if not self.element.outputsFolded:
 | |
|                 for i, out in enumerate(self.element.algorithm.outputs):
 | |
|                         text = self.getAdjustedText(out.description)
 | |
|                         h = fm.height() * 1.2 * (i + 2)
 | |
|                         h = h + ModelerGraphicItem.BOX_HEIGHT / 2.0
 | |
|                         pt = QPointF(-ModelerGraphicItem.BOX_WIDTH / 2
 | |
|                                      + 33, h)
 | |
|                         painter.drawText(pt, text)
 | |
|         if self.pixmap:
 | |
|             painter.drawPixmap(-(ModelerGraphicItem.BOX_WIDTH / 2.0) + 3, -8,
 | |
|                                self.pixmap)
 | |
| 
 | |
|     def getLinkPointForParameter(self, paramIndex):
 | |
|         offsetX = 25
 | |
|         if isinstance(self.element, Algorithm) and self.element.paramsFolded:
 | |
|             paramIndex = -1
 | |
|             offsetX = 17
 | |
|         font = QFont('Verdana', 8)
 | |
|         fm = QFontMetricsF(font)
 | |
|         if isinstance(self.element, Algorithm):
 | |
|             h = -(fm.height() * 1.2) * (paramIndex + 2) - fm.height() / 2.0 + 8
 | |
|             h = h - ModelerGraphicItem.BOX_HEIGHT / 2.0
 | |
|         else:
 | |
|             h = 0
 | |
|         return QPointF(-ModelerGraphicItem.BOX_WIDTH / 2 + offsetX, h)
 | |
| 
 | |
|     def getLinkPointForOutput(self, outputIndex):
 | |
|         if isinstance(self.element, Algorithm) and self.element.algorithm.outputs:
 | |
|             outputIndex = (outputIndex if not self.element.outputsFolded else -1)
 | |
|             text = self.getAdjustedText(self.element.algorithm.outputs[outputIndex].description)
 | |
|             font = QFont('Verdana', 8)
 | |
|             fm = QFontMetricsF(font)
 | |
|             w = fm.width(text)
 | |
|             h = fm.height() * 1.2 * (outputIndex + 1) + fm.height() / 2.0
 | |
|             y = h + ModelerGraphicItem.BOX_HEIGHT / 2.0 + 5
 | |
|             x = (-ModelerGraphicItem.BOX_WIDTH / 2 + 33 + w
 | |
|                  + 5 if not self.element.outputsFolded else 10)
 | |
|             return QPointF(x, y)
 | |
|         else:
 | |
|             return QPointF(0, 0)
 | |
| 
 | |
|     def itemChange(self, change, value):
 | |
|         if change == QGraphicsItem.ItemPositionHasChanged:
 | |
|             for arrow in self.arrows:
 | |
|                 arrow.updatePath()
 | |
|             self.element.pos = self.pos()
 | |
| 
 | |
|         return value
 | |
| 
 | |
|     def polygon(self):
 | |
|         font = QFont('Verdana', 8)
 | |
|         fm = QFontMetricsF(font)
 | |
|         hUp = fm.height() * 1.2 * (len(self.element.parameters) + 2)
 | |
|         hDown = fm.height() * 1.2 * (len(self.element.outputs) + 2)
 | |
|         pol = QPolygonF([
 | |
|             QPointF(-(ModelerGraphicItem.BOX_WIDTH + 2) / 2,
 | |
|                     -(ModelerGraphicItem.BOX_HEIGHT + 2) / 2 - hUp),
 | |
|             QPointF(-(ModelerGraphicItem.BOX_WIDTH + 2) / 2,
 | |
|                     (ModelerGraphicItem.BOX_HEIGHT + 2) / 2 + hDown),
 | |
|             QPointF((ModelerGraphicItem.BOX_WIDTH + 2) / 2,
 | |
|                     (ModelerGraphicItem.BOX_HEIGHT + 2) / 2 + hDown),
 | |
|             QPointF((ModelerGraphicItem.BOX_WIDTH + 2) / 2,
 | |
|                     -(ModelerGraphicItem.BOX_HEIGHT + 2) / 2 - hUp),
 | |
|             QPointF(-(ModelerGraphicItem.BOX_WIDTH + 2) / 2,
 | |
|                     -(ModelerGraphicItem.BOX_HEIGHT + 2) / 2 - hUp)
 | |
|         ])
 | |
|         return pol
 | |
| 
 | |
| 
 | |
| class FlatButtonGraphicItem(QGraphicsItem):
 | |
| 
 | |
|     WIDTH = 16
 | |
|     HEIGHT = 16
 | |
| 
 | |
|     def __init__(self, icon, position, action):
 | |
|         super(FlatButtonGraphicItem, self).__init__(None, None)
 | |
|         self.setAcceptHoverEvents(True)
 | |
|         self.setFlag(QGraphicsItem.ItemIsMovable, False)
 | |
|         self.pixmap = icon.pixmap(self.WIDTH, self.HEIGHT,
 | |
|                                   state=QIcon.On)
 | |
|         self.position = position
 | |
|         self.isIn = False
 | |
|         self.action = action
 | |
| 
 | |
|     def mousePressEvent(self, event):
 | |
|         self.action()
 | |
| 
 | |
|     def paint(self, painter, option, widget=None):
 | |
|         pt = QPointF(-self.WIDTH / 2, -self.HEIGHT / 2) + self.position
 | |
|         rect = QRectF(pt.x(), pt.y(), self.WIDTH, self.HEIGHT)
 | |
|         if self.isIn:
 | |
|             painter.setPen(QPen(Qt.transparent, 1))
 | |
|             painter.setBrush(QBrush(Qt.lightGray,
 | |
|                              Qt.SolidPattern))
 | |
|         else:
 | |
|             painter.setPen(QPen(Qt.transparent, 1))
 | |
|             painter.setBrush(QBrush(Qt.transparent,
 | |
|                              Qt.SolidPattern))
 | |
|         painter.drawRect(rect)
 | |
|         painter.drawPixmap(pt.x(), pt.y(), self.pixmap)
 | |
| 
 | |
|     def boundingRect(self):
 | |
|         rect = QRectF(self.position.x() - self.WIDTH / 2,
 | |
|                       self.position.y() - self.HEIGHT / 2,
 | |
|                       self.WIDTH,
 | |
|                       self.HEIGHT)
 | |
|         return rect
 | |
| 
 | |
|     def hoverEnterEvent(self, event):
 | |
|         self.isIn = True
 | |
|         self.update()
 | |
| 
 | |
|     def hoverLeaveEvent(self, event):
 | |
|         self.isIn = False
 | |
|         self.update()
 | |
| 
 | |
| 
 | |
| class FoldButtonGraphicItem(FlatButtonGraphicItem):
 | |
| 
 | |
|     WIDTH = 11
 | |
|     HEIGHT = 11
 | |
| 
 | |
|     icons = {True: QIcon(os.path.join(pluginPath, 'images', 'plus.png')),
 | |
|              False: QIcon(os.path.join(pluginPath, 'images', 'minus.png'))}
 | |
| 
 | |
|     def __init__(self, position, action, folded):
 | |
|         self.folded = folded
 | |
|         icon = self.icons[self.folded]
 | |
|         super(FoldButtonGraphicItem, self).__init__(icon, position, action)
 | |
| 
 | |
|     def mousePressEvent(self, event):
 | |
|         self.folded = not self.folded
 | |
|         icon = self.icons[self.folded]
 | |
|         self.pixmap = icon.pixmap(self.WIDTH, self.HEIGHT,
 | |
|                                   state=QIcon.On)
 | |
|         self.action(self.folded)
 |