2012-10-05 23:28:47 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
"""
|
|
|
|
***************************************************************************
|
|
|
|
NumberInputPanel.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. *
|
|
|
|
* *
|
|
|
|
***************************************************************************
|
|
|
|
"""
|
2016-09-27 19:51:06 +02:00
|
|
|
from builtins import str
|
2012-10-05 23:28:47 +02:00
|
|
|
|
|
|
|
__author__ = 'Victor Olaya'
|
|
|
|
__date__ = 'August 2012'
|
|
|
|
__copyright__ = '(C) 2012, Victor Olaya'
|
2013-10-01 20:52:22 +03:00
|
|
|
|
2012-10-05 23:28:47 +02:00
|
|
|
# This will get replaced with a git SHA1 when you do a git archive
|
2013-10-01 20:52:22 +03:00
|
|
|
|
2012-10-05 23:28:47 +02:00
|
|
|
__revision__ = '$Format:%H$'
|
|
|
|
|
2015-05-18 21:04:20 +03:00
|
|
|
import os
|
2016-11-11 11:02:42 +10:00
|
|
|
import math
|
2015-05-18 21:04:20 +03:00
|
|
|
|
2016-04-29 11:39:26 +02:00
|
|
|
from qgis.PyQt import uic
|
2016-04-22 10:38:48 +02:00
|
|
|
from qgis.PyQt.QtCore import pyqtSignal
|
2016-05-13 13:17:48 +03:00
|
|
|
from qgis.PyQt.QtWidgets import QDialog
|
2015-05-18 21:04:20 +03:00
|
|
|
|
2017-03-04 19:41:23 +01:00
|
|
|
from qgis.core import QgsExpression
|
|
|
|
from qgis.gui import QgsExpressionBuilderDialog
|
2016-09-13 06:45:02 +02:00
|
|
|
from processing.core.parameters import ParameterNumber, ParameterVector, ParameterRaster
|
|
|
|
from processing.core.outputs import OutputNumber, OutputVector, OutputRaster
|
2016-09-12 06:17:23 +02:00
|
|
|
from processing.modeler.ModelerAlgorithm import ValueFromInput, ValueFromOutput, CompoundValue
|
2012-09-15 18:25:25 +03:00
|
|
|
|
2015-05-18 21:04:20 +03:00
|
|
|
pluginPath = os.path.split(os.path.dirname(__file__))[0]
|
2016-11-11 11:02:42 +10:00
|
|
|
NUMBER_WIDGET, NUMBER_BASE = uic.loadUiType(
|
|
|
|
os.path.join(pluginPath, 'ui', 'widgetNumberSelector.ui'))
|
2015-05-18 21:04:20 +03:00
|
|
|
WIDGET, BASE = uic.loadUiType(
|
2016-09-09 07:02:54 +02:00
|
|
|
os.path.join(pluginPath, 'ui', 'widgetBaseSelector.ui'))
|
2015-05-18 21:04:20 +03:00
|
|
|
|
2013-10-01 20:52:22 +03:00
|
|
|
|
2016-11-11 11:02:42 +10:00
|
|
|
class ModellerNumberInputPanel(BASE, WIDGET):
|
|
|
|
"""
|
|
|
|
Number input panel for use inside the modeller - this input panel
|
|
|
|
is based off the base input panel and includes a text based line input
|
|
|
|
for entering values. This allows expressions and other non-numeric
|
|
|
|
values to be set, which are later evalauted to numbers when the model
|
|
|
|
is run.
|
|
|
|
"""
|
2012-09-15 18:25:25 +03:00
|
|
|
|
2016-01-19 09:00:13 +01:00
|
|
|
hasChanged = pyqtSignal()
|
|
|
|
|
2016-11-11 11:02:42 +10:00
|
|
|
def __init__(self, param, modelParametersDialog):
|
|
|
|
super(ModellerNumberInputPanel, self).__init__(None)
|
2013-10-09 19:21:11 +03:00
|
|
|
self.setupUi(self)
|
|
|
|
|
2016-09-09 07:02:54 +02:00
|
|
|
self.param = param
|
2016-09-12 06:17:23 +02:00
|
|
|
self.modelParametersDialog = modelParametersDialog
|
|
|
|
if param.default:
|
|
|
|
self.setValue(param.default)
|
2016-09-09 07:02:54 +02:00
|
|
|
self.btnSelect.clicked.connect(self.showExpressionsBuilder)
|
|
|
|
self.leText.textChanged.connect(lambda: self.hasChanged.emit())
|
2016-01-19 09:00:13 +01:00
|
|
|
|
2016-04-08 13:21:29 +02:00
|
|
|
def showExpressionsBuilder(self):
|
2016-09-09 07:02:54 +02:00
|
|
|
context = self.param.expressionContext()
|
2016-11-11 11:02:42 +10:00
|
|
|
dlg = QgsExpressionBuilderDialog(None, str(self.leText.text()), self, 'generic', context)
|
|
|
|
|
|
|
|
context.popScope()
|
|
|
|
values = self.modelParametersDialog.getAvailableValuesOfType(ParameterNumber, OutputNumber)
|
|
|
|
variables = {}
|
|
|
|
for value in values:
|
|
|
|
if isinstance(value, ValueFromInput):
|
|
|
|
name = value.name
|
|
|
|
element = self.modelParametersDialog.model.inputs[name].param
|
|
|
|
desc = element.description
|
|
|
|
else:
|
|
|
|
name = "%s_%s" % (value.alg, value.output)
|
|
|
|
alg = self.modelParametersDialog.model.algs[value.alg]
|
|
|
|
out = alg.algorithm.getOutputFromName(value.output)
|
2017-03-05 11:37:57 +01:00
|
|
|
desc = self.tr("Output '{0}' from algorithm '{1}'").format(out.description, alg.description)
|
2016-11-11 11:02:42 +10:00
|
|
|
variables[name] = desc
|
|
|
|
values = self.modelParametersDialog.getAvailableValuesOfType(ParameterVector, OutputVector)
|
|
|
|
values.extend(self.modelParametersDialog.getAvailableValuesOfType(ParameterRaster, OutputRaster))
|
|
|
|
for value in values:
|
|
|
|
if isinstance(value, ValueFromInput):
|
|
|
|
name = value.name
|
|
|
|
element = self.modelParametersDialog.model.inputs[name].param
|
|
|
|
desc = element.description
|
|
|
|
else:
|
|
|
|
name = "%s_%s" % (value.alg, value.output)
|
|
|
|
alg = self.modelParametersDialog.model.algs[value.alg]
|
|
|
|
element = alg.algorithm.getOutputFromName(value.output)
|
2017-03-05 11:37:57 +01:00
|
|
|
desc = self.tr("Output '{0}' from algorithm '{1}'").format(element.description, alg.description)
|
|
|
|
variables['%s_minx' % name] = self.tr("Minimum X of {0}").format(desc)
|
|
|
|
variables['%s_miny' % name] = self.tr("Minimum Y of {0}").format(desc)
|
|
|
|
variables['%s_maxx' % name] = self.tr("Maximum X of {0}").format(desc)
|
|
|
|
variables['%s_maxy' % name] = self.tr("Maximum Y of {0}").format(desc)
|
2016-11-11 11:02:42 +10:00
|
|
|
if isinstance(element, (ParameterRaster, OutputRaster)):
|
2017-03-05 11:37:57 +01:00
|
|
|
variables['%s_min' % name] = self.tr("Minimum value of {0}").format(desc)
|
|
|
|
variables['%s_max' % name] = self.tr("Maximum value of {0}").format(desc)
|
|
|
|
variables['%s_avg' % name] = self.tr("Mean value of {0}").format(desc)
|
|
|
|
variables['%s_stddev' % name] = self.tr("Standard deviation of {0}").format(desc)
|
2016-11-11 11:02:42 +10:00
|
|
|
for variable, desc in variables.items():
|
|
|
|
dlg.expressionBuilder().registerItem("Modeler", variable, "@" + variable, desc, highlightedItem=True)
|
|
|
|
|
2016-05-13 09:32:13 +03:00
|
|
|
dlg.setWindowTitle(self.tr('Expression based input'))
|
2016-04-08 13:21:29 +02:00
|
|
|
if dlg.exec_() == QDialog.Accepted:
|
|
|
|
exp = QgsExpression(dlg.expressionText())
|
|
|
|
if not exp.hasParserError():
|
2016-09-09 07:02:54 +02:00
|
|
|
self.setValue(dlg.expressionText())
|
2012-09-15 18:25:25 +03:00
|
|
|
|
|
|
|
def getValue(self):
|
2016-11-11 11:02:42 +10:00
|
|
|
value = self.leText.text()
|
|
|
|
values = []
|
|
|
|
for param in self.modelParametersDialog.model.parameters:
|
|
|
|
if isinstance(param, ParameterNumber):
|
|
|
|
if "@" + param.name in value:
|
|
|
|
values.append(ValueFromInput(param.name))
|
|
|
|
for alg in list(self.modelParametersDialog.model.algs.values()):
|
|
|
|
for out in alg.algorithm.outputs:
|
2017-03-29 12:51:59 +10:00
|
|
|
if isinstance(out, OutputNumber) and "@%s_%s" % (alg.name(), out.name) in value:
|
|
|
|
values.append(ValueFromOutput(alg.name(), out.name))
|
2016-11-11 11:02:42 +10:00
|
|
|
if values:
|
|
|
|
return CompoundValue(values, value)
|
2016-09-12 06:17:23 +02:00
|
|
|
else:
|
2016-11-11 11:02:42 +10:00
|
|
|
return value
|
2015-11-18 16:26:55 +11:00
|
|
|
|
2016-09-07 14:30:20 +02:00
|
|
|
def setValue(self, value):
|
2016-09-27 19:51:06 +02:00
|
|
|
self.leText.setText(str(value))
|
2016-11-11 11:02:42 +10:00
|
|
|
|
|
|
|
|
|
|
|
class NumberInputPanel(NUMBER_BASE, NUMBER_WIDGET):
|
|
|
|
"""
|
|
|
|
Number input panel for use outside the modeller - this input panel
|
|
|
|
contains a user friendly spin box for entering values. It also
|
|
|
|
allows expressions to be evaluated, but these expressions are evaluated
|
|
|
|
immediately after entry and are not stored anywhere.
|
|
|
|
"""
|
|
|
|
|
|
|
|
hasChanged = pyqtSignal()
|
|
|
|
|
|
|
|
def __init__(self, param):
|
|
|
|
super(NumberInputPanel, self).__init__(None)
|
|
|
|
self.setupUi(self)
|
|
|
|
|
|
|
|
self.spnValue.setExpressionsEnabled(True)
|
|
|
|
|
|
|
|
self.param = param
|
|
|
|
if self.param.isInteger:
|
|
|
|
self.spnValue.setDecimals(0)
|
|
|
|
else:
|
|
|
|
# Guess reasonable step value
|
|
|
|
if self.param.max is not None and self.param.min is not None:
|
|
|
|
try:
|
|
|
|
self.spnValue.setSingleStep(self.calculateStep(float(self.param.min), float(self.param.max)))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if self.param.max is not None:
|
|
|
|
self.spnValue.setMaximum(self.param.max)
|
|
|
|
else:
|
|
|
|
self.spnValue.setMaximum(999999999)
|
|
|
|
if self.param.min is not None:
|
|
|
|
self.spnValue.setMinimum(self.param.min)
|
|
|
|
else:
|
|
|
|
self.spnValue.setMinimum(-999999999)
|
|
|
|
|
|
|
|
# set default value
|
|
|
|
if param.default is not None:
|
|
|
|
self.setValue(param.default)
|
|
|
|
try:
|
|
|
|
self.spnValue.setClearValue(float(param.default))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
elif self.param.min is not None:
|
|
|
|
try:
|
|
|
|
self.setValue(float(self.param.min))
|
|
|
|
self.spnValue.setClearValue(float(self.param.min))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
self.setValue(0)
|
|
|
|
self.spnValue.setClearValue(0)
|
|
|
|
self.btnSelect.setFixedHeight(self.spnValue.height())
|
|
|
|
|
|
|
|
self.btnSelect.clicked.connect(self.showExpressionsBuilder)
|
|
|
|
self.spnValue.valueChanged.connect(lambda: self.hasChanged.emit())
|
|
|
|
|
|
|
|
def showExpressionsBuilder(self):
|
|
|
|
context = self.param.expressionContext()
|
|
|
|
dlg = QgsExpressionBuilderDialog(None, str(self.spnValue.value()), self, 'generic', context)
|
|
|
|
|
|
|
|
dlg.setWindowTitle(self.tr('Expression based input'))
|
|
|
|
if dlg.exec_() == QDialog.Accepted:
|
|
|
|
exp = QgsExpression(dlg.expressionText())
|
|
|
|
if not exp.hasParserError():
|
|
|
|
try:
|
|
|
|
val = float(exp.evaluate(context))
|
|
|
|
self.setValue(val)
|
|
|
|
except:
|
|
|
|
return
|
|
|
|
|
|
|
|
def getValue(self):
|
|
|
|
return self.spnValue.value()
|
|
|
|
|
|
|
|
def setValue(self, value):
|
|
|
|
try:
|
|
|
|
self.spnValue.setValue(float(value))
|
|
|
|
except:
|
|
|
|
return
|
|
|
|
|
|
|
|
def calculateStep(self, minimum, maximum):
|
|
|
|
value_range = maximum - minimum
|
|
|
|
if value_range <= 1.0:
|
|
|
|
step = value_range / 10.0
|
|
|
|
# round to 1 significant figrue
|
|
|
|
return round(step, -int(math.floor(math.log10(step))))
|
|
|
|
else:
|
|
|
|
return 1.0
|