forked from drumminhands/drumminhands_photobooth
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathphotobooth_no_upload.py
473 lines (412 loc) · 16.8 KB
/
photobooth_no_upload.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
#!/usr/bin/env python
# created by steve@stevesiden.com
# modified from chris@drumminhands.com
# see instructions at http://www.drumminhands.com/2014/06/15/raspberry-pi-photo-booth/
import os
import glob
import time
import traceback
from time import sleep
import RPi.GPIO as GPIO #using physical pin numbering change in future?
import picamera # http://picamera.readthedocs.org/en/release-1.4/install2.html
import atexit
import sys, getopt
import socket
import pygame
import cups
import Image, ImageDraw, ImageFont
import pytumblr # https://github.com/tumblr/pytumblr
from twython import Twython
import config
import shutil
from signal import alarm, signal, SIGALRM, SIGKILL
import Image, ImageDraw
########################
### Variables Config ####
########################
led1_pin = 16 # LED 1 #15
led2_pin = 10 # LED 2 #19
led3_pin = 21 # LED 3 #21
led4_pin = 23 # LED 4 #23
button1_pin = 25 # pin for the start button
button2_pin = 4 # pin for printer switch
button3_pin = 17 # pin for button to end the program, but not shutdown the pi
post_online = 0 # default 1. Change to 0 if you don't want to upload pics.
total_pics = 4 # number of pics to be taken
capture_delay = 1 # delay between pics
prep_delay = 7 # number of seconds at step 1 as users prep to have photo taken
gif_delay = 50 # How much time between frames in the animated gif
restart_delay = 12 # how long to display finished message before beginning a new session
#paper_total = 16 # number of pages the printer can handle
#printed_count = 0 # number of pages printed since last refill
printer_switch = 0
backup_pics = 1 # backup pics = 1, no backup, change to 0
monitor_w = 800 #800
monitor_h = 480 #480
transform_x = 640 #640 # how wide to scale the jpg when replaying
transfrom_y = 480 #480 # how high to scale the jpg when replaying
offset_x = 10 # how far off to left corner to display photos
offset_y = 0 # how far off to left corner to display photos
replay_delay = 1 # how much to wait in-between showing pics on-screen after taking
replay_cycles = 1 # how many times to show each photo on-screen after taking
test_server = 'www.google.com'
real_path = os.path.dirname(os.path.realpath(__file__))
font = ImageFont.truetype("/usr/share/fonts/truetype/freefont/FreeSerif.ttf", 200)
# Setup the tumblr OAuth Client
client = pytumblr.TumblrRestClient(
config.tumblr_consumer_key,
config.tumblr_consumer_secret,
config.tumblr_oath_token,
config.tumblr_oath_secret,
);
#setup the twitter api client
twitter_api = Twython(
config.twitter_CONSUMER_KEY,
config.twitter_CONSUMER_SECRET,
config.twitter_ACCESS_KEY,
config.twitter_ACCESS_SECRET,
);
####################
### Other Config ###
####################
GPIO.setmode(GPIO.BCM)
GPIO.setup(led1_pin,GPIO.OUT) # LED 1
GPIO.setup(led2_pin,GPIO.OUT) # LED 2
GPIO.setup(led3_pin,GPIO.OUT) # LED 3
GPIO.setup(led4_pin,GPIO.OUT) # LED 4
GPIO.setup(button1_pin, GPIO.IN, pull_up_down=GPIO.PUD_UP) # falling edge detection on button 1
GPIO.setup(button2_pin, GPIO.IN, pull_up_down=GPIO.PUD_UP) # falling edge detection on button 2
GPIO.setup(button3_pin, GPIO.IN, pull_up_down=GPIO.PUD_UP) # falling edge detection on button 3
GPIO.output(led1_pin,False);
GPIO.output(led2_pin,False);
GPIO.output(led3_pin,False);
GPIO.output(led4_pin,False);
#################
### Functions ###
#################
def cleanup():
print('Ended abruptly')
GPIO.cleanup()
#atexit.register(cleanup)
def shut_it_down(channel):
print "Shutting down..."
#GPIO.output(led1_pin,True)
GPIO.output(led2_pin,True)
#GPIO.output(led3_pin,True)
#GPIO.output(led4_pin,True)
os.system("sudo halt")
def exit_photobooth(channel):
print "Photo booth app ended. RPi still running"
#GPIO.output(led1_pin,True)
time.sleep(3)
raise SystemExit
def clear_pics(foo): #why is this function being passed an arguments?
#delete files in folder on startup
files = glob.glob(config.file_path + '*')
for f in files:
os.remove(f)
#light the lights in series to show completed
print "Deleted previous pics"
#GPIO.output(led1_pin,False) #turn off the lights
GPIO.output(led2_pin,False)
#GPIO.output(led3_pin,False)
#GPIO.output(led4_pin,False)
def is_connected():
try:
# see if we can resolve the host name -- tells us if there is
# a DNS listening
host = socket.gethostbyname(test_server)
# connect to the host -- tells us if the host is actually
# reachable
s = socket.create_connection((host, 80), 2)
return True
except:
pass
return False
def init_pygame():
pygame.init()
size = (pygame.display.Info().current_w, pygame.display.Info().current_h)
pygame.display.set_caption('Photo Booth Pics')
pygame.mouse.set_visible(False) #hide the mouse cursor
return pygame.display.set_mode(size, pygame.FULLSCREEN)
def countdown(camera):
overlay_renderer = None
for j in range(1,4):
img = Image.new("RGB", (monitor_w, monitor_h))
draw = ImageDraw.Draw(img)
draw.text((monitor_w/2,monitor_h/2), str(4-j), (255, 255, 255), font=font)
if not overlay_renderer:
overlay_renderer = camera.add_overlay(img.tostring(),layer=3,size=img.size,alpha=28);
else:
overlay_renderer.update(img.tostring())
sleep(1)
img = Image.new("RGB", (monitor_w, monitor_h))
draw = ImageDraw.Draw(img)
draw.text((monitor_w/2,monitor_h/2), " ", (255, 255, 255), font=font)
overlay_renderer.update(img.tostring())
def show_image(image_path):
screen = init_pygame()
img=pygame.image.load(image_path)
img = pygame.transform.scale(img,(transform_x,transfrom_y))
screen.blit(img,(offset_x,offset_y))
pygame.display.flip()
def create_mosaic(jpg_group):
now = jpg_group
##moving original pics to backup
##copypics = "cp " + file_path + now + "*.jpg "+ file_path
##print copypics
##os.system(copypics)
##resizing + montaging
#print "Resizing Pics..." #necessary?
##convert -resize 968x648 /home/pi/photobooth/pics/*.jpg /home/pi/photobooth/pics_tmp/*_tmp.jpg
#graphicsmagick = "gm mogrify -resize 968x648 " + config.file_path + now + "*.jpg"
#copypics = "cp " + config.file_path + now + "*.jpg "+ config.file_path
##print "Resizing with command: " + graphicsmagick
#os.system(graphicsmagick)
#os.system(copypics)
#print "Montaging Pics..."
#graphicsmagick = "gm montage " + config.file_path + now + "*.jpg -tile 2x2 -geometry 1000x699+10+10 " + config.file_path + now + "_picmontage.jpg"
#print "Montaging images with command: " + graphicsmagick
#os.system(graphicsmagick)
#print "Adding Label..."
#graphicsmagick = "gm convert -append "+real_path+ "/assets/bn_booth_label_h.jpg " + config.file_path + now + "_picmontage.jpg " + config.file_path + now + "_print.jpg"
#print "Adding label with command: " + graphicsmagick
#os.system(graphicsmagick)
image = list()
image.append(Image.open(config.file_path + now + '-01.jpg'))
image.append(Image.open(config.file_path + now + '-02.jpg'))
image.append(Image.open(config.file_path + now + '-03.jpg'))
image.append(Image.open(config.file_path + now + '-04.jpg'))
x_pic = 500
y_pic = 375
x_border = 40
y_border = 10
x_total = 1181
y_total = 1748
new_pic = Image.new('RGB', (x_total, y_total), (255, 255, 255))
new_pic.paste(image[0].resize((x_pic,y_pic), Image.ANTIALIAS), (x_border,y_border))
new_pic.paste(image[1].resize((x_pic,y_pic), Image.ANTIALIAS), (x_border,1*y_pic+2*y_border))
new_pic.paste(image[2].resize((x_pic,y_pic), Image.ANTIALIAS), (x_border,2*y_pic+3*y_border))
new_pic.paste(image[3].resize((x_pic,y_pic), Image.ANTIALIAS), (x_border,3*y_pic+4*y_border))
new_pic.paste(image[0].resize((x_pic,y_pic), Image.ANTIALIAS), (x_total-x_border-x_pic,y_border))
new_pic.paste(image[1].resize((x_pic,y_pic), Image.ANTIALIAS), (x_total-x_border-x_pic,1*y_pic+2*y_border))
new_pic.paste(image[2].resize((x_pic,y_pic), Image.ANTIALIAS), (x_total-x_border-x_pic,2*y_pic+3*y_border))
new_pic.paste(image[3].resize((x_pic,y_pic), Image.ANTIALIAS), (x_total-x_border-x_pic,3*y_pic+4*y_border))
new_pic.save('/home/pi/photobooth/backups/' + now + '_total.jpg')
#try:
# new_pic.save('/home/pi/photobooth/backups/' + now + '_total.jpg')
#except:
# print "Flash drive not found"
def print_pics(jpg_group):
now = jpg_group
print "Printing..."
#printing
printcommand = "lp -d Canon_CP910 " + '/home/pi/photobooth/backups/' + now + "_total.jpg"
os.system(printcommand)
#if(printed_count == paper_total):
# show_image(real_path + "/assets/empty_printer.png")
# GPIO.wait_for_edge(button2_pin, GPIO.FALLING)
# printed_count = 0
#else:
# printed_count = printed_count + 1
def tweet_pics(jpg_group):
now = jpg_group
twitter_photo = open('/home/pi/photobooth/backups/' + now + '_total.jpg','rb')
twitter_api.update_status_with_media(media=twitter_photo, status='#RPiBooth')
def display_pics(jpg_group):
# this section is an unbelievable nasty hack - for some reason Pygame
# needs a keyboardinterrupt to initialise in some limited circs (second time running)
class Alarm(Exception):
pass
def alarm_handler(signum, frame):
raise Alarm
signal(SIGALRM, alarm_handler)
alarm(3)
try:
screen = init_pygame()
alarm(0)
except Alarm:
raise KeyboardInterrupt
for i in range(0, replay_cycles): #show pics a few times
for i in range(1, total_pics+1): #show each pic
filename = config.file_path + jpg_group + "-0" + str(i) + ".jpg"
show_image(filename);
time.sleep(replay_delay) # pause
def pics_backup(now):
print "Backing Up Photos"
shutil.copy('/home/pi/photobooth/pics/' + now + '-01.jpg', '/media/backup/pics/')
shutil.copy('/home/pi/photobooth/pics/' + now + '-02.jpg', '/media/backup/pics/')
shutil.copy('/home/pi/photobooth/pics/' + now + '-03.jpg', '/media/backup/pics/')
shutil.copy('/home/pi/photobooth/pics/' + now + '-04.jpg', '/media/backup/pics/')
shutil.copy('/home/pi/photobooth/pics/' + now + '.gif', '/media/backup/pics/')
shutil.copy('/home/pi/photobooth/backups/' + now + '_total.jpg', '/media/backup/pics/')
def tumblr_backup(now):
#shutil.copyfile('/home/pi/photobooth/backup' + now + '-01.jpg', '/home/pi/photobooth/backup')
client.create_photo(config.tumblr_blog, state="published", tags=["RPiBooth", "photo", "photo 1"], data=config.file_path + now + '-01.jpg')
client.create_photo(config.tumblr_blog, state="published", tags=["RPiBooth", "photo", "photo 2"], data=config.file_path + now + '-02.jpg')
client.create_photo(config.tumblr_blog, state="published", tags=["RPiBooth", "photo", "photo 3"], data=config.file_path + now + '-03.jpg')
client.create_photo(config.tumblr_blog, state="published", tags=["RPiBooth", "photo", "photo 4"], data=config.file_path + now + '-04.jpg')
client.create_photo(config.tumblr_blog, state="published", tags=["RPiBooth", "film strip"], data=config.file_path + now + '_total.jpg')
# define the photo taking function for when the big button is pressed
def start_photobooth():
################################# Begin Step 1 ##################################
show_image(real_path + "/assets/blank.png")
print "Get Ready"
GPIO.output(led2_pin,True);
show_image(real_path + "/assets/instructions.png")
sleep(prep_delay)
#GPIO.output(led2_pin,False)
show_image(real_path + "/assets/blank.png")
camera = picamera.PiCamera()
pixel_width = 800 #1000 #originally 500: use a smaller size to process faster, and tumblr will only take up to 500 pixels wide for animated gifs
#pixel_height = monitor_h * pixel_width // monitor_w #optimize for monitor size
pixel_height = 480 #666
camera.resolution = (pixel_width, pixel_height)
camera.vflip = False
camera.hflip = False
camera.start_preview()
#sleep(2) #warm up camera
################################# Begin Step 2 #################################
print "Taking pics"
now = time.strftime("%Y%m%d%H%M%S") #get the current date and time for the start of the filename
try: #take the photos
#for i, filename in enumerate(camera.capture_continuous(config.file_path + now + '-' + '{counter:02d}.jpg')):
for i in range(0, total_pics):
countdown(camera)
filename = config.file_path + now + '-0' + str(i+1) + '.jpg'
camera.capture(filename)
GPIO.output(led2_pin,True) #turn on the LED
print(filename)
sleep(0.25) #pause the LED on for just a bit
GPIO.output(led2_pin,False) #turn off the LED
sleep(capture_delay) # pause in-between shots
if i == total_pics-1:
break
finally:
camera.stop_preview()
camera.close()
########################### Begin Step 3 #################################
#Make Mosaic
try:
create_mosaic(now)
except Exception, e:
tb = sys.exc_info()[2]
traceback.print_exception(e.__class__, e, tb)
printflag = True
tweetflag = True
#check for tweeting or printing
for s in sys.argv:
if (s == "p"):
printflag = True
if (s == "t"):
tweetflag = True
#PRINT MOSAIC if flag is set
if(printer_switch == 0):
if(printflag):
try:
print_pics(now)
except Exception, e:
tb = sys.exc_info()[2]
traceback.print_exception(e.__class__, e, tb)
########################### Begin Step 4 #################################
print "Creating an animated gif"
if post_online:
show_image(real_path + "/assets/uploading.png")
else:
show_image(real_path + "/assets/processing.png")
GPIO.output(led2_pin,True) #turn on the LED
graphicsmagick = "gm convert -size 500x333 -delay " + str(gif_delay) + " " + config.file_path + now + "*.jpg " + config.file_path + now + ".gif"
os.system(graphicsmagick) #make the .gif
#im = Image.open(config.file_path + now + "-01.jpg")
#im.save(config.file_path + now + "-01.gif")
if post_online: # turn off posting pics online in the variable declarations at the top of this document
print "Uploading to tumblr. Please check RPiBooth.com soon."
connected = is_connected() #check to see if you have an internet connection
while connected:
try:
file_to_upload = config.file_path + now + ".gif"
client.create_photo(config.tumblr_blog, state="published", tags=["RPiBooth", "gif"], data=file_to_upload)
break
except ValueError:
print "Oops. No internect connection. Upload later."
try: #make a text file as a note to upload the .gif later
file = open(config.file_path + now + "-FILENOTUPLOADED.txt",'w') # Trying to create a new file or open one
file.close()
except:
print('Something went wrong. Could not write file.')
sys.exit(0) # quit Python
#tumblr_backup(now)
#GPIO.output(led4_pin,True) #turn on the LED
try:
display_pics(now)
except Exception, e:
tb = sys.exc_info()[2]
traceback.print_exception(e.__class__, e, tb)
# TWEET PICS if flag is set
if(tweetflag):
try:
pics_backup(now)
except Exception, e:
tb = sys.exc_info()[2]
traceback.print_exception(e.__class__, e, tb)
pygame.quit()
print "Done"
GPIO.output(led4_pin,False) #turn off the LED
if post_online:
show_image(real_path + "/assets/finished_connected.png")
else:
show_image(real_path + "/assets/finished_offline.png")
time.sleep(restart_delay)
show_image(real_path + "/assets/intro.png");
####################
### Main Program ###
####################
# when a falling edge is detected on button2_pin and button3_pin, regardless of whatever
# else is happening in the program, their function will be run
#GPIO.add_event_detect(button2_pin, GPIO.FALLING, callback=shut_it_down, bouncetime=300)
#choose one of the two following lines to be un-commented
GPIO.add_event_detect(button3_pin, GPIO.FALLING, callback=exit_photobooth, bouncetime=100) #use third button to exit python. Good while developing
#GPIO.add_event_detect(button3_pin, GPIO.FALLING, callback=clear_pics, bouncetime=300) #use the third button to clear pics stored on the SD card from previous events
# Check which frame buffer drivers are available
# Start with fbcon since directfb hangs with composite output
drivers = ['fbcon', 'directfb', 'svgalib']
found = False
for driver in drivers:
# Make sure that SDL_VIDEODRIVER is set
if not os.getenv('SDL_VIDEODRIVER'):
os.putenv('SDL_VIDEODRIVER', driver)
try:
pygame.display.init()
except pygame.error:
print 'Driver: {0} failed.'.format(driver)
continue
found = True
break
if not found:
raise Exception('No suitable video driver found!')
# delete files in folder on startup
files = glob.glob(config.file_path + '*')
for f in files:
os.remove(f)
print "Photo booth app running..."
#GPIO.output(led1_pin,True); #light up the lights to show the app is running
GPIO.output(led2_pin,True);
#GPIO.output(led3_pin,True);
#GPIO.output(led4_pin,True);
time.sleep(3)
#GPIO.output(led1_pin,False); #turn off the lights
GPIO.output(led2_pin,False);
#GPIO.output(led3_pin,False);
#GPIO.output(led4_pin,False);
#GPIO.add_event_detect(button2_pin, GPIO.FALLING, callback=shut_it_down, bouncetime=300)
#GPIO.add_event_detect(button3_pin, GPIO.FALLING, callback=exit_photobooth)
show_image(real_path + "/assets/intro.png");
if(GPIO.input(button2_pin) == 1):
printer_switch = True
try:
while True:
GPIO.wait_for_edge(button1_pin, GPIO.FALLING)
time.sleep(0.2) #debounce
start_photobooth()
finally:
cleanup()