mirror of
				https://github.com/qgis/QGIS.git
				synced 2025-10-26 00:04:03 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			437 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			437 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| ***************************************************************************
 | |
|     ModelerDialog.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"
 | |
| 
 | |
| import os
 | |
| import re
 | |
| import sys
 | |
| from pathlib import Path
 | |
| 
 | |
| from qgis.PyQt.QtCore import (
 | |
|     QCoreApplication,
 | |
|     QDir,
 | |
|     QRectF,
 | |
|     QPoint,
 | |
|     QPointF,
 | |
|     pyqtSignal,
 | |
|     QUrl,
 | |
|     QFileInfo,
 | |
| )
 | |
| from qgis.PyQt.QtWidgets import QMessageBox, QFileDialog
 | |
| from qgis.core import (
 | |
|     Qgis,
 | |
|     QgsApplication,
 | |
|     QgsProcessing,
 | |
|     QgsProject,
 | |
|     QgsProcessingModelParameter,
 | |
|     QgsProcessingModelAlgorithm,
 | |
|     QgsSettings,
 | |
|     QgsProcessingContext,
 | |
|     QgsFileUtils,
 | |
| )
 | |
| from qgis.gui import (
 | |
|     QgsProcessingParameterDefinitionDialog,
 | |
|     QgsProcessingParameterWidgetContext,
 | |
|     QgsModelGraphicsScene,
 | |
|     QgsModelDesignerDialog,
 | |
|     QgsProcessingContextGenerator,
 | |
|     QgsProcessingParametersGenerator,
 | |
| )
 | |
| from qgis.utils import iface
 | |
| 
 | |
| from processing.gui.AlgorithmDialog import AlgorithmDialog
 | |
| from processing.modeler.ModelerParameterDefinitionDialog import (
 | |
|     ModelerParameterDefinitionDialog,
 | |
| )
 | |
| from processing.modeler.ModelerParametersDialog import ModelerParametersDialog
 | |
| from processing.modeler.ModelerScene import ModelerScene
 | |
| from processing.modeler.ModelerUtils import ModelerUtils
 | |
| from processing.modeler.ProjectProvider import PROJECT_PROVIDER_ID
 | |
| from processing.script.ScriptEditorDialog import ScriptEditorDialog
 | |
| from processing.tools.dataobjects import createContext
 | |
| 
 | |
| pluginPath = os.path.split(os.path.dirname(__file__))[0]
 | |
| 
 | |
| 
 | |
| class ModelerDialog(QgsModelDesignerDialog):
 | |
|     CANVAS_SIZE = 4000
 | |
| 
 | |
|     update_model = pyqtSignal()
 | |
| 
 | |
|     dlgs = []
 | |
| 
 | |
|     @staticmethod
 | |
|     def create(model=None):
 | |
|         """
 | |
|         Workaround crappy sip handling of QMainWindow. It doesn't know that we are using the deleteonclose
 | |
|         flag, so happily just deletes dialogs as soon as they go out of scope. The only workaround possible
 | |
|         while we still have to drag around this Python code is to store a reference to the sip wrapper so that
 | |
|         sip doesn't get confused. The underlying object will still be deleted by the deleteonclose flag though!
 | |
|         """
 | |
|         dlg = ModelerDialog(model)
 | |
|         ModelerDialog.dlgs.append(dlg)
 | |
|         return dlg
 | |
| 
 | |
|     def __init__(self, model=None, parent=None):
 | |
|         super().__init__(parent)
 | |
| 
 | |
|         if iface is not None:
 | |
|             self.toolbar().setIconSize(iface.iconSize())
 | |
|             self.setStyleSheet(iface.mainWindow().styleSheet())
 | |
| 
 | |
|         scene = ModelerScene(self)
 | |
|         scene.setSceneRect(QRectF(0, 0, self.CANVAS_SIZE, self.CANVAS_SIZE))
 | |
|         self.setModelScene(scene)
 | |
| 
 | |
|         self.view().ensureVisible(0, 0, 10, 10)
 | |
|         self.view().scale(self.logicalDpiX() / 96, self.logicalDpiY() / 96)
 | |
| 
 | |
|         self.actionOpen().triggered.connect(self.openModel)
 | |
|         self.actionSaveInProject().triggered.connect(self.saveInProject)
 | |
| 
 | |
|         if model is not None:
 | |
|             _model = model.create()
 | |
|             _model.setSourceFilePath(model.sourceFilePath())
 | |
|             self.setModel(_model)
 | |
| 
 | |
|         self.view().centerOn(0, 0)
 | |
| 
 | |
|         self.processing_context = createContext()
 | |
| 
 | |
|         class ContextGenerator(QgsProcessingContextGenerator):
 | |
| 
 | |
|             def __init__(self, context):
 | |
|                 super().__init__()
 | |
|                 self.processing_context = context
 | |
| 
 | |
|             def processingContext(self):
 | |
|                 return self.processing_context
 | |
| 
 | |
|         self.context_generator = ContextGenerator(self.processing_context)
 | |
| 
 | |
|     def createExecutionDialog(self):
 | |
|         dlg = AlgorithmDialog(self.model().create(), parent=self)
 | |
|         return dlg
 | |
| 
 | |
|     def saveInProject(self):
 | |
|         if not self.validateSave(QgsModelDesignerDialog.SaveAction.SaveInProject):
 | |
|             return
 | |
| 
 | |
|         self.model().setSourceFilePath(None)
 | |
| 
 | |
|         project_provider = QgsApplication.processingRegistry().providerById(
 | |
|             PROJECT_PROVIDER_ID
 | |
|         )
 | |
|         project_provider.add_model(self.model())
 | |
| 
 | |
|         self.update_model.emit()
 | |
|         self.messageBar().pushMessage(
 | |
|             "",
 | |
|             self.tr("Model was saved inside current project"),
 | |
|             level=Qgis.MessageLevel.Success,
 | |
|             duration=5,
 | |
|         )
 | |
| 
 | |
|         self.setDirty(False)
 | |
|         QgsProject.instance().setDirty(True)
 | |
| 
 | |
|     def saveModel(self, saveAs) -> bool:
 | |
|         if not self.validateSave(QgsModelDesignerDialog.SaveAction.SaveAsFile):
 | |
|             return False
 | |
| 
 | |
|         model_name_matched_file_name = self.model().modelNameMatchesFilePath()
 | |
|         if self.model().sourceFilePath() and not saveAs:
 | |
|             filename = self.model().sourceFilePath()
 | |
|         else:
 | |
|             if self.model().sourceFilePath():
 | |
|                 initial_path = Path(self.model().sourceFilePath())
 | |
|             elif self.model().name():
 | |
|                 initial_path = Path(ModelerUtils.modelsFolders()[0]) / (
 | |
|                     self.model().name() + ".model3"
 | |
|                 )
 | |
|             else:
 | |
|                 initial_path = Path(ModelerUtils.modelsFolders()[0])
 | |
| 
 | |
|             filename, _ = QFileDialog.getSaveFileName(
 | |
|                 self,
 | |
|                 self.tr("Save Model"),
 | |
|                 initial_path.as_posix(),
 | |
|                 self.tr("Processing models (*.model3 *.MODEL3)"),
 | |
|             )
 | |
|             if not filename:
 | |
|                 return False
 | |
| 
 | |
|             filename = QgsFileUtils.ensureFileNameHasExtension(filename, ["model3"])
 | |
|             self.model().setSourceFilePath(filename)
 | |
| 
 | |
|             if not self.model().name() or self.model().name() == self.tr("model"):
 | |
|                 self.setModelName(Path(filename).stem)
 | |
|             elif saveAs and model_name_matched_file_name:
 | |
|                 # if saving as, and the model name used to match the filename, then automatically update the
 | |
|                 # model name to match the new file name
 | |
|                 self.setModelName(Path(filename).stem)
 | |
| 
 | |
|         if not self.model().toFile(filename):
 | |
|             if saveAs:
 | |
|                 QMessageBox.warning(
 | |
|                     self,
 | |
|                     self.tr("I/O error"),
 | |
|                     self.tr("Unable to save edits. Reason:\n {0}").format(
 | |
|                         str(sys.exc_info()[1])
 | |
|                     ),
 | |
|                 )
 | |
|             else:
 | |
|                 QMessageBox.warning(
 | |
|                     self,
 | |
|                     self.tr("Can't save model"),
 | |
|                     self.tr(
 | |
|                         "This model can't be saved in its original location (probably you do not "
 | |
|                         "have permission to do it). Please, use the 'Save as…' option."
 | |
|                     ),
 | |
|                 )
 | |
|             return False
 | |
| 
 | |
|         self.update_model.emit()
 | |
|         if saveAs:
 | |
|             self.messageBar().pushMessage(
 | |
|                 "",
 | |
|                 self.tr('Model was saved to <a href="{}">{}</a>').format(
 | |
|                     QUrl.fromLocalFile(filename).toString(),
 | |
|                     QDir.toNativeSeparators(filename),
 | |
|                 ),
 | |
|                 level=Qgis.MessageLevel.Success,
 | |
|                 duration=5,
 | |
|             )
 | |
| 
 | |
|         self.setDirty(False)
 | |
|         return True
 | |
| 
 | |
|     def openModel(self):
 | |
|         if not self.checkForUnsavedChanges():
 | |
|             return
 | |
| 
 | |
|         settings = QgsSettings()
 | |
|         last_dir = settings.value("Processing/lastModelsDir", QDir.homePath())
 | |
|         filename, selected_filter = QFileDialog.getOpenFileName(
 | |
|             self,
 | |
|             self.tr("Open Model"),
 | |
|             last_dir,
 | |
|             self.tr("Processing models (*.model3 *.MODEL3)"),
 | |
|         )
 | |
|         if filename:
 | |
|             settings.setValue(
 | |
|                 "Processing/lastModelsDir",
 | |
|                 QFileInfo(filename).absoluteDir().absolutePath(),
 | |
|             )
 | |
|             self.loadModel(filename)
 | |
| 
 | |
|     def repaintModel(self, showControls=True):
 | |
|         scene = ModelerScene(self)
 | |
|         scene.setSceneRect(QRectF(0, 0, self.CANVAS_SIZE, self.CANVAS_SIZE))
 | |
| 
 | |
|         if not showControls:
 | |
|             scene.setFlag(QgsModelGraphicsScene.Flag.FlagHideControls)
 | |
| 
 | |
|         showComments = QgsSettings().value(
 | |
|             "/Processing/Modeler/ShowComments", True, bool
 | |
|         )
 | |
|         if not showComments:
 | |
|             scene.setFlag(QgsModelGraphicsScene.Flag.FlagHideComments)
 | |
| 
 | |
|         context = createContext()
 | |
|         self.setModelScene(scene)
 | |
|         # create items later that setModelScene to setup link to messageBar to the scene
 | |
|         scene.createItems(self.model(), context)
 | |
| 
 | |
|     def create_widget_context(self):
 | |
|         """
 | |
|         Returns a new widget context for use in the model editor
 | |
|         """
 | |
|         widget_context = QgsProcessingParameterWidgetContext()
 | |
|         widget_context.setProject(QgsProject.instance())
 | |
|         if iface is not None:
 | |
|             widget_context.setMapCanvas(iface.mapCanvas())
 | |
|             widget_context.setActiveLayer(iface.activeLayer())
 | |
|         widget_context.setModel(self.model())
 | |
|         return widget_context
 | |
| 
 | |
|     def autogenerate_parameter_name(self, parameter):
 | |
|         """
 | |
|         Automatically generates and sets a new parameter's name, based on the parameter's
 | |
|         description and ensuring that it is unique for the model.
 | |
|         """
 | |
|         safeName = QgsProcessingModelAlgorithm.safeName(parameter.description())
 | |
|         name = safeName.lower()
 | |
|         i = 2
 | |
|         while self.model().parameterDefinition(name):
 | |
|             name = safeName.lower() + str(i)
 | |
|             i += 1
 | |
|         parameter.setName(name)
 | |
| 
 | |
|     def addInput(self, paramType, pos=None):
 | |
|         if paramType not in [
 | |
|             param.id()
 | |
|             for param in QgsApplication.instance().processingRegistry().parameterTypes()
 | |
|         ]:
 | |
|             return
 | |
| 
 | |
|         new_param = None
 | |
|         comment = None
 | |
|         if ModelerParameterDefinitionDialog.use_legacy_dialog(paramType=paramType):
 | |
|             dlg = ModelerParameterDefinitionDialog(self.model(), paramType)
 | |
|             if dlg.exec():
 | |
|                 new_param = dlg.param
 | |
|                 comment = dlg.comments()
 | |
|         else:
 | |
|             # yay, use new API!
 | |
|             context = createContext()
 | |
|             widget_context = self.create_widget_context()
 | |
|             dlg = QgsProcessingParameterDefinitionDialog(
 | |
|                 type=paramType,
 | |
|                 context=context,
 | |
|                 widgetContext=widget_context,
 | |
|                 algorithm=self.model(),
 | |
|             )
 | |
|             dlg.registerProcessingContextGenerator(self.context_generator)
 | |
|             if dlg.exec():
 | |
|                 new_param = dlg.createParameter()
 | |
|                 self.autogenerate_parameter_name(new_param)
 | |
|                 comment = dlg.comments()
 | |
| 
 | |
|         if new_param is not None:
 | |
|             if pos is None or not pos:
 | |
|                 pos = self.getPositionForParameterItem()
 | |
|             if isinstance(pos, QPoint):
 | |
|                 pos = QPointF(pos)
 | |
|             component = QgsProcessingModelParameter(new_param.name())
 | |
|             component.setDescription(new_param.name())
 | |
|             component.setPosition(pos)
 | |
| 
 | |
|             component.comment().setDescription(comment)
 | |
|             component.comment().setPosition(
 | |
|                 component.position()
 | |
|                 + QPointF(component.size().width(), -1.5 * component.size().height())
 | |
|             )
 | |
| 
 | |
|             self.beginUndoCommand(self.tr("Add Model Input"))
 | |
|             self.model().addModelParameter(new_param, component)
 | |
|             self.repaintModel()
 | |
|             # self.view().ensureVisible(self.scene.getLastParameterItem())
 | |
|             self.endUndoCommand()
 | |
| 
 | |
|     def getPositionForParameterItem(self):
 | |
|         MARGIN = 20
 | |
|         BOX_WIDTH = 200
 | |
|         BOX_HEIGHT = 80
 | |
|         if len(self.model().parameterComponents()) > 0:
 | |
|             maxX = max(
 | |
|                 [
 | |
|                     i.position().x()
 | |
|                     for i in list(self.model().parameterComponents().values())
 | |
|                 ]
 | |
|             )
 | |
|             newX = min(MARGIN + BOX_WIDTH + maxX, self.CANVAS_SIZE - BOX_WIDTH)
 | |
|         else:
 | |
|             newX = MARGIN + BOX_WIDTH / 2
 | |
|         return QPointF(newX, MARGIN + BOX_HEIGHT / 2)
 | |
| 
 | |
|     def addAlgorithm(self, alg_id, pos=None):
 | |
|         alg = QgsApplication.processingRegistry().createAlgorithmById(alg_id)
 | |
|         if not alg:
 | |
|             return
 | |
| 
 | |
|         dlg = ModelerParametersDialog(alg, self.model())
 | |
|         if dlg.exec():
 | |
|             alg = dlg.createAlgorithm()
 | |
|             if pos is None or not pos:
 | |
|                 alg.setPosition(self.getPositionForAlgorithmItem())
 | |
|             else:
 | |
|                 alg.setPosition(pos)
 | |
| 
 | |
|             alg.comment().setPosition(
 | |
|                 alg.position() + QPointF(alg.size().width(), -1.5 * alg.size().height())
 | |
|             )
 | |
| 
 | |
|             output_offset_x = alg.size().width()
 | |
|             output_offset_y = 1.5 * alg.size().height()
 | |
|             for out in alg.modelOutputs():
 | |
|                 alg.modelOutput(out).setPosition(
 | |
|                     alg.position() + QPointF(output_offset_x, output_offset_y)
 | |
|                 )
 | |
|                 output_offset_y += 1.5 * alg.modelOutput(out).size().height()
 | |
| 
 | |
|             self.beginUndoCommand(self.tr("Add Algorithm"))
 | |
|             id = self.model().addChildAlgorithm(alg)
 | |
|             self.repaintModel()
 | |
|             self.endUndoCommand()
 | |
| 
 | |
|             res, errors = self.model().validateChildAlgorithm(id)
 | |
|             if not res:
 | |
|                 self.view().scene().showWarning(
 | |
|                     QCoreApplication.translate(
 | |
|                         "ModelerDialog", "Algorithm “{}” is invalid"
 | |
|                     ).format(alg.description()),
 | |
|                     self.tr("Algorithm is Invalid"),
 | |
|                     QCoreApplication.translate(
 | |
|                         "ModelerDialog",
 | |
|                         "<p>The “{}” algorithm is invalid, because:</p><ul><li>{}</li></ul>",
 | |
|                     ).format(alg.description(), "</li><li>".join(errors)),
 | |
|                     level=Qgis.MessageLevel.Warning,
 | |
|                 )
 | |
|             else:
 | |
|                 self.view().scene().messageBar().clearWidgets()
 | |
| 
 | |
|     def getPositionForAlgorithmItem(self):
 | |
|         MARGIN = 20
 | |
|         BOX_WIDTH = 200
 | |
|         BOX_HEIGHT = 80
 | |
|         if self.model().childAlgorithms():
 | |
|             maxX = max(
 | |
|                 [
 | |
|                     alg.position().x()
 | |
|                     for alg in list(self.model().childAlgorithms().values())
 | |
|                 ]
 | |
|             )
 | |
|             maxY = max(
 | |
|                 [
 | |
|                     alg.position().y()
 | |
|                     for alg in list(self.model().childAlgorithms().values())
 | |
|                 ]
 | |
|             )
 | |
|             newX = min(MARGIN + BOX_WIDTH + maxX, self.CANVAS_SIZE - BOX_WIDTH)
 | |
|             newY = min(MARGIN + BOX_HEIGHT + maxY, self.CANVAS_SIZE - BOX_HEIGHT)
 | |
|         else:
 | |
|             newX = MARGIN + BOX_WIDTH / 2
 | |
|             newY = MARGIN * 2 + BOX_HEIGHT + BOX_HEIGHT / 2
 | |
|         return QPointF(newX, newY)
 | |
| 
 | |
|     def exportAsScriptAlgorithm(self):
 | |
|         dlg = ScriptEditorDialog(parent=iface.mainWindow())
 | |
| 
 | |
|         dlg.editor.setText(
 | |
|             "\n".join(
 | |
|                 self.model().asPythonCode(
 | |
|                     QgsProcessing.PythonOutputType.PythonQgsProcessingAlgorithmSubclass,
 | |
|                     4,
 | |
|                 )
 | |
|             )
 | |
|         )
 | |
|         dlg.show()
 |