-
Notifications
You must be signed in to change notification settings - Fork 3
/
pp_videoplayer.py
813 lines (685 loc) · 35.3 KB
/
pp_videoplayer.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
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
# -*- coding: utf-8 -*-
import os
from pp_omxdriver import OMXDriver
from pp_player import Player
from pp_utils import parse_rectangle
from pp_displaymanager import DisplayManager
import copy
import numpy as np
class VideoPlayer(Player):
"""
plays a track using omxplayer
_init_ iniitalises state and checks resources are available.
use the returned instance reference in all other calls.
At the end of the path (when closed) do not re-use, make instance= None and start again.
States - 'initialised' when done successfully
Initialisation is immediate so just returns with error code.
"""
debug = False
#debug = True
def __init__(self,
show_id,
showlist,
root,
canvas,
show_params,
track_params ,
pp_dir,
pp_home,
pp_profile,
end_callback,
command_callback):
# initialise items common to all players
Player.__init__( self,
show_id,
showlist,
root,
canvas,
show_params,
track_params ,
pp_dir,
pp_home,
pp_profile,
end_callback,
command_callback)
# print ' !!!!!!!!!!!videoplayer init'
self.mon.trace(self,'')
self.dm=DisplayManager()
# get player parameters
if self.track_params['omx-audio'] != "":
self.omx_audio= self.track_params['omx-audio']
else:
self.omx_audio= self.show_params['omx-audio']
if self.omx_audio != "": self.omx_audio= "-o "+ self.omx_audio
self.omx_max_volume_text=self.track_params['omx-max-volume']
if self.omx_max_volume_text != "":
self.omx_max_volume= int(self.omx_max_volume_text)
else:
self.omx_max_volume=0
if self.track_params['omx-volume'] != "":
self.omx_volume= self.track_params['omx-volume']
else:
self.omx_volume= self.show_params['omx-volume']
if self.omx_volume != "":
self.omx_volume= int(self.omx_volume)
else:
self.omx_volume=0
self.omx_volume=min(self.omx_volume,self.omx_max_volume)
if self.track_params['omx-window'] != '':
self.omx_window= self.track_params['omx-window']
else:
self.omx_window= self.show_params['omx-window']
if self.track_params['omx-other-options'] != '':
self.omx_other_options= self.track_params['omx-other-options']
else:
self.omx_other_options= self.show_params['omx-other-options']
if self.track_params['freeze-at-start'] != '':
self.freeze_at_start= self.track_params['freeze-at-start']
else:
self.freeze_at_start= self.show_params['freeze-at-start']
if self.track_params['freeze-at-end'] != '':
freeze_at_end_text= self.track_params['freeze-at-end']
else:
freeze_at_end_text= self.show_params['freeze-at-end']
if freeze_at_end_text == 'yes':
self.freeze_at_end_required=True
else:
self.freeze_at_end_required=False
if self.track_params['seamless-loop'] == 'yes':
self.seamless_loop=' --loop '
else:
self.seamless_loop=''
if self.track_params['pause-timeout'] != '':
pause_timeout_text= self.track_params['pause-timeout']
else:
pause_timeout_text= self.show_params['pause-timeout']
if pause_timeout_text.isdigit():
self.pause_timeout= int(pause_timeout_text)
else:
self.pause_timeout=0
# initialise video playing state and signals
self.quit_signal=False
self.unload_signal=False
self.play_state='initialised'
self.frozen_at_end=False
self.pause_timer=None
# LOAD - creates and omxplayer instance, loads a track and then pause
def load(self,track,loaded_callback,enable_menu):
# instantiate arguments
self.track=track
self.loaded_callback=loaded_callback #callback when loaded
# print '!!!!!!!!!!! videoplayer load',self.track
self.mon.log(self,"Load track received from show Id: "+ str(self.show_id) + ' ' +self.track)
self.mon.trace(self,'')
# do common bits of load
Player.pre_load(self)
# set up video display
if self.track_params['display-name'] != "":
video_display_name = self.track_params['display-name']
else:
video_display_name = self.show_canvas_display_name
status,message,self.omx_display_id=self.dm.id_of_display(video_display_name)
if status == 'error':
self.mon.err(self,message)
self.play_state='load-failed'
if self.loaded_callback is not None:
self.loaded_callback('error','Display not connected: '+ video_display_name)
return
# set up video window and calc orientation
status,message,command,has_window,x1,y1,x2,y2= self.parse_video_window(self.omx_window,self.omx_display_id)
if status == 'error':
self.mon.err(self,'omx window error: ' + message + ' in ' + self.omx_window)
self.play_state='load-failed'
if self.loaded_callback is not None:
self.loaded_callback('error','omx window error: ' + message + ' in ' + self.omx_window)
return
else:
if has_window is True:
self.omx_window_processed= '--win " '+ str(x1) + ' ' + str(y1) + ' ' + str(x2) + ' ' + str(y2) + ' " '
else:
self.omx_window_processed=self.omx_aspect_mode
# load the plugin, this may modify self.track and enable the plugin drawign to canvas
if self.track_params['plugin'] != '':
status,message=self.load_plugin()
if status == 'error':
self.mon.err(self,message)
self.play_state='load-failed'
if self.loaded_callback is not None:
self.loaded_callback('error',message)
return
# load the images and text
status,message=self.load_x_content(enable_menu)
if status == 'error':
self.mon.err(self,message)
self.play_state='load-failed'
if self.loaded_callback is not None:
self.loaded_callback('error',message)
return
if not (track[0:3] in ('udp','tcp') or track[0:4] in ('rtsp','rtmp')):
if not os.path.exists(track):
self.mon.err(self,"Track file not found: "+ track)
self.play_state='load-failed'
if self.loaded_callback is not None:
self.loaded_callback('error','track file not found: '+ track)
return
self.omx=OMXDriver(self.canvas,self.pp_dir)
self.start_state_machine_load(self.track)
# SHOW - show a track
def show(self,ready_callback,finished_callback,closed_callback):
# print "!!!! videoplayer show"
# instantiate arguments
self.ready_callback=ready_callback # callback when ready to show video
self.finished_callback=finished_callback # callback when finished showing
self.closed_callback=closed_callback
self.mon.trace(self,'')
# do animation at start etc.
Player.pre_show(self)
# start show state machine
self.start_state_machine_show()
# UNLOAD - abort a load when omplayer is loading or loaded
def unload(self):
self.mon.trace(self,'')
self.mon.log(self,">unload received from show Id: "+ str(self.show_id))
self.start_state_machine_unload()
# CLOSE - quits omxplayer from 'pause at end' state
def close(self,closed_callback):
self.mon.trace(self,'')
self.mon.log(self,">close received from show Id: "+ str(self.show_id))
self.closed_callback=closed_callback
self.start_state_machine_close()
def input_pressed(self,symbol):
if symbol[0:4] == 'omx-':
if symbol[0:5] in ('omx-+','omx--','omx-='):
if symbol[4] in ('+','='):
self.inc_volume()
else:
self.dec_volume()
else:
self.control(symbol[4])
elif symbol == 'inc-volume':
self.inc_volume()
elif symbol == 'dec-volume':
self.dec_volume()
elif symbol == 'pause':
self.pause()
elif symbol == 'go':
self.go()
elif symbol == 'unmute':
self.unmute()
elif symbol == 'mute':
self.mute()
elif symbol == 'pause-on':
self.pause_on()
elif symbol == 'pause-off':
self.pause_off()
elif symbol == 'stop':
self.stop()
# respond to normal stop
def stop(self):
self.mon.log(self,">stop received from show Id: "+ str(self.show_id))
# cancel the pause timer
if self.pause_timer != None:
self.canvas.after_cancel(self.pause_timer)
self.pause_timer=None
# send signal to freeze the track - causes either pause or quit depends on freeze at end
if self.freeze_at_end_required is True:
if self.play_state == 'showing' and self.frozen_at_end is False:
self.frozen_at_end=True
# pause the track
self.omx.pause('freeze at end from user stop')
self.quit_signal=True
# and return to show so it can end the track and the video in track ready callback
## if self.finished_callback is not None:
## # print 'finished from stop'
## self.finished_callback('pause_at_end','pause at end')
else:
self.mon.log(self,"!<stop rejected")
else:
# freeze not required and its showing just stop the video
if self.play_state=='showing':
self.quit_signal=True
else:
self.mon.log(self,"!<stop rejected")
def inc_volume(self):
self.mon.log(self,">inc-volume received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done':
self.omx.inc_volume()
return True
else:
self.mon.log(self,"!<inc-volume rejected " + self.play_state)
return False
def dec_volume(self):
self.mon.log(self,">dec-volume received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done':
self.omx.dec_volume()
return True
else:
self.mon.log(self,"!<dec-volume rejected " + self.play_state)
return False
def mute(self):
self.mon.log(self,">mute received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done':
self.omx.mute()
return True
else:
self.mon.log(self,"!<mute rejected " + self.play_state)
return False
def unmute(self):
self.mon.log(self,">unmute received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done':
self.omx.unmute()
return True
else:
self.mon.log(self,"!<unmute rejected " + self.play_state)
return False
# toggle pause
def pause(self):
self.mon.log(self,">toggle pause received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done':
self.omx.toggle_pause('user')
if self.omx.paused is True and self.pause_timeout>0:
# kick off the pause teimeout timer
self.pause_timer=self.canvas.after(self.pause_timeout*1000,self.pause_timeout_callback)
else:
# cancel the pause timer
if self.pause_timer != None:
self.canvas.after_cancel(self.pause_timer)
self.pause_timer=None
return True
else:
self.mon.log(self,"!<pause rejected " + self.play_state)
return False
def pause_timeout_callback(self):
self.pause_off()
self.pause_timer=None
# pause on
def pause_on(self):
self.mon.log(self,">pause on received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done':
self.omx.pause_on()
if self.omx.paused is True and self.pause_timeout>0:
# kick off the pause teimeout timer
self.pause_timer=self.canvas.after(self.pause_timeout*1000,self.pause_timeout_callback)
return True
else:
self.mon.log(self,"!<pause on rejected " + self.play_state)
return False
# pause off
def pause_off(self):
self.mon.log(self,">pause off received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done':
self.omx.pause_off()
if self.omx.paused is False:
# cancel the pause timer
if self.pause_timer != None:
self.canvas.after_cancel(self.pause_timer)
self.pause_timer=None
return True
else:
self.mon.log(self,"!<pause off rejected " + self.play_state)
return False
# go after freeze at start
def go(self):
self.mon.log(self,">go received from show Id: "+ str(self.show_id))
if self.play_state == 'showing' and self.omx.paused_at_start == 'True':
self.omx.go()
return True
else:
self.mon.log(self,"!<go rejected " + self.play_state)
return False
# other control when playing
def control(self,char):
if self.play_state == 'showing' and self.frozen_at_end is False and self.omx.paused_at_start == 'done' and char not in ('q'):
self.mon.log(self,"> send control to omx: "+ char)
self.omx.control(char)
return True
else:
self.mon.log(self,"!<control rejected")
return False
# ***********************
# track showing state machine
# **********************
"""
STATES OF STATE MACHINE
Threre are ongoing states and states that are set just before callback
>init - Create an instance of the class
<On return - state = initialised - - init has been completed, do not generate errors here
>load
Fatal errors should be detected in load. If so loaded_callback is called with 'load-failed'
Ongoing - state=loading - load called, waiting for load to complete
< loaded_callback with status = normal
state=loaded - load has completed and video paused before first frame
<loaded_callback with status=error
state= load-failed - omxplayer process has been killed because of failure to load
On getting the loaded_callback with status=normal the track can be shown using show
Duration obtained from track should always cause pause_at_end. if not please tell me as the fudge factor may need adjusting.
>show
show assumes a track has been loaded and is paused.
Ongoing - state=showing - video is showing
<finished_callback with status = pause_at_end
state=showing but frozen_at_end is True
<closed_callback with status= normal
state = closed - video has ended omxplayer has terminated.
On getting finished_callback with status=pause_at end a new track can be shown and then use close to quit the video when new track is ready
On getting closed_callback with status= nice_day omxplayer closing should not be attempted as it is already closed
Do not generate user errors in Show. Only generate system erros such as illegal state abd then use end()
>close
Ongoing state - closing - omxplayer processes are dying due to quit sent
<closed_callback with status= normal - omxplayer is dead, can close the track instance.
>unload
Ongoing states - start_unload and unloading - omxplayer processes are dying due to quit sent.
when unloading is complete state=unloaded
I have not added a callback to unload. its easy to add one if you want.
closed is needed because wait_for end in pp_show polls for closed and does not use closed_callback
"""
def start_state_machine_load(self,track):
self.track=track
# initialise all the state machine variables
self.loading_count=0 # initialise loading timeout counter
self.play_state='loading'
# load the selected track
# options= ' ' + self.omx_audio+ ' --vol -6000 ' + self.omx_window_processed + ' ' + self.seamless_loop + ' ' + self.omx_other_options +" "
options= ' --display '+ str(self.omx_display_id) + ' --no-osd ' + self.omx_audio+ ' --vol -6000 ' + self.omx_window_processed + self.omx_rotate +' ' + self.seamless_loop + ' ' + self.omx_other_options +" "
self.omx.load(track,self.freeze_at_start,options,self.mon.pretty_inst(self),self.omx_volume,self.omx_max_volume)
# self.mon.log (self,'Send load command track '+ self.track + 'with options ' + options + 'from show Id: '+ str(self.show_id))
# print 'omx.load started ',self.track
# and start polling for state changes
self.tick_timer=self.canvas.after(50, self.load_state_machine)
def start_state_machine_unload(self):
# print ('videoplayer - starting unload',self.play_state)
if self.play_state in('closed','initialised','unloaded'):
# omxplayer already closed
self.play_state='unloaded'
# print ' closed so no need to unload'
else:
if self.play_state == 'loaded':
# load already complete so set unload signal and kick off state machine
self.play_state='start_unload'
self.unloading_count=0
self.unload_signal=True
self.tick_timer=self.canvas.after(50, self.load_state_machine)
elif self.play_state == 'loading':
# wait for load to complete before unloading - ???? must do this because does not respond to quit when loading
# state machine will go to start_unloading state and stop omxplayer
self.unload_signal=True
else:
self.mon.err(self,'illegal state in unload method: ' + self.play_state)
self.end('error','illegal state in unload method: '+ self.play_state)
def start_state_machine_show(self):
if self.play_state == 'loaded':
# print '\nstart show state machine ' + self.play_state
self.play_state='showing'
self.freeze_signal=False # signal that user has pressed stop
self.must_quit_signal=False
# show the track and content
self.omx.show(self.freeze_at_end_required)
self.mon.log (self,'>showing track from show Id: '+ str(self.show_id))
# and start polling for state changes
# print 'start show state machine show'
self.tick_timer=self.canvas.after(0, self.show_state_machine)
# race condition don't start state machine as unload in progress
elif self.play_state == 'start_unload':
pass
else:
self.mon.fatal(self,'illegal state in show method ' + self.play_state)
self.play_state='show-failed'
if self.finished_callback is not None:
self.finished_callback('error','illegal state in show method: ' + self.play_state)
def start_state_machine_close(self):
self.quit_signal=True
# print 'start close state machine close'
self.tick_timer=self.canvas.after(0, self.show_state_machine)
def load_state_machine(self):
# print 'vidoeplayer state is'+self.play_state
if self.play_state == 'loading':
# if omxdriver says loaded then can finish normally
# self.mon.log(self," State machine: " + self.play_state)
if self.omx.end_play_signal is True:
# got nice day before the first timestamp
self.mon.warn(self,self.track)
self.mon.warn(self,"loading - omxplayer ended before starting track with reason: " + self.omx.end_play_reason + ' at ' +str(self.omx.video_position))
self.omx.kill()
self.mon.err(self,'omxplayer ended before loading track')
self.play_state = 'load-failed'
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
if self.loaded_callback is not None:
self.loaded_callback('error','omxplayer ended before loading track')
else:
# end play signal false - continue waiting for first timestamp
self.loading_count+=1
# video has loaded
if self.omx.start_play_signal is True:
self.mon.log(self,"Loading complete from show Id: "+ str(self.show_id)+ ' ' +self.track)
self.mon.log(self,'Got video duration from track, frezing at: '+ str(self.omx.duration)+ ' microsecs.')
if self.unload_signal is True:
# print('unload sig=true state= start_unload')
# need to unload, kick off state machine in 'start_unload' state
self.play_state='start_unload'
self.unloading_count=0
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
self.tick_timer=self.canvas.after(200, self.load_state_machine)
else:
self.play_state = 'loaded'
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
if self.loaded_callback is not None:
# print 'callback when loaded'
self.loaded_callback('normal','video loaded')
else:
# start play signal false - continue to wait
if self.loading_count>400: #40 seconds
# deal with omxplayer crashing while loading and hence not receive start_play_signal
self.mon.warn(self,self.track)
self.mon.warn(self,"loading - videoplayer counted out: " + self.omx.end_play_reason + ' at ' + str(self.omx.video_position))
self.omx.kill()
self.mon.warn(self,'videoplayer counted out when loading track ')
self.play_state = 'load-failed'
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
if self.loaded_callback is not None:
self.loaded_callback('error','omxplayer counted out when loading track')
else:
self.tick_timer=self.canvas.after(100, self.load_state_machine) #200
elif self.play_state == 'start_unload':
# omxplayer reports it is terminating
# self.mon.log(self," State machine: " + self.play_state)
# deal with unload signal
if self.unload_signal is True:
self.unload_signal=False
self.omx.stop()
if self.omx.end_play_signal is True:
self.omx.end_play_signal=False
self.mon.log(self," <end play signal received with reason: " + self.omx.end_play_reason + ' at: ' + str(self.omx.video_position))
# omxplayer has been closed
if self.omx.end_play_reason == 'nice_day':
# no problem with omxplayer
self.play_state='unloading'
self.unloading_count=0
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
self.tick_timer=self.canvas.after(50, self.load_state_machine)
else:
# unexpected reason
self.mon.err(self,'unexpected reason at unload: '+self.omx.end_play_reason)
self.end('error','unexpected reason at unload: '+self.omx.end_play_reason)
else:
# end play signal false
self.tick_timer=self.canvas.after(50, self.load_state_machine)
elif self.play_state == 'unloading':
# wait for unloading to complete
self.mon.log(self," State machine: " + self.play_state)
# if spawned process has closed can change to closed state
if self.omx.is_running() is False:
self.mon.log(self," <omx process is dead")
self.play_state='unloaded'
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
else:
# process still running
self.unloading_count+=1
if self.unloading_count>10:
# deal with omxplayer not terminating at the end of a track
self.mon.warn(self,self.track)
self.mon.warn(self," <unloading - omxplayer failed to close at: " + str(self.omx.video_position))
self.mon.warn(self,'omxplayer should now be killed ')
self.omx.kill()
self.play_state='unloaded'
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
else:
self.tick_timer=self.canvas.after(200, self.load_state_machine)
else:
self.mon.err(self,'illegal state in load state machine: ' + self.play_state)
self.end('error','load state machine in illegal state: '+ self.play_state)
def show_state_machine(self):
# print self.play_state
# if self.play_state != 'showing': print 'show state is '+self.play_state
if self.play_state == 'showing':
# service any queued stop signals by sending quit to omxplayer
# self.mon.log(self," State machine: " + self.play_state)
if self.quit_signal is True:
self.quit_signal=False
self.mon.log(self," quit video - Send stop to omxdriver")
self.omx.stop()
self.tick_timer=self.canvas.after(50, self.show_state_machine)
# omxplayer reports it is terminating
elif self.omx.end_play_signal is True:
self.omx.end_play_signal=False
self.mon.log(self,"end play signal received with reason: " + self.omx.end_play_reason + ' at: ' + str(self.omx.video_position))
# paused at end of track so return so calling prog can release the pause
if self.omx.end_play_reason == 'pause_at_end':
self.frozen_at_end=True
if self.finished_callback is not None:
self.finished_callback('pause_at_end','pause at end')
elif self.omx.end_play_reason == 'nice_day':
# no problem with omxplayer
self.play_state='closing'
self.closing_count=0
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
self.tick_timer=self.canvas.after(50, self.show_state_machine)
else:
# unexpected reason
self.mon.err(self,'unexpected reason at end of show '+self.omx.end_play_reason)
self.play_state='show-failed'
if self.finished_callback is not None:
self.finished_callback('error','unexpected reason at end of show: '+ self.omx.end_play_reason)
else:
# nothing to do just try again
# print 'showing - try again'
self.tick_timer=self.canvas.after(50, self.show_state_machine)
elif self.play_state == 'closing':
# wait for closing to complete
self.mon.log(self," State machine: " + self.play_state)
if self.omx.is_running() is False:
# if spawned process has closed can change to closed state
self.mon.log(self," <omx process is dead")
self.play_state = 'closed'
# print 'process dead going to closed'
self.omx=None
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
if self.closed_callback is not None:
self.closed_callback('normal','omxplayer closed')
else:
# process still running
self.closing_count+=1
# print 'closing - waiting for process to die',self.closing_count
if self.closing_count>10:
# deal with omxplayer not terminating at the end of a track
# self.mon.warn(self,self.track)
# self.mon.warn(self,"omxplayer failed to close at: " + str(self.omx.video_position))
self.mon.warn(self,'failed to close - omxplayer now being killed with SIGINT')
self.omx.kill()
# print 'closing - precess will not die so ita been killed with SIGINT'
self.play_state = 'closed'
self.omx=None
self.mon.log(self," Entering state : " + self.play_state + ' from show Id: '+ str(self.show_id))
if self.closed_callback is not None:
self.closed_callback('normal','closed omxplayer after sigint')
else:
self.tick_timer=self.canvas.after(200, self.show_state_machine)
elif self.play_state=='closed':
# needed because wait_for_end polls the state and does not use callback
self.mon.log(self," State machine: " + self.play_state)
self.tick_timer=self.canvas.after(200, self.show_state_machine)
else:
self.mon.err(self,'unknown state in show/close state machine ' + self.play_state)
self.play_state='show-failed'
if self.finished_callback is not None:
self.finished_callback('error','show state machine in unknown state: '+ self.play_state)
def parse_video_window(self,line,display_id):
# model other than 4 video is rotated by hdmi_display_rotate in config.txt
if self.dm.model_of_pi() == 4:
rotation= self.dm.real_display_orientation(self.omx_display_id)
else:
rotation = 'normal'
if rotation == 'normal':
self.omx_rotate=''
elif rotation == 'right':
self.omx_rotate= ' --orientation 90 '
elif rotation =='left':
self.omx_rotate = ' --orientation 270 '
else:
#inverted
self.omx_rotate = ' --orientation 180 '
fields = line.split()
# check there is a command field
if len(fields) < 1:
return 'error','no type field: '+line,'',False,0,0,0,0
# deal with types which have no paramters
if fields[0] in ('original','letterbox','fill','default','stretch'):
if len(fields) != 1:
return 'error','number of fields for original: '+line,'',False,0,0,0,0
if fields[0] in ('letterbox','fill','stretch'):
self.omx_aspect_mode = ' --aspect-mode '+fields[0]
else:
self.omx_aspect_mode=''
return 'normal','',fields[0],False,0,0,0,0
# deal with warp which has 0 or 1 or 4 parameters (1 is x+y+w*h)
# check basic syntax
if fields[0] != 'warp':
return 'error','not a valid type: '+fields[0],'',False,0,0,0,0
if len(fields) not in (1,2,5):
return 'error','wrong number of coordinates for warp: '+ line,'',False,0,0,0,0
# deal with window coordinates, just warp
if len(fields) == 1:
has_window=True
x1=self.show_canvas_x1
y1=self.show_canvas_y1
x2=self.show_canvas_x2
y2=self.show_canvas_y2
else:
# window is specified warp + dimesions etc.
status,message,x1,y1,x2,y2=parse_rectangle(' '.join(fields[1:]))
if status == 'error':
return 'error',message,'',False,0,0,0,0
has_window=True
x1_res,y1_res,x2_res,y2_res=self.transform_for_rotation(display_id,rotation,x1,y1,x2,y2)
return 'normal','',fields[0],has_window,x1_res,y1_res,x2_res,y2_res
def transform_for_rotation(self,display_id,rotation,x1,y1,x2,y2):
# adjust rotation of video for display rotation
video_width=x2-x1
video_height=y2-y1
display_width,display_height=self.dm.real_display_dimensions(display_id)
if rotation =='right':
x1_res=display_height-video_height -y1
x2_res=display_height -y1
y1_res= y1
y2_res=video_width +y1
elif rotation=='normal':
x1_res=x1
y1_res=y1
x2_res=x2
y2_res=y2
elif rotation =='left':
x1_res= y1
x2_res=video_height +y1
y1_res= display_width-video_width-x1
y2_res= display_width-x1
else:
# inverted
x2_res= display_width -x1
x1_res=display_width-video_width -x1
y2_res= display_height-y1
y1_res= display_height-video_height-y1
if VideoPlayer.debug:
print ('\nWarp calculation for Display Id',display_id,rotation)
print ('Video Window',x1,y1,x2,y2)
print ('video width/height', video_width,video_height)
print ('display width/height',display_width,display_height)
print ('Translated Window',x1_res,y1_res,x2_res,y2_res)
return x1_res,y1_res,x2_res,y2_res