# -*- coding: utf-8 -*- """ *************************************************************************** ShortestPath.py --------------------- Date : November 2016 Copyright : (C) 2016 by Alexander Bruy Email : alexander dot bruy 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__ = 'Alexander Bruy' __date__ = 'November 2016' __copyright__ = '(C) 2016, Alexander Bruy' # This will get replaced with a git SHA1 when you do a git archive __revision__ = '$Format:%H$' import os from qgis.PyQt.QtGui import QIcon from qgis.core import QgsWkbTypes, QgsFeature, QgsGeometry, QgsPoint from qgis.analysis import (QgsVectorLayerDirector, QgsNetworkDistanceStrategy, QgsGraphBuilder, QgsGraphAnalyzer ) from qgis.utils import iface from processing.core.GeoAlgorithm import GeoAlgorithm from processing.core.GeoAlgorithmExecutionException import GeoAlgorithmExecutionException from processing.core.parameters import (ParameterVector, ParameterPoint, ParameterNumber, ParameterString, ParameterTableField, ParameterSelection ) from processing.core.outputs import (OutputNumber, OutputVector ) from processing.tools import dataobjects pluginPath = os.path.split(os.path.split(os.path.dirname(__file__))[0])[0] class ShortestPath(GeoAlgorithm): INPUT_VECTOR = 'INPUT_VECTOR' START_POINT = 'START_POINT' END_POINT = 'END_POINT' DIRECTION_FIELD = 'DIRECTION_FIELD' VALUE_FORWARD = 'VALUE_FORWARD' VALUE_BACKWARD = 'VALUE_BACKWARD' VALUE_BOTH = 'VALUE_BOTH' DEFAULT_DIRECTION = 'DEFAULT_DIRECTION' TOLERANCE = 'TOLERANCE' PATH_LENGTH = 'PATH_LENGTH' OUTPUT_LAYER = 'OUTPUT_LAYER' def getIcon(self): return QIcon(os.path.join(pluginPath, 'images', 'networkanalysis.svg')) def defineCharacteristics(self): self.DIRECTIONS = {self.tr('Forward direction'): QgsVectorLayerDirector.DirectionForward, self.tr('Backward direction'): QgsVectorLayerDirector.DirectionForward, self.tr('Both directions'): QgsVectorLayerDirector.DirectionForward } self.UNITS = {self.tr('Meters'): 1, self.tr('Kilometers'): 1000 } self.name, self.i18n_name = self.trAlgorithm('Shortest path') self.group, self.i18n_group = self.trAlgorithm('Network analysis') self.addParameter(ParameterVector(self.INPUT_VECTOR, self.tr('Vector layer representing road network'), [dataobjects.TYPE_VECTOR_LINE])) self.addParameter(ParameterPoint(self.START_POINT, self.tr('Start point'))) self.addParameter(ParameterPoint(self.END_POINT, self.tr('End point'))) params = [] params.append(ParameterTableField(self.DIRECTION_FIELD, self.tr('Road direction field'), self.INPUT_VECTOR, optional=True)) params.append(ParameterString(self.VALUE_FORWARD, self.tr('Value for forward direction'), '', optional=True)) params.append(ParameterString(self.VALUE_BACKWARD, self.tr('Value for backward direction'), '', optional=True)) params.append(ParameterString(self.VALUE_BOTH, self.tr('Value for both directions'), '', optional=True)) params.append(ParameterSelection(self.DEFAULT_DIRECTION, self.tr('Default road direction'), list(self.DIRECTIONS.keys()), default=2)) params.append(ParameterNumber(self.TOLERANCE, self.tr('Topology tolerance'), 0.0, 0.0, 99999999.999999)) for p in params: p.isAdvanced = True self.addParameter(p) self.addOutput(OutputNumber(self.PATH_LENGTH, self.tr('Path length'))) self.addOutput(OutputVector(self.OUTPUT_LAYER, self.tr('Shortest path'), datatype=[dataobjects.TYPE_VECTOR_LINE])) def processAlgorithm(self, progress): layer = dataobjects.getObjectFromUri( self.getParameterValue(self.INPUT_VECTOR)) startPoint = self.getParameterValue(self.START_POINT) endPoint = self.getParameterValue(self.END_POINT) fieldName = self.getParameterValue(self.DIRECTION_FIELD) forwardValue = self.getParameterValue(self.VALUE_FORWARD) backwardValue = self.getParameterValue(self.VALUE_BACKWARD) bothValue = self.getParameterValue(self.VALUE_BOTH) defaultDirection = self.getParameterValue(self.DEFAULT_DIRECTION) tolerance = self.getParameterValue(self.TOLERANCE) writer = self.getOutputFromName( self.OUTPUT_LAYER).getVectorWriter( layer.fields().toList(), QgsWkbTypes.LineString, layer.crs()) tmp = startPoint.split(',') startPoint = QgsPoint(float(tmp[0]), float(tmp[1])) tmp = endPoint.split(',') endPoint = QgsPoint(float(tmp[0]), float(tmp[1])) field = -1 if fieldName is not None: field = layer.fields().lookupField(fieldName) director = QgsVectorLayerDirector(layer, field, forwardValue, backwardValue, bothValue, defaultDirection) strategy = QgsNetworkDistanceStrategy() director.addStrategy(strategy) builder = QgsGraphBuilder(iface.mapCanvas().mapSettings().destinationCrs(), iface.mapCanvas().hasCrsTransformEnabled(), tolerance) progress.setInfo(self.tr('Building graph...')) snappedPoints = director.makeGraph(builder, [startPoint, endPoint]) progress.setInfo(self.tr('Calculating shortest path...')) graph = builder.graph() idxStart = graph.findVertex(snappedPoints[0]) idxEnd = graph.findVertex(snappedPoints[1]) tree, cost = QgsGraphAnalyzer.dijkstra(graph, idxStart, 0) if tree[idxEnd] == -1: raise GeoAlgorithmExecutionException( self.tr('There is no route from start point to end point.')) route = [] cost = 0.0 current = idxEnd while current != idxStart: cost += graph.edge(tree[current]).cost(0) route.append(graph.vertex(graph.edge(tree[current]).inVertex()).point()) current = graph.edge(tree[current]).outVertex() route.append(snappedPoints[0]) route.reverse() self.setOutputValue(self.PATH_LENGTH, cost) progress.setInfo(self.tr('Writting results...')) geom = QgsGeometry.fromPolyline(route) feat = QgsFeature() feat.setGeometry(geom) writer.addFeature(feat) del writer