forked from anitagraser/TimeManager
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtimemanagercontrol.py
436 lines (367 loc) · 19.7 KB
/
timemanagercontrol.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
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
#!/usr/bin/python
# -*- coding: UTF-8 -*-
from datetime import datetime
from qgis.core import *
from timemanagerguicontrol import *
from timelayer import *
from timevectorlayer import *
from timelayermanager import *
from timemanagerprojecthandler import *
class TimeManagerControl(QObject):
"""Controls the logic behind the GUI. Signals are processed here."""
def __init__(self,iface):
"""initialize the plugin control"""
QObject.__init__(self)
self.iface = iface
#QMessageBox.information(self.iface.mainWindow(),'Debug','TimeManagerControl.__init__()')
def disableAnimationExport(self):
"""disable the animation export button"""
try:
self.guiControl.disableAnimationExport()
except AttributeError:
pass
def restoreDefaults(self):
"""restore plugin default settings"""
self.animationFrameLength = 2000 # default to 2000 milliseconds
self.playBackwards = False # play forwards by default
self.saveAnimation = False
self.currentMapTimePosition = datetime.now()
self.projectHandler.writeSetting('active',True)
self.setTimeFrameType('days')
self.setTimeFrameSize(1)
self.animationActivated = False
def initGui(self):
"""initialize the plugin dock"""
#QMessageBox.information(self.iface.mainWindow(),'Debug','TimeManagerControl.initGui()')
self.loopAnimation = False
self.saveAnimationPath = os.path.expanduser('~')
self.projectHandler = TimeManagerProjectHandler(self.iface)
self.timeLayerManager = TimeLayerManager(self.iface)
# QGIS iface connections
self.iface.projectRead.connect(self.readSettings)
self.iface.newProjectCreated.connect(self.restoreDefaults)
self.iface.newProjectCreated.connect(self.disableAnimationExport)
self.iface.mapCanvas().renderComplete.connect(self.waitAfterRenderComplete)
# establish connections to QgsMapLayerRegistry
QgsMapLayerRegistry.instance().layerWillBeRemoved.connect(self.timeLayerManager.removeTimeLayer)
QgsMapLayerRegistry.instance().removeAll.connect(self.timeLayerManager.clearTimeLayerList)
QgsMapLayerRegistry.instance().removeAll.connect(self.disableAnimationExport)
self.restoreDefaults()
self.guiControl = TimeManagerGuiControl(self.iface,self.timeLayerManager)
self.guiControl.showOptions.connect(self.showOptionsDialog)
self.guiControl.exportVideo.connect(self.exportVideo)
self.guiControl.toggleTime.connect(self.toggleTimeManagement)
self.guiControl.back.connect(self.stepBackward)
self.guiControl.forward.connect(self.stepForward)
self.guiControl.play.connect(self.toggleAnimation)
self.guiControl.signalCurrentTime.connect(self.setCurrentTimePosition)
self.guiControl.signalTimeFrameType.connect(self.setTimeFrameType)
self.guiControl.signalTimeFrameSize.connect(self.setTimeFrameSize)
self.guiControl.saveOptionsStart.connect(self.timeLayerManager.clearTimeLayerList)
self.guiControl.saveOptionsEnd.connect(self.writeSettings)
self.guiControl.saveOptionsEnd.connect(self.timeLayerManager.refresh) # sets the time restrictions again
self.guiControl.signalAnimationOptions.connect(self.setAnimationOptions)
self.guiControl.registerTimeLayer.connect(self.timeLayerManager.registerTimeLayer)
# create actions
# F8 button press - show time manager settings
self.actionShowSettings = QAction(u"Show Time Manager Settings", self.iface.mainWindow())
self.iface.registerMainWindowAction(self.actionShowSettings, "F8")
self.guiControl.addActionShowSettings(self.actionShowSettings)
self.actionShowSettings.triggered.connect(self.showOptionsDialog)
# establish connections to timeLayerManager
self.timeLayerManager.timeRestrictionsRefreshed.connect(self.guiControl.refreshTimeRestrictions)
self.timeLayerManager.projectTimeExtentsChanged.connect(self.guiControl.updateTimeExtents)
self.timeLayerManager.toggledManagement.connect(self.toggleOnOff)
self.timeLayerManager.lastLayerRemoved.connect(self.disableAnimationExport)
def setAnimationOptions(self,length,playBackwards,loopAnimation):
"""set length and play direction of the animation""" #animationFrameLength,playBackwards,loopAnimation
self.animationFrameLength = length
self.playBackwards = playBackwards
self.loopAnimation = loopAnimation
def showOptionsDialog(self):
"""show options dialog"""
self.stopAnimation()
self.guiControl.showOptionsDialog(self.timeLayerManager.getTimeLayerList(),self.animationFrameLength,self.playBackwards,self.loopAnimation)
def exportVideo(self):
"""export 'video' - currently only image sequence"""
self.saveAnimationPath = str(QFileDialog.getExistingDirectory (self.iface.mainWindow(),'Pick export destination',self.saveAnimationPath))
if self.saveAnimationPath:
self.saveAnimation = True
self.loopAnimation = False # on export looping has to be deactivated
self.toggleAnimation()
QMessageBox.information(self.iface.mainWindow(),'Export Video','Image sequence is being saved to '+self.saveAnimationPath+'.\n\nPlease wait until the process is finished.')
def unload(self):
"""unload the plugin"""
#QMessageBox.information(self.iface.mainWindow(),'Debug','TimeManagerControl.unload()')
self.timeLayerManager.deactivateTimeManagement()
self.iface.unregisterMainWindowAction(self.actionShowSettings)
self.guiControl.unload()
self.iface.projectRead.disconnect(self.readSettings)
self.iface.newProjectCreated.disconnect(self.restoreDefaults)
self.iface.newProjectCreated.disconnect(self.disableAnimationExport)
QgsMapLayerRegistry.instance().layerWillBeRemoved.disconnect(self.timeLayerManager.removeTimeLayer)
QgsMapLayerRegistry.instance().removeAll.disconnect(self.timeLayerManager.clearTimeLayerList)
QgsMapLayerRegistry.instance().removeAll.disconnect(self.disableAnimationExport)
def toggleAnimation(self):
"""toggle animation on/off"""
if self.animationActivated:
self.animationActivated = False
else:
self.animationActivated = True
self.startAnimation()
self.animationFrameCounter = 0
expectedNumberOfFrames = self.timeLayerManager.getFrameCount()
if expectedNumberOfFrames == 0: # will be zero if no layer is time managed
self.animationActivated = False
self.exportNameDigits = len(str(expectedNumberOfFrames))
def toggleOnOff(self,turnOn):
"""write plugin status (on/off) to project settings"""
if turnOn:
self.projectHandler.writeSetting('active',True)
#self.guiControl.showLabel = True
pass
else:
self.projectHandler.writeSetting('active',False)
#self.guiControl.showLabel = False
pass
self.guiControl.refreshMapCanvas('toggleOnOff')
def startAnimation(self):
"""kick-start the animation, afterwards the animation will run based on signal chains"""
self.waitAfterRenderComplete()
def waitAfterRenderComplete(self, painter=None):
"""when the map canvas signals renderComplete, wait defined millisec until next animation step"""
if self.saveAnimation: # make animation/export run as fast as possible
self.playAnimation(painter)
else:
QTimer.singleShot(self.animationFrameLength,self.playAnimation)
def playAnimation(self,painter=None):
"""play animation in map window"""
if not self.animationActivated:
return
# check if the end of the project time extents has been reached
projectTimeExtents = self.timeLayerManager.getProjectTimeExtents()
currentTime = self.timeLayerManager.getCurrentTimePosition()
if self.saveAnimation:
fileName = os.path.join(self.saveAnimationPath,"frame"+str(self.animationFrameCounter).zfill(self.exportNameDigits)+".png")
self.saveCurrentMap(fileName)
self.animationFrameCounter += 1
try:
if self.playBackwards:
if currentTime > projectTimeExtents[0]:
self.stepBackward()
else:
if self.loopAnimation:
self.resetAnimation()
else:
self.stopAnimation()
else:
if currentTime < projectTimeExtents[1]:
self.stepForward()
else:
if self.loopAnimation:
self.resetAnimation()
else:
self.stopAnimation()
except TypeError:
self.stopAnimation()
def saveCurrentMap(self,fileName):
"""saves the content of the map canvas to file"""
self.iface.mapCanvas().saveAsImage(fileName)
def stopAnimation(self):
"""stop the animation in case it's running"""
if self.saveAnimation:
QMessageBox.information(self.iface.mainWindow(),'Export finished','The export finished successfully!')
self.saveAnimation = False
self.animationActivated = False
self.guiControl.turnPlayButtonOff()
def resetAnimation(self):
"""reset animation to start over from the beginning"""
projectTimeExtents = self.timeLayerManager.getProjectTimeExtents()
self.setCurrentTimePosition(projectTimeExtents[0])
def toggleTimeManagement(self):
"""toggle time management on/off"""
self.stopAnimation()
self.timeLayerManager.toggleTimeManagement()
def stepBackward(self):
"""move one step backward in time"""
self.timeLayerManager.stepBackward()
self.writeSettings()
def stepForward(self):
"""move one step forward in time"""
self.timeLayerManager.stepForward()
self.writeSettings()
def getCurrentTimePosition(self):
"""get timeLayerManager's current time position"""
return self.timeLayerManager.getCurrentTimePosition()
def setCurrentTimePosition(self,timePosition):
"""set timeLayerManager's current time position"""
original = timePosition
if type(timePosition) == QDateTime:
# convert QDateTime to datetime :S
timePosition = datetime.strptime( str(timePosition.toString('yyyy-MM-dd hh:mm:ss.zzz')) ,"%Y-%m-%d %H:%M:%S.%f")
elif type(timePosition) == int or type(timePosition) == float:
timePosition = datetime.fromtimestamp(timePosition)
if timePosition == self.currentMapTimePosition:
return
self.currentMapTimePosition = timePosition
#QMessageBox.information(self.iface.mainWindow(),'Info','original = '+str(original)+' - timePosition = '+str(timePosition))
self.guiControl.refreshTimeRestrictions(timePosition,'setCurrentTimePosition')
self.timeLayerManager.setCurrentTimePosition(timePosition)
if self.timeLayerManager.hasActiveLayers() and self.timeLayerManager.isEnabled():
self.guiControl.refreshMapCanvas('setCurrentTimePosition'+str(timePosition))
#if self.timeLayerManager.hasActiveLayers():
# self.guiControl.showLabel = True
#else:
# self.guiControl.showLabel = False
def setTimeFrameType(self,timeFrameType):
"""set timeLayerManager's time frame type"""
self.timeLayerManager.setTimeFrameType(timeFrameType)
self.writeSettings()
if self.timeLayerManager.hasActiveLayers():
self.guiControl.refreshMapCanvas('setTimeFrameType')
def setTimeFrameSize(self,timeFrameSize):
"""set timeLayerManager's time frame size"""
self.timeLayerManager.setTimeFrameSize(timeFrameSize)
self.writeSettings()
if self.timeLayerManager.hasActiveLayers():
self.guiControl.refreshMapCanvas('setTimeFrameSize')
def writeSettings(self):
"""write all relevant settings to the project file """
(timeLayerManagerSettings,timeLayerList) = self.timeLayerManager.getSaveString()
if timeLayerManagerSettings:
settings= { 'animationFrameLength': self.animationFrameLength,
'playBackwards': self.playBackwards,
'loopAnimation': self.loopAnimation,
'timeLayerManager': timeLayerManagerSettings,
'timeLayerList': timeLayerList,
'currentMapTimePosition': self.currentMapTimePosition,
'timeFrameType': self.timeLayerManager.getTimeFrameType(),
'timeFrameSize': self.timeLayerManager.getTimeFrameSize() }
self.projectHandler.writeSettings(settings)
def readSettings(self):
"""load and restore settings from project file"""
# list of settings to restore
settings= { 'animationFrameLength': self.animationFrameLength,
'playBackwards': self.playBackwards,
'loopAnimation': self.loopAnimation,
'timeLayerManager': '',
'timeLayerList': [], #QStringList,
'currentMapTimePosition': self.currentMapTimePosition,
'timeFrameType': self.timeLayerManager.getTimeFrameType(),
'timeFrameSize': self.timeLayerManager.getTimeFrameSize(),
'active': True }
settings = self.projectHandler.readSettings(settings)
# list of restore functions and associated default values
functions = {
'currentMapTimePosition': (self.restoreSettingCurrentMapTimePosition,None), # this has to be first, because otherwise it might get over-written by other methods
'animationFrameLength': (self.restoreSettingAnimationFrameLength,1),
'playBackwards': (self.restoreSettingPlayBackwards,0),
'loopAnimation': (self.restoreSettingLoopAnimation,0),
'timeLayerManager': (self.restoreSettingTimeLayerManager,None),
'timeLayerList': (self.restoreTimeLayers,None),
'timeFrameType': (self.restoreSettingTimeFrameType,'days'),
'timeFrameSize': (self.restoreSettingTimeFrameSize,1),
'active': (self.restoreSettingActive,1)
}
savedTimePosition = datetime.fromtimestamp(1)
try: # save the timePosition first because it might get over-written by successive functions
savedTimePosition = datetime.fromtimestamp(settings['currentMapTimePosition'])
except KeyError:
pass
except TypeError:
pass
# now restore all settings
for setting,(func,value) in functions.items():
try:
value = settings[setting]
except KeyError:
pass
try:
func(value)
except TypeError:
QMessageBox.information(self.iface.mainWindow(),'Error','An error occured while loading: '+setting+'\nValue: '+str(value)+'\nType: '+str(type(value)))
# finally, set the currentMapTimePosition
if savedTimePosition:
#QMessageBox.information(self.iface.mainWindow(),'Info','savedTimePosition = '+str(savedTimePosition))
self.restoreSettingCurrentMapTimePosition(savedTimePosition)
def restoreSettingAnimationFrameLength(self,value):
"""restore animationFrameLength"""
if value:
self.animationFrameLength = value
def restoreSettingPlayBackwards(self,value):
"""restore playBackwards"""
if value:
self.playBackwards = value
def restoreSettingLoopAnimation(self,value):
"""restore loopAnimation"""
if value:
self.loopAnimation = value
def restoreSettingTimeLayerManager(self,value):
"""restore timeLayerManager"""
if value:
self.timeLayerManager.restoreFromSaveString(value)
def restoreTimeLayers(self,value):
"""restore all time layers"""
if value:
layerInfo = value
if len(layerInfo):
self.guiControl.enableAnimationExport()
for l in layerInfo: # for every layer entry
l = l.split(';')
layer = QgsMapLayerRegistry.instance().mapLayer(l[0]) # get the layer
if not layer:
break
# this should be a python class factory
if type(layer).__name__ == "QgsRasterLayer":
timeLayerClass = TimeRasterLayer # get the correct class to use
elif type(layer).__name__ == "QgsVectorLayer":
timeLayerClass = TimeVectorLayer
if timeLayerClass == TimeVectorLayer:
layer.setSubsetString(l[1]) # restore the original subset string, only available for vector layers!
startTimeAttribute=l[2]
endTimeAttribute=l[3]
isEnabled=l[4]
timeFormat=l[5]
try:
offset=l[6]
except IndexError: # old versions didn't have an offset option
offset=0
try: # here we use the previously determined class
timeLayer = timeLayerClass(layer,startTimeAttribute,endTimeAttribute,isEnabled,timeFormat,offset)
except InvalidTimeLayerError, e:
QMessageBox.information(self.iface.mainWindow(),'Error','An error occured while trying to add layer '+layer.name()+' to TimeManager.\n'+e.value)
return False
if not timeLayer:
break
self.timeLayerManager.registerTimeLayer(timeLayer)
self.guiControl.showLabel = True
self.guiControl.refreshMapCanvas('restoreTimeLayer')
return True
def restoreSettingCurrentMapTimePosition(self,value):
"""restore currentMapTimePosition"""
if value:
try:
self.setCurrentTimePosition(value)
except:
QMessageBox.information(self.iface.mainWindow(),'Error','An error occured in self.setCurrentTimePosition')
try:
self.guiControl.refreshTimeRestrictions(value,'readSettings')
except:
QMessageBox.information(self.iface.mainWindow(),'Error','An error occured in self.guiControl.refreshTimeRestrictions')
def restoreSettingTimeFrameType(self,value):
"""restore timeFrameType"""
if value:
self.guiControl.setTimeFrameType(value)
self.setTimeFrameType(value)
def restoreSettingTimeFrameSize(self,value):
"""restore timeFrameSize"""
if value:
self.guiControl.setTimeFrameSize(value)
def restoreSettingActive(self,value):
"""restore activity setting"""
if value:
self.timeLayerManager.activateTimeManagement()
self.guiControl.setActive(True)
else: # if the status indicates "off"
self.timeLayerManager.deactivateTimeManagement()
self.guiControl.setActive(False)