-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathSTANDALONE.py
287 lines (229 loc) · 10.7 KB
/
STANDALONE.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
"""
STANDALONE.py
This script executes the seilaplan core algorithms without the help of QGIS.
No GUI is shown, so no parameters can be changed by hand in a window. All
settings are read out from a previously saved seilaplan project file.
To run the script, the following variables have to be set:
- Line ~21: Define path to python script files in your QGIS installation.
The path should be similar to the one already written.
- Line ~120 - ~150: Define some paths for input and output and select what
algorithms should be run.
"""
import sys
# ################# PATH TO CURRENT QGIS PYTHON LOCATION ######################
if 'WIN' in sys.platform.upper():
sys.path.append(r'C:\Program Files\QGIS 3.4\apps\qgis-ltr\python')
# sys.path.append(r'C:\Program Files (x86)\QGIS 3.4\apps\qgis-ltr\python')
# Other possible paths:
# r'C:\Program Files\QGIS 3.4\apps\qgis-ltr\python'
# r'C:\Program Files\QGIS 3.10\apps\qgis\python'
# #############################################################################
import os
# Add shipped libraries to python path (reportlab)
libPath = os.path.join(os.path.dirname(__file__), 'lib')
if libPath not in sys.path:
sys.path.insert(0, libPath)
import traceback
from qgis.core import QgsTask, QgsApplication
from qgis.PyQt.QtCore import pyqtSignal, QTranslator, QCoreApplication
from SEILAPLAN.tools.configHandler import ConfigHandler
from SEILAPLAN.core.mainSeilaplan import main as mainSeilaplan
from SEILAPLAN.core.cablelineFinal import preciseCable, updateWithCableCoordinates
from SEILAPLAN.gui.adjustmentPlot import AdjustmentPlot, calculatePlotDimensions
from SEILAPLAN.tools.outputReport import (generateReportText, generateReport,
createOutputFolder, generateShortReport)
from SEILAPLAN.tools.outputGeo import (organizeDataForExport, generateCoordTable, writeGeodata)
from SEILAPLAN.tools.birdViewMapExtractor import extractMapBackground
from SEILAPLAN.tools.globals import ResultQuality
class ProcessingTask(QgsTask):
"""
This is a Dummy Class to handle progress information events from the
algorithm. Normally, this would be handled by the QGIS task manager.
"""
# Signals
sig_jobEnded = pyqtSignal(bool)
sig_jobError = pyqtSignal(str)
sig_value = pyqtSignal(float)
sig_range = pyqtSignal(list)
sig_text = pyqtSignal(str)
sig_result = pyqtSignal(list)
def __init__(self, confHandler, description="Dummy"):
super().__init__(description, QgsTask.CanCancel)
self.state = False
self.exception = None
self.confHandler: ConfigHandler = confHandler
self.projInfo = confHandler.project
self.result = None
self.status = []
def isCanceled(self):
return
def emit(*args):
return
def cancel(self):
super().cancel()
def getStatusAsStr(self):
return max(self.status)
def optimizeCableLine(conf: ConfigHandler):
""" This function performs the optimization of the pole locations and
calculates the finale cable line.
"""
# Ready configuration data from project file
print('Set up configuration and input values...')
success = conf.prepareForCalculation(runOptimization=True)
if not success:
print('ERROR: Error while preparing config data for optimization.')
exit()
# Initialize dummy task manager
task = ProcessingTask(conf)
res = None
try:
print('Start optimization...')
# Start optimization
res = mainSeilaplan(task, conf.project)
except Exception as e:
# Catch errors and print them to console
print(traceback.format_exc())
exit()
if not res:
print('ERROR: Error during optimization algorithm:')
print(task.exception)
exit()
return task.getStatusAsStr(), res['cableline'], res['optSTA'], \
res['force'], config.project.poles
def calculateFinalCableLine(conf: ConfigHandler):
""" This function only calculates the finale cable line. Pole locations
are read from a project file.
"""
print('Set up configuration and input values...')
success = conf.prepareForCalculation(runOptimization=False)
if not success:
print('ERROR: Error while preparing config data for optimization.')
exit()
print('Load pole setup from file...')
optiResults, _ = conf.prepareResultWithoutOptimization()
parameters = conf.params.getSimpleParameterDict()
print('Calculate precise cable line..')
cable, force, cable_possible = preciseCable(parameters, conf.project.poles,
optiResults['optSTA'])
resultQuality = ResultQuality.SuccessfulRerun
if not cable_possible:
resultQuality = ResultQuality.CableLiftsOff
return resultQuality, cable, optiResults['optSTA'], force, conf.project.poles
if __name__ == "__main__":
# This section controls which functions are executed and what project
# file is loaded.
# ################## ALL CONFIGURATIONS GO HERE ###########################
# Location of QGIS installation: the easiest way to find it for your system
# is to use the Scripting in the Python Console from within QGIS and look
# at the output from running: QgsApplication.prefixPath()
# on Ubuntu it's: "/usr"
qgis_install_location = r'C:\OSGeo4W\apps\qgis'
# Define the project file you want to load
savedProjectFile = r'N:\forema\FPS\Projekte_der_Gruppe\Seillinienplanung\2c_AP1_Projekte\Martin_Ammann\Ammann_Buriwand\Versuche_Leo\seilaplan_2020_24_08\Projekteinstellungen.txt'
# Define which functions the code should perform
# 'optimize': Run optimization algorithm to define pole positions and
# calculate final cable line.
# 'cableline': Only calculate final cable line, no optimization. Position
# of poles will be extracted from project file. Only works
# if the optimization algorithm was run before the project
# file was saved.
perform = 'optimize'
# Do you want do generate output data? (PDFs, CSV, ...)
createOutput = True # or: False
# Where do you want output to be saved?
outputLocation = '/home/pi/Seilaplan'
# #########################################################################
QgsApplication.setPrefixPath(qgis_install_location, True)
# Create a reference to the QgsApplication. Setting the
# second argument to False disables the GUI.
qgs = QgsApplication([], False)
# Load providers
qgs.initQgis()
# Load translations
translator = QTranslator()
translator.load(os.path.join(os.path.dirname(__file__), 'i18n', 'SeilaplanPlugin_de.qm'))
QCoreApplication.installTranslator(translator)
# Project settings are loaded
print('Load configuration from project file...')
config: ConfigHandler = ConfigHandler()
configLoaded = config.loadSettings(savedProjectFile)
if not configLoaded:
print(f"ERROR: Project file does not exist or cannot be loaded.")
qgs.exitQgis()
exit()
if perform == 'optimize':
# Run optimization including final cable line calculation
status, cableline, optSTA, forces, poles = optimizeCableLine(config)
else:
# Run calculation of cable line
status, cableline, optSTA, forces, poles = calculateFinalCableLine(config)
# Now that the cable line is calculated, analyze the ground clearance
groundClear = config.project.profile.updateProfileAnalysis(cableline)
cableline = {**cableline, **groundClear}
print(f"Optimization status: {status}")
# Output creation
#################
if createOutput and status != ResultQuality.LineNotComplete:
# If you dont want that a certain output type is created, comment out
# the related code block below
project = config.project
profile = project.profile
polesList = [pole for pole in poles.poles if pole['active']]
projName = project.generateProjectName()
outputLoc, projName_unique = createOutputFolder(outputLocation, projName)
updateWithCableCoordinates(cableline, project.points['A'],
project.azimut)
# Save project file
config.saveSettings(os.path.join(outputLoc, 'Projekteinstellungen.json'))
resultDict = {
'force': forces,
'cableline': cableline,
'optSTA_arr': [optSTA],
'duration': ['-', '-', '-'], # Dummy data for report
}
# Create short report PDF
####
print('Create short report...')
generateShortReport(config, resultDict, projName_unique, outputLoc)
# Create detailed report PDF
####
print('Create detailed report...')
reportText = generateReportText(config, resultDict, projName_unique)
generateReport(reportText, outputLoc)
# Create plot PDF
###
print('Create plot...')
plotSavePath = os.path.join(outputLoc, 'Diagramm.pdf')
width, height, ratio = calculatePlotDimensions(profile.di_disp, profile.zi_disp)
printPlot = AdjustmentPlot(None, width, height, 150, withBirdView=True, profilePlotRatio=ratio)
printPlot.initData(profile.di_disp, profile.zi_disp,
profile.peakLoc_x, profile.peakLoc_z,
profile.surveyPnts)
printPlot.updatePlot(poles.getAsArray(), cableline, True)
printPlot.layoutDiagrammForPrint(projName_unique, polesList, poles.direction)
imgPath = None
# Create Bird View
###
xlim, ylim = printPlot.createBirdView(polesList, project.azimut)
# Extract the map background
imgPath = extractMapBackground(outputLoc, xlim, ylim,
project.points['A'], project.azimut)
printPlot.addBackgroundMap(imgPath)
printPlot.exportPdf(plotSavePath)
os.remove(imgPath)
# Generate geo data
###
print('Create geo data...')
# Put geo data in separate sub folder
savePath = os.path.join(outputLoc, 'geodata')
os.makedirs(savePath)
epsg = project.heightSource.spatialRef
geodata = organizeDataForExport(polesList, cableline, profile)
# Generate shape files
writeGeodata(geodata, 'SHP', epsg, savePath)
# Generate coordinate tables (CSV)
###
print('Create csv files...')
generateCoordTable(cableline, profile, polesList, savePath)
print('STANDALONE finished')
qgs.exitQgis()