-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathPictureFrame2020.py
671 lines (627 loc) · 28.5 KB
/
PictureFrame2020.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
#!/usr/bin/python
from __future__ import absolute_import, division, print_function, unicode_literals
''' Simplified slideshow system using ImageSprite and without threading for background
loading of images (so may show delay for v large images).
Also has a minimal use of PointText and TextBlock system with reduced codepoints
and reduced grid_size to give better resolution for large characters.
Also shows a simple use of MQTT to control the slideshow parameters remotely
see http://pi3d.github.io/html/FAQ.html and https://www.thedigitalpictureframe.com/control-your-digital-picture-frame-with-home-assistents-wifi-presence-detection-and-mqtt/
and https://www.cloudmqtt.com/plans.html
USING exif info to rotate images
ESC to quit, 's' to reverse, any other key to move on one.
'''
import os
import time
import random
import math
import demo
import pi3d
import locale
import subprocess
import numpy as np
from pi3d.Texture import MAX_SIZE
from PIL import Image, ExifTags, ImageFilter # these are needed for getting exif data from images
import PictureFrame2020config as config
class Pic:
def __init__(self, fname, orientation=1, mtime=None, dt=None, fdt=None, location="", aspect=1.5):
self.fname = fname
self.orientation = orientation
self.mtime = mtime
self.dt = dt
self.fdt = fdt
self.location = location
self.aspect = aspect
self.shown_with = None # set to pic_num of image this was paired with
try:
locale.setlocale(locale.LC_TIME, config.LOCALE)
except:
print("error trying to set local to {}".format(config.LOCALE))
#####################################################
# these variables can be altered using MQTT messaging
#####################################################
time_delay = config.TIME_DELAY
fade_time = config.FADE_TIME
shuffle = config.SHUFFLE
subdirectory = config.SUBDIRECTORY
date_from = None
date_to = None
quit = False
paused = False # NB must be set to True *only* after the first iteration of the show!
#####################################################
# only alter below here if you're keen to experiment!
#####################################################
if config.KENBURNS:
kb_up = True
config.FIT = False
config.BLUR_EDGES = False
if config.BLUR_ZOOM < 1.0:
config.BLUR_ZOOM = 1.0
delta_alpha = 1.0 / (config.FPS * fade_time) # delta alpha
last_file_change = 0.0 # holds last change time in directory structure
next_check_tm = time.time() + config.CHECK_DIR_TM # check if new file or directory every n seconds
#####################################################
# some functions to tidy subsequent code
#####################################################
# Concatenate the specified images horizontally. Clip the taller
# image to the height of the shorter image.
def create_image_pair(im1, im2):
sep = 8 # separation between the images
# scale widest image to same width as narrower to avoid drastic cropping on mismatched images
if im1.width > im2.width:
im1 = im1.resize((im2.width, int(im1.height * im2.width / im1.width)))
else:
im2 = im2.resize((im1.width, int(im2.height * im1.width / im2.width)))
dst = Image.new('RGB', (im1.width + im2.width + sep, min(im1.height, im2.height)))
dst.paste(im1, (0, 0))
dst.paste(im2, (im1.width + sep, 0))
return dst
def orientate_image(im, orientation):
if orientation == 2:
im = im.transpose(Image.FLIP_LEFT_RIGHT)
elif orientation == 3:
im = im.transpose(Image.ROTATE_180) # rotations are clockwise
elif orientation == 4:
im = im.transpose(Image.FLIP_TOP_BOTTOM)
elif orientation == 5:
im = im.transpose(Image.FLIP_LEFT_RIGHT).transpose(Image.ROTATE_270)
elif orientation == 6:
im = im.transpose(Image.ROTATE_270)
elif orientation == 7:
im = im.transpose(Image.FLIP_LEFT_RIGHT).transpose(Image.ROTATE_90)
elif orientation == 8:
im = im.transpose(Image.ROTATE_90)
return im
def tex_load(pic_num, iFiles, size=None):
global date_from, date_to, next_pic_num
if type(pic_num) is int:
#fname = iFiles[pic_num][0]
#orientation = iFiles[pic_num][1]
fname = iFiles[pic_num].fname
orientation = iFiles[pic_num].orientation
if iFiles[pic_num].shown_with is not None:
return None # this image already show this round so skip
else: # allow file name to be passed to this function ie for missing file image
fname = pic_num
orientation = 1
try:
ext = os.path.splitext(fname)[1].lower()
if ext in ('.heif','.heic'):
im = convert_heif(fname)
else:
im = Image.open(fname)
if config.DELAY_EXIF and type(pic_num) is int: # don't do this if passed a file name
if iFiles[pic_num].dt is None or iFiles[pic_num].fdt is None: # dt and fdt set to None before exif read
(orientation, dt, fdt, location, aspect) = get_exif_info(fname, im)
iFiles[pic_num].orientation = orientation
iFiles[pic_num].dt = dt
iFiles[pic_num].fdt = fdt
iFiles[pic_num].location = location
iFiles[pic_num].aspect = aspect
if date_from is not None:
if dt < time.mktime(date_from + (0, 0, 0, 0, 0, 0)):
return None
if date_to is not None:
if dt > time.mktime(date_to + (0, 0, 0, 0, 0, 0)):
return None
# If PORTRAIT_PAIRS active and this is a portrait pic, try to find another one to pair it with
if config.PORTRAIT_PAIRS and iFiles[pic_num].aspect < 1.0:
im2 = None
# Search the whole list for another portrait image, starting with the "next"
# assuming previous images in sequence have already been shown
# TODO poss very time consuming to call get_exif_info
# TODO back and next will bring up different image combinations, or maybe just fail
if pic_num < len(iFiles) - 1: # i.e can't do this on the last image in list
for f_rec in iFiles[pic_num + 1:]:
if f_rec.dt is None or f_rec.fdt is None: # dt and fdt set to None before exif read
(f_orientation, f_dt, f_fdt, f_location, f_aspect) = get_exif_info(f_rec.fname)
f_rec.orientation = f_orientation
f_rec.dt = f_dt
f_rec.fdt = f_fdt
f_rec.location = f_location
f_rec.aspect = f_aspect
if f_rec.aspect < 1.0 and f_rec.shown_with is None:
im2 = Image.open(f_rec.fname)
f_rec.shown_with = pic_num
break
if im2 is not None:
if orientation > 1:
im = orientate_image(im, orientation)
if f_rec.orientation > 1:
im2 = orientate_image(im2, f_rec.orientation)
im = create_image_pair(im, im2)
orientation = 1
(w, h) = im.size
max_dimension = MAX_SIZE # TODO changing MAX_SIZE causes serious crash on linux laptop!
if not config.AUTO_RESIZE: # turned off for 4K display - will cause issues on RPi before v4
max_dimension = 3840 # TODO check if mipmapping should be turned off with this setting.
if w > max_dimension:
im = im.resize((max_dimension, int(h * max_dimension / w)), resample=Image.BICUBIC)
elif h > max_dimension:
im = im.resize((int(w * max_dimension / h), max_dimension), resample=Image.BICUBIC)
if orientation > 1:
im = orientate_image(im, orientation)
if config.BLUR_EDGES and size is not None:
wh_rat = (size[0] * im.height) / (size[1] * im.width)
if abs(wh_rat - 1.0) > 0.01: # make a blurred background
(sc_b, sc_f) = (size[1] / im.height, size[0] / im.width)
if wh_rat > 1.0:
(sc_b, sc_f) = (sc_f, sc_b) # swap round
(w, h) = (round(size[0] / sc_b / config.BLUR_ZOOM), round(size[1] / sc_b / config.BLUR_ZOOM))
(x, y) = (round(0.5 * (im.width - w)), round(0.5 * (im.height - h)))
box = (x, y, x + w, y + h)
blr_sz = (int(x * 512 / size[0]) for x in size)
im_b = im.resize(size, resample=0, box=box).resize(blr_sz)
im_b = im_b.filter(ImageFilter.GaussianBlur(config.BLUR_AMOUNT))
im_b = im_b.resize(size, resample=Image.BICUBIC)
im_b.putalpha(round(255 * config.EDGE_ALPHA)) # to apply the same EDGE_ALPHA as the no blur method.
im = im.resize((int(x * sc_f) for x in im.size), resample=Image.BICUBIC)
"""resize can use Image.LANCZOS (alias for Image.ANTIALIAS) for resampling
for better rendering of high-contranst diagonal lines. NB downscaled large
images are rescaled near the start of this try block if w or h > max_dimension
so those lines might need changing too.
"""
im_b.paste(im, box=(round(0.5 * (im_b.width - im.width)),
round(0.5 * (im_b.height - im.height))))
im = im_b # have to do this as paste applies in place
tex = pi3d.Texture(im, blend=True, m_repeat=True, automatic_resize=config.AUTO_RESIZE,
free_after_load=True)
#tex = pi3d.Texture(im, blend=True, m_repeat=True, automatic_resize=config.AUTO_RESIZE,
# mipmap=config.AUTO_RESIZE, free_after_load=True) # poss try this if still some artifacts with full resolution
except Exception as e:
if config.VERBOSE:
print('''Couldn't load file {} giving error: {}'''.format(fname, e))
tex = None
return tex
# --- Sanitize the specified string by removing any chars not found in config.CODEPOINTS
def sanitize_string(string):
return ''.join([c for c in string if c in config.CODEPOINTS])
def check_changes():
global last_file_change
update = False
for root, _, _ in os.walk(config.PIC_DIR):
mod_tm = os.stat(root).st_mtime
if mod_tm > last_file_change:
last_file_change = mod_tm
update = True
return update
def get_files(dt_from=None, dt_to=None):
# dt_from and dt_to are either None or tuples (2016,12,25)
if dt_from is not None:
dt_from = time.mktime(dt_from + (0, 0, 0, 0, 0, 0))
if dt_to is not None:
dt_to = time.mktime(dt_to + (0, 0, 0, 0, 0, 0))
global shuffle, EXIF_DATID, last_file_change
file_list = []
extensions = ['.png','.jpg','.jpeg','.heif','.heic'] # can add to these
picture_dir = os.path.join(config.PIC_DIR, subdirectory)
for root, _dirnames, filenames in os.walk(picture_dir):
mod_tm = os.stat(root).st_mtime # time of alteration in a directory
if mod_tm > last_file_change:
last_file_change = mod_tm
for filename in filenames:
ext = os.path.splitext(filename)[1].lower()
if ext in extensions and not '.AppleDouble' in root and not filename.startswith('.'):
file_path_name = os.path.join(root, filename)
include_flag = True
orientation = 1 # this is default - unrotated
dt = None # if exif data not read - used for checking in tex_load
fdt = None
location = ""
aspect = 1.5 # assume landscape aspect until we determine otherwise
if not config.DELAY_EXIF and EXIF_DATID is not None and EXIF_ORIENTATION is not None:
(orientation, dt, fdt, location, aspect) = get_exif_info(file_path_name)
if (dt_from is not None and dt < dt_from) or (dt_to is not None and dt > dt_to):
include_flag = False
if include_flag:
# iFiles now list of lists [file_name, orientation, file_changed_date, exif_date, exif_formatted_date, aspect]
file_list.append(Pic(file_path_name,
orientation,
os.path.getmtime(file_path_name),
dt,
fdt,
location,
aspect))
if shuffle:
file_list.sort(key=lambda x: x.mtime) # will be later files last
temp_list_first = file_list[-config.RECENT_N:]
temp_list_last = file_list[:-config.RECENT_N]
random.seed()
random.shuffle(temp_list_first)
random.shuffle(temp_list_last)
file_list = temp_list_first + temp_list_last
else:
file_list.sort() # if not suffled; sort by name
return file_list, len(file_list) # tuple of file list, number of pictures
def get_exif_info(file_path_name, im=None):
dt = os.path.getmtime(file_path_name) # so use file last modified date
orientation = 1
location = ""
aspect = 1.5 # assume landscape aspect until we determine otherwise
try:
if im is None:
im = Image.open(file_path_name) # lazy operation so shouldn't load (better test though)
aspect = im.width / im.height
exif_data = im._getexif() # TODO check if/when this becomes proper function
if EXIF_DATID in exif_data:
exif_dt = time.strptime(exif_data[EXIF_DATID], '%Y:%m:%d %H:%M:%S')
dt = time.mktime(exif_dt)
if EXIF_ORIENTATION in exif_data:
orientation = int(exif_data[EXIF_ORIENTATION])
if orientation == 6 or orientation == 8:
aspect = 1.0 / aspect # image rotated 270 or 90 degrees
if config.LOAD_GEOLOC and geo.EXIF_GPSINFO in exif_data:
location = geo.get_location(exif_data[geo.EXIF_GPSINFO])
except Exception as e: # NB should really check error here but it's almost certainly due to lack of exif data
if config.VERBOSE:
print('trying to read exif', e)
fdt = time.strftime(config.SHOW_TEXT_FM, time.localtime(dt))
return (orientation, dt, fdt, location, aspect)
def convert_heif(fname):
try:
import pyheif
from PIL import Image
heif_file = pyheif.read(fname)
image = Image.frombytes(heif_file.mode, heif_file.size, heif_file.data,
"raw", heif_file.mode, heif_file.stride)
return image
except:
print("have you installed pyheif?")
EXIF_DATID = None # this needs to be set before get_files() above can extract exif date info
EXIF_ORIENTATION = None
for k in ExifTags.TAGS:
if ExifTags.TAGS[k] == 'DateTimeOriginal':
EXIF_DATID = k
if ExifTags.TAGS[k] == 'Orientation':
EXIF_ORIENTATION = k
if config.LOAD_GEOLOC:
import PictureFrame2020geo as geo
##############################################
# MQTT functionality - see https://www.thedigitalpictureframe.com/
##############################################
iFiles = []
nFi = 0
next_pic_num = 0
if config.USE_MQTT:
try:
import paho.mqtt.client as mqtt
def on_connect(client, userdata, flags, rc):
if config.MQTT_ID == "":
id = ""
else:
id = "{}/".format(config.MQTT_ID)
client.subscribe("{}date_from".format(id), qos=0) # needs payload as 2019:06:01 or divided by ":", "/", "-" or "."
client.subscribe("{}date_to".format(id), qos=0)
client.subscribe("{}time_delay".format(id), qos=0) # payload seconds for each slide
client.subscribe("{}fade_time".format(id), qos=0) # payload seconds for fade time between slides
client.subscribe("{}shuffle".format(id), qos=0) # payload on, yes, true (not case sensitive) will set shuffle on and reshuffle
client.subscribe("{}quit".format(id), qos=0)
client.subscribe("{}paused".format(id), qos=0) # payload on, yes, true pause, off, no, false un-pause. Anything toggle state
client.subscribe("{}back".format(id), qos=0)
client.subscribe("{}next".format(id), qos=0)
client.subscribe("{}subdirectory".format(id), qos=0) # payload string must match a subdirectory of pic_dir
client.subscribe("{}delete".format(id), qos=0) # delete current image, copy to dir set in config
client.subscribe("{}text_on".format(id), qos=0) # toggle file name on and off. payload text show time in seconds
client.subscribe("{}date_on".format(id), qos=0) # toggle date (exif if avail else file) on and off. payload show time
client.subscribe("{}location_on".format(id), qos=0) # toggle location (if enabled) on and off. payload show time
client.subscribe("{}text_off".format(id), qos=0) # turn all name, date, location off
client.subscribe("{}text_refresh".format(id), qos=0) # restarts current slide showing text set above
client.subscribe("{}brightness".format(id), qos=0) # set shader brightness
client.publish("{}paused".format(id), payload="off", qos=0) # un-pause the slideshow on start
if config.VERBOSE:
print("Connected to MQTT broker")
def on_message(client, userdata, message):
# TODO not ideal to have global but probably only reasonable way to do it
global pic_num, next_pic_num, iFiles, nFi, date_from, date_to, time_delay, text_start_tm
global delta_alpha, fade_time, shuffle, quit, paused, nexttm, subdirectory
TRUTH_VALS = {"on":True, "off":False, "true":True, "false":False, "yes":True, "no":False}
msg = message.payload.decode("utf-8")
try:
float_msg = float(msg)
except:
float_msg = 0.0
reselect = False
refresh = False
if config.MQTT_ID == "":
id = ""
else:
id = "{}/".format(config.MQTT_ID)
if message.topic == "{}date_from".format(id): # NB entered as mqtt string "2016:12:25"
try:
msg = msg.replace(".",":").replace("/",":").replace("-",":")
df = msg.split(":")
date_from = tuple(int(i) for i in df)
if len(date_from) != 3:
raise Exception("invalid date format")
except:
date_from = None
reselect = True
elif message.topic == "{}date_to".format(id):
try:
msg = msg.replace(".",":").replace("/",":").replace("-",":")
df = msg.split(":")
date_to = tuple(int(i) for i in df)
if len(date_to) != 3:
raise Exception("invalid date format")
except:
date_to = None
reselect = True
elif message.topic == "{}time_delay".format(id):
if float_msg > 0.0:
time_delay = float_msg
elif message.topic == "{}fade_time".format(id):
if float_msg > 0.0:
fade_time = float_msg
delta_alpha = 1.0 / (config.FPS * fade_time)
elif message.topic == "{}shuffle".format(id):
msg_val = msg.lower()
shuffle = TRUTH_VALS[msg_val] if msg_val in TRUTH_VALS else False
reselect = True
elif message.topic == "{}quit".format(id):
quit = True
elif message.topic == "{}paused".format(id):
msg_val = msg.lower()
paused = TRUTH_VALS[msg_val] if msg_val in TRUTH_VALS else not paused # toggle from previous value
text_start_tm = -0.1
elif message.topic == "{}back".format(id):
next_pic_num -= 2
refresh = True
elif message.topic == "{}next".format(id):
refresh = True
elif message.topic == "{}subdirectory".format(id):
subdirectory = msg.strip()
reselect = True
elif message.topic == "{}delete".format(id):
f_to_delete = iFiles[pic_num].fname
move_to_dir = os.path.expanduser("/home/pi/DeletedPictures") # NB hard coded - may not be suitable location
if not os.path.exists(move_to_dir):
os.system("sudo -u pi mkdir {}".format(move_to_dir)) # problems with ownership using python func
os.system("sudo mv '{}' '{}'".format(f_to_delete, move_to_dir)) # and with SMB drives
iFiles.pop(pic_num)
nFi -= 1
refresh = True
elif message.topic == "{}text_on".format(id):
config.SHOW_TEXT_TM = float_msg if float_msg > 2.0 else 0.33 * config.TIME_DELAY
config.SHOW_TEXT ^= 1
text_start_tm = -0.1
elif message.topic == "{}date_on".format(id):
config.SHOW_TEXT_TM = float_msg if float_msg > 2.0 else 0.33 * config.TIME_DELAY
config.SHOW_TEXT ^= 2
text_start_tm = -0.1
elif message.topic == "{}location_on".format(id):
config.SHOW_TEXT_TM = float_msg if float_msg > 2.0 else 0.33 * config.TIME_DELAY
config.SHOW_TEXT ^= 4
text_start_tm = -0.1
elif message.topic == "{}text_off".format(id):
config.SHOW_TEXT = 0
text_start_tm = -0.1
elif message.topic == "{}text_refresh".format(id):
next_pic_num -= 1
refresh = True
elif message.topic == "{}brightness".format(id):
slide.unif[55] = float_msg
if reselect:
iFiles, nFi = get_files(date_from, date_to)
next_pic_num = 0
if refresh:
if next_pic_num < -1:
next_pic_num = -1
nexttm = time.time() - 86400.0 # end current pic next frame,
# nexttm setting will start next image (next_pic_num + 1) on next frame
# reset Pic.shown_with for images previously paired with next pic
for pic in (pic_num, next_pic_num + 1):
pic %= len(iFiles)
for p in iFiles:
if p.shown_with == pic:
p.shown_with = None
break # shouldn't be possible to be more than one?
# set up MQTT listening
client = mqtt.Client()
client.on_connect = on_connect
client.on_message = on_message
client.username_pw_set(config.MQTT_LOGIN, config.MQTT_PASSWORD)
client.connect(config.MQTT_SERVER, config.MQTT_PORT, 60)
client.loop_start()
except Exception as e:
if config.VERBOSE:
print("MQTT not set up because of: {}".format(e)) # sometimes starts paused
##############################################
DISPLAY = pi3d.Display.create(x=config.DISPLAY_X, y=config.DISPLAY_Y,
w=config.DISPLAY_W, h=config.DISPLAY_H, frames_per_second=config.FPS,
display_config=pi3d.DISPLAY_CONFIG_HIDE_CURSOR, background=config.BACKGROUND)
CAMERA = pi3d.Camera(is_3d=False)
shader = pi3d.Shader(config.SHADER)
slide = pi3d.Sprite(camera=CAMERA, w=DISPLAY.width, h=DISPLAY.height, z=5.0)
slide.set_shader(shader)
slide.unif[47] = config.EDGE_ALPHA
slide.unif[54] = config.BLEND_TYPE
slide.unif[55] = 1.0 # brightness used by shader [18][1]
if config.KEYBOARD:
kbd = pi3d.Keyboard()
# images in iFiles list
nexttm = 0.0
iFiles, nFi = get_files(date_from, date_to)
next_pic_num = 0
sfg = None # slide for background
sbg = None # slide for foreground
# PointText and TextBlock. If SHOW_TEXT_TM <= 0 then this is just used for no images message
grid_size = math.ceil(len(config.CODEPOINTS) ** 0.5)
font = pi3d.Font(config.FONT_FILE, codepoints=config.CODEPOINTS, grid_size=grid_size)
text = pi3d.PointText(font, CAMERA, max_chars=200, point_size=config.SHOW_TEXT_SZ)
textblock = pi3d.TextBlock(x=0, y=-DISPLAY.height // 2 + (config.SHOW_TEXT_SZ // 2) + 20,
z=0.1, rot=0.0, char_count=199,
text_format="{}".format(" "), size=0.99,
spacing="F", justify=0.5, space=0.02, colour=(1.0, 1.0, 1.0, 1.0))
text.add_text_block(textblock)
"""
back_shader = pi3d.Shader("mat_flat")
text_bkg = pi3d.Sprite(w=DISPLAY.width, h=90, y=-DISPLAY.height * 0.4 - 20, z=4.0)
text_bkg.set_shader(back_shader)
text_bkg.set_material((0, 0, 0))
"""
bkg_ht = DISPLAY.height // 3
text_bkg_array = np.zeros((bkg_ht, 1, 4), dtype=np.uint8)
text_bkg_array[:,:,3] = np.linspace(0, 170, bkg_ht).reshape(-1, 1)
text_bkg_tex = pi3d.Texture(text_bkg_array, blend=True, free_after_load=True)
text_bkg = pi3d.Plane(w=DISPLAY.width, h=bkg_ht, y=-DISPLAY.height // 2 + bkg_ht // 2, z=4.0)
back_shader = pi3d.Shader("uv_flat")
text_bkg.set_draw_details(back_shader, [text_bkg_tex])
num_run_through = 0
while DISPLAY.loop_running():
tm = time.time()
if (tm > nexttm and not paused) or (tm - nexttm) >= 86400.0: # this must run first iteration of loop
if nFi > 0:
nexttm = tm + time_delay
sbg = sfg
sfg = None
start_pic_num = next_pic_num
loop_count = 0
while sfg is None: # keep going through until a usable picture is found
pic_num = next_pic_num
sfg = tex_load(pic_num, iFiles, (DISPLAY.width, DISPLAY.height))
next_pic_num += 1
if next_pic_num >= nFi:
num_run_through += 1
if shuffle and num_run_through >= config.RESHUFFLE_NUM:
num_run_through = 0
random.shuffle(iFiles)
next_pic_num = 0
for f_rec in iFiles:
f_rec.shown_with = None # reset all the portrait mode show in pairs
loop_count += 1
if loop_count > nFi: #i.e. no images found where tex_load doesn't return None
nFi = 0
break
text_start_tm = -fade_time # used as flag for text setting and amount to delay start
if sfg is None:
sfg = tex_load(config.NO_FILES_IMG, 1, (DISPLAY.width, DISPLAY.height))
if sfg is None:
print("NO FILES image not loaded, check path!")
break
sbg = sfg
a = 0.0 # alpha - proportion front image to back
#text_start_tm = tm
if sbg is None: # first time through
sbg = sfg
slide.set_textures([sfg, sbg])
slide.unif[45:47] = slide.unif[42:44] # transfer front width and height factors to back
slide.unif[51:53] = slide.unif[48:50] # transfer front width and height offsets
wh_rat = (DISPLAY.width * sfg.iy) / (DISPLAY.height * sfg.ix)
fit = config.FIT if iFiles[pic_num].aspect >= 1.0 else config.FIT_PORTRAIT
if (wh_rat > 1.0 and fit) or (wh_rat <= 1.0 and not fit):
sz1, sz2, os1, os2 = 42, 43, 48, 49
else:
sz1, sz2, os1, os2 = 43, 42, 49, 48
wh_rat = 1.0 / wh_rat
slide.unif[sz1] = wh_rat
slide.unif[sz2] = 1.0
slide.unif[os1] = (wh_rat - 1.0) * 0.5
slide.unif[os2] = 0.0
if config.KENBURNS:
ken_time = nexttm - time.time() # some time will have passed doing file loading etc
xstep, ystep = (slide.unif[i] * 2.0 / ken_time for i in (48, 49))
slide.unif[48] = 0.0
slide.unif[49] = 0.0
#kb_up = not kb_up # toggle direction for each slide
if text_start_tm < 0.0:
info_strings = []
if config.SHOW_TEXT > 0 or paused: #was SHOW_TEXT_TM > 0.0
if (config.SHOW_TEXT & 1) == 1: # name
info_strings.append(sanitize_string(os.path.basename(iFiles[pic_num].fname)))
if (config.SHOW_TEXT & 2) == 2: # date
info_strings.append(iFiles[pic_num].fdt)
if config.LOAD_GEOLOC and (config.SHOW_TEXT & 4) == 4: # location
loc_string = sanitize_string(iFiles[pic_num].location.strip())
if loc_string:
info_strings.append(loc_string)
if (config.SHOW_TEXT & 8) == 8: # folder
info_strings.append(sanitize_string(os.path.basename(os.path.dirname(iFiles[pic_num].fname))))
if paused:
info_strings.append("PAUSED")
final_string = " • ".join(info_strings)
textblock.set_text(text_format=final_string, wrap=config.TEXT_WIDTH)
last_ch = len(final_string)
adj_y = text.locations[:last_ch,1].min() + DISPLAY.height // 2 # y pos of last char rel to bottom of screen
textblock.set_position(y = (textblock.y - adj_y + config.SHOW_TEXT_SZ))
else: # could have a NO IMAGES selected and being drawn
textblock.set_text(text_format="{}".format(" "))
textblock.colouring.set_colour(alpha=0.0)
text_bkg.set_alpha(0.0)
text.regen()
text_start_tm = tm - text_start_tm # i.e by setting to larger neg number delays text start
if config.KENBURNS:
t_factor = nexttm - tm
if kb_up:
t_factor = ken_time - t_factor
slide.unif[48] = xstep * t_factor
slide.unif[49] = ystep * t_factor
if a < 1.0: # transition is happening
a += delta_alpha
if a > 1.0:
a = 1.0
slide.unif[44] = a * a * (3.0 - 2.0 * a)
else: # no transition effect safe to resuffle etc
if tm > next_check_tm:
if check_changes():
iFiles, nFi = get_files(date_from, date_to)
num_run_through = 0
next_pic_num = 0
next_check_tm = tm + config.CHECK_DIR_TM # once per hour
slide.draw()
if nFi <= 0:
textblock.set_text("NO IMAGES SELECTED")
textblock.colouring.set_colour(alpha=1.0)
next_tm = tm + 1.0
text.regen()
elif tm < (text_start_tm + config.SHOW_TEXT_TM) and (config.SHOW_TEXT > 0 or paused):#config.SHOW_TEXT_TM > 0.0:
# this sets alpha for the TextBlock from 0 to 1 then back to 0
dt = (tm - text_start_tm + 0.1) / config.SHOW_TEXT_TM
ramp_pt = max(4.0, config.SHOW_TEXT_TM / 4.0)
alpha = max(0.0, min(1.0, ramp_pt * (a - abs(1.0 - 2.0 * dt)))) # cap text alpha at image alpha
textblock.colouring.set_colour(alpha=alpha)
text.regen()
text_bkg.set_alpha(alpha)
if len(textblock.text_format.strip()) > 0: #only draw background if text there
text_bkg.draw()
text.draw()
if config.KEYBOARD:
k = kbd.read()
if k != -1:
nexttm = time.time() - 86400.0
if k==27: #ESC
break
if k==ord(' '):
paused = not paused
if k==ord('s'): # go back a picture
next_pic_num -= 2
if next_pic_num < -1:
next_pic_num = -1
if quit: # set by MQTT
break
try:
client.loop_stop()
except Exception as e:
if config.VERBOSE:
print("this was going to fail if previous try failed!")
if config.KEYBOARD:
kbd.close()
DISPLAY.destroy()