-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathsquishbox.py
1113 lines (1015 loc) · 46.7 KB
/
squishbox.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
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
#!/usr/bin/env python3
"""SquishBox Raspberry Pi interface
This script provides an interface for
`Fluidpatcher <https://github.com/GeekFunkLabs/fluidpatcher>`_,
a performance-oriented patch-based wrapper around the
FluidSynth <https://fluidsynth.org>`_ software synthesizer,
on a headlesss (i.e. lacking monitor and keyboard) Raspberry Pi,
using a character LCD, a momentary button/stompswitch, and a
pushbutton rotary encoder.
This script can also be imported as a module to create other applications
for the `SquishBox <https://www.geekfunklabs.com/products/squishbox>`_
The `SquishBox` class initializes the character LCD and sets up GPIO pins for
for the encoder, stompswitch, and output pins (including LED). It provides
convenience methods for writing to the LCD, polling inputs/updating the
display, standard menu and input functions, and a few utilities such as
shell command access and wifi control.
Requires:
- RPi.GPIO
- fluidpatcher
"""
__version__ = '0.8.5'
import re
import subprocess
import sys
import threading
import time
import traceback
import RPi.GPIO as GPIO
# user-adjustable settings
LCD_RS = 2; LCD_EN = 3; LCD_DATA = 11, 5, 6, 13 # LCD pins
ROT_L = 22; ROT_R = 10; BTN_R = 9 # rotary encoder R/L pins + button
BTN_SW = 27; PIN_LED = 17 # stompbutton and LED
ACTIVE = GPIO.LOW # voltage level for pressed buttons
PIN_OUT = PIN_LED, 12, 16, 26 # free pins - see gpio_set()
COLS, ROWS = 16, 2 # LCD display size
SCROLL_TIME = 0.4; SCROLL_PAUSE = 4 # scrolling text options
HOLD_TIME = 1.0; MENU_TIMEOUT = 5.0 # menu timimng options
BLINK_TIME = 0.1 # default text blink time
POLL_TIME = 0.01 # default button polling interval
BOUNCE_TIME = 0.02 # button debounce time
EXEC_TIME = 50e-6 # increase if LCD displays garbage
STOMP_CHAN = 16 # MIDI channel for stompswitch
STOMP_MOMENT = 30 # CC number for momentary message
STOMP_TOGGLE = 31 # CC number for toggle message
# FluidBox MIDI controls
MIDI_CTRL = None # MIDI channel for controls
MIDI_DEC = None # CC for patch/value decrement
MIDI_INC = None # CC for patch/value increment
MIDI_PATCH = None # knob/slider CC for selecting patches
MIDI_SELECT = None # CC for menu/select
MIDI_ESCAPE = None # CC for escape/back
MIDI_SHUTDOWN = None # CC to trigger shutdown when held
# custom characters
c = """\
| | | | | ### | # # | # \
# | | #|## ##| |# #| # | ## \
# | | ##| ### |## | # | # # | # #\
# | ## #|# ## | # |# ###| # # | | # #\
# |# # |### | ### |# #| | # | # \
#| | # |## ##|# #| # | |### \
| | | |#####| | # |### \
| | | | | | | \
""".replace('|', '').replace('#', '1').replace(' ', '0')
charbits = [[int(c[i:i + 5], 2) for i in range(j, 320, 40)] for j in range(0, 40, 5)]
BACKSLASH, TILDE, CHECK, XMARK, SUBDIR, WIFIUP, WIFIDOWN, MIDIACT = [chr(i) for i in range(8)]
FNCHARS = " abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_./" + BACKSLASH
USCHARS = """ abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$%^&*|/_?.,;:'"`-+=<>()[]{}""" + BACKSLASH + TILDE
# button states
UP = 0; DOWN = 1; HELD = 2
# events
NULL = 0; DEC = 1; INC = 2; SELECT = 3; ESCAPE = 4
class SquishBox():
"""An interface for RPi using character LCD and buttons
Attributes:
buttoncallback: When the state of a button connected to BTN_SW
changes, this function is called with 1 if the button was
pressed, 0 if it was released.
wifistatus: contains either the WIFIUP or WIFIDOWN character
depending on the last-known status of the wifi adapter
"""
def __init__(self):
"""Initializes the LCD and GPIO"""
GPIO.setwarnings(False)
GPIO.setmode(GPIO.BCM)
for chan in (ROT_R, ROT_L, BTN_R, BTN_SW):
if chan:
pud = GPIO.PUD_UP if ACTIVE == GPIO.LOW else GPIO.PUD_DOWN
GPIO.setup(chan, GPIO.IN, pull_up_down=pud)
for chan in (LCD_RS, LCD_EN, *LCD_DATA, *PIN_OUT):
if chan:
GPIO.setup(chan, GPIO.OUT)
for btn in (BTN_R, BTN_SW):
if btn:
GPIO.add_event_detect(btn, GPIO.BOTH, callback=self._button_event)
for enc in (ROT_L, ROT_R):
if enc:
GPIO.add_event_detect(enc, GPIO.BOTH, callback=self._encoder_event)
self.state = {BTN_R: UP, BTN_SW: UP}
self.timer = {BTN_R: 0, BTN_SW: 0}
self.encstate = 0b000000
self.encvalue = 0
self.buttoncallback = None
self.nextevent = NULL
for val in (0x33, 0x32, 0x28, 0x0c, 0x06):
self._lcd_send(val)
self.lcd_clear()
for loc, bits in enumerate(charbits):
self._lcd_send(0x40 | loc << 3)
for row in bits:
self._lcd_send(row, 1)
self.wifi_state()
sys.excepthook = lambda etype, err, tb: self.display_error(err, etype=etype, tb=tb)
def update(self, idle=POLL_TIME, callback=True):
"""Polls buttons and updates LCD
Call in the main loop of a program to poll the buttons and rotary
encoder, and update the LCD if necessary. Sleeps for a small amount
of time before returning so other processes can run.
Returns an event code based on the state of the buttons. If
buttoncallback is set and callback=True, the stompswitch calls
that function instead of sending an event.
* NULL (0) - no event
* DEC (1) - encoder step counter-clockwise or stompswitch tap
* INC (2) - encoder step clockwise or encoder button tap
* SELECT (3) - encoder button held for HOLD_TIME seconds
* ESCAPE (4) - stompswitch held for HOLD_TIME seconds
Args:
idle: number of seconds to sleep before returning
callback: if False ignores buttoncallback
Returns: an integer event code
"""
callback = self.buttoncallback if callback else None
t = time.time()
for r in range(ROWS):
text = list(self.buffer[r])
if len(text) > COLS:
if t > self.scrolltimer:
self.scrollpos[r] += 1
if self.scrollpos[r] > len(text) - COLS + SCROLL_PAUSE:
self.scrollpos[r] = -SCROLL_PAUSE
i = max(0, self.scrollpos[r])
i = min(i, len(text) - COLS)
text = text[i : i+COLS]
if self.blinktimer > 0:
for i, c in enumerate(self.blinked[r]):
if c:
text[i] = c
self._lcd_putchars(text, r, 0)
if t > self.scrolltimer:
self.scrolltimer = time.time() + SCROLL_TIME
if t > self.blinktimer:
self.blinked = [[""] * COLS for _ in range(ROWS)]
self.blinktimer = 0
event = self.nextevent
self.nextevent = NULL
if event: return event
for b in BTN_R, BTN_SW:
if t - self.timer[b] > BOUNCE_TIME:
if b and GPIO.input(b) == ACTIVE:
if self.state[b] == UP:
self.state[b] = DOWN
if b == BTN_SW and callback:
callback(1)
elif self.state[b] == DOWN and t - self.timer[b] >= HOLD_TIME:
self.state[b] = HELD
if b == BTN_R: event = SELECT
elif b == BTN_SW and not callback: event = ESCAPE
else:
if self.state[b] != UP and b == BTN_SW and callback:
callback(0)
if self.state[b] == DOWN:
if b == BTN_R: event = INC
elif b == BTN_SW and not callback: event = DEC
self.state[b] = UP
if self.encvalue > 0: event = INC
elif self.encvalue < 0: event = DEC
self.encvalue = 0
time.sleep(idle)
return event
def lcd_clear(self, now=True):
"""Clear the LCD
Sends a clear command to the LCD and clears the
text and blink buffers.
Args:
now: if False just clear buffers to reduce flickering,
but requires an update to take effect
"""
if now:
self._lcd_send(0x01)
self._lcd_setcursorpos(0, 0)
time.sleep(2e-3)
self.buffer = [" " * COLS for _ in range(ROWS)]
self.written = [[""] * COLS for _ in range(ROWS)]
self.blinked = [[""] * COLS for _ in range(ROWS)]
self.scrollpos = [0] * ROWS
self.scrolltimer = 0
self.blinktimer = 0
def lcd_write(self, text, row, col=0, mode='', now=False):
"""Writes text to the LCD
Writes text to the LCD starting at row, col. Characters are
stored in a buffer until the user calls update(). Can be
called with now=True if the LCD needs to be updated now,
usually because another process would delay updates.
Args:
text: string to write
row: the row at which to start writing
col: the column at which to start writing
mode: if 'ljust' or 'rjust' pad with spaces, 'scroll' scrolls
text to the right if it is long enough, otherwise place text
starting at row, col
now: if True update LCD now
"""
if mode == 'scroll':
if len(text) > COLS:
self.buffer[row] = text
self.scrollpos[row] = -SCROLL_PAUSE
else:
mode = 'ljust'
if mode == 'ljust':
self.buffer[row] = text[:COLS].ljust(COLS)
elif mode == 'rjust':
self.buffer[row] = text[-COLS:].rjust(COLS)
elif mode != 'scroll':
self.buffer[row] = (self.buffer[row][:col]
+ text[: COLS-col]
+ self.buffer[row][col+len(text) :])[:COLS]
if now: self.update(idle=0)
def lcd_blink(self, text, row=0, col=0, mode='', delay=BLINK_TIME):
"""Blink a character/message on the LCD
Write text on the LCD that disappears after a delay. Text
written by lcd_write() will reappear. Calling this with
an empty string removes any current blinks. If a blink
is already in progress when this is called, the new one
is ignored.
Args:
text: string to write, '' to clear blinks
row: the row at which to place text
col: the column at which to place text
mode: if 'ljust' or 'rjust' pad with spaces,
otherwise place text starting at row, col
delay: time to wait before removing text
"""
if text == '':
self.blinked = [[""] * COLS for _ in range(ROWS)]
self.blinktimer = 0
elif self.blinktimer == 0:
if mode == 'ljust':
text = text[: COLS-col].ljust(COLS-col)
elif mode == 'rjust':
text = text[col-COLS :].rjust(COLS-col)
for i, c in enumerate(text[: COLS-col]):
self.blinked[row][col + i] = c
self.blinktimer = time.time() + delay
def progresswheel_start(self):
"""Shows an animation while another process runs
Displays a spinning character in the lower right corner of the
LCD that runs in a thread after this function returns, to give
the user some feedback while a long-running process completes.
"""
self.spinning = True
self.spin = threading.Thread(target=self._progresswheel_spin)
self.spin.start()
def progresswheel_stop(self):
"""Removes the spinning character"""
self.spinning = False
self.spin.join()
def waitfortap(self, t=0):
"""Waits until a button is pressed or some time has passed
Args:
t: seconds to wait, if 0 wait forever
Returns: True if button was pressed, False if time expired
"""
tstop = time.time() + t
while True:
if t and time.time() > tstop:
return False
if self.update(callback=False) != NULL:
return True
def choose_opt(self, opts, row, i=0, mode='rjust', timeout=MENU_TIMEOUT):
"""Lets the user choose an option from a list
Displays options from a list of choices. User can scroll through
options with DEC/INC, choose with SELECT, or cancel with ESCAPE
or timeout. Menu options can scroll.
Args:
opts: list of strings to display as the choices
row: the row on which to show the choices
i: index of the choice to display first
mode: 'rjust' by default - see lcd_write()
timeout: seconds to wait, if -1 wait forever
Returns: index of option chosen, or -1 if canceled
"""
i = i % len(opts)
while True:
self.lcd_write(opts[i], row, mode=mode)
t = time.time()
while timeout == -1 or time.time() - t < timeout:
event = self.update(callback=False)
if event == INC:
i = (i + 1) % len(opts)
break
elif event == DEC:
i = (i - 1) % len(opts)
break
elif event == SELECT:
self._lcd_flash(opts[i], row, mode)
return i
elif event == ESCAPE:
timeout = 0
else:
# self.lcd_write(" " * COLS, row)
return -1
def choose_val(self, val, minval, maxval, inc, fmt=f'>{COLS}', timeout=MENU_TIMEOUT, func=None):
"""Lets the user modify a numeric parameter
Displays a number on the bottom row of the LCD and allows the user to
scroll its value over a range by specified increment using DEC/INC.
A function can be called with the current value after each increment
to demonstrate the result. User can set value with SELECT, or
cancel with ESCAPE or timeout.
Args:
val: the starting value
minval: the lower limit of the value
maxval: the upper limit of the value
inc: the step size to change when incrementing/decrementing the value
fmt: a format specifier for printing the value nicely
timeout: seconds to wait, if -1 forever
func: a function to call with the value every time it changes
Returns: selected value, or None if time expired or ESCAPE
"""
while True:
self.lcd_write(format(val, fmt), ROWS - 1, mode='rjust')
t = time.time()
while timeout == -1 or time.time() - t < timeout:
event = self.update(callback=False)
if event == INC:
val = min(val + inc, maxval)
if func: func(val)
break
elif event == DEC:
val = max(val - inc, minval)
if func: func(val)
break
elif event == SELECT:
self._lcd_flash(format(val, fmt), ROWS - 1, mode='rjust')
return val
elif event == ESCAPE:
timeout = 0
else:
return None
def confirm_choice(self, text='', row=ROWS-1, timeout=MENU_TIMEOUT):
"""Offers a yes/no choice
Displays some text and lets the user toggle between a check mark
and an X with DEC/INC and choose with SELECT.
Args:
text: string to write
row: the row to display the choice
timeout: seconds to wait, if -1 wait forever
Returns: 1 if check is selected, 0 if time expires or ESCAPE
"""
self.lcd_write(text[:COLS - 1], row)
c = 1
while True:
self.lcd_write([XMARK, CHECK][c], row, COLS - 1)
t = time.time()
while timeout == -1 or time.time() - t < timeout:
event = self.update(callback=False)
if event in (DEC, INC):
c ^= 1
break
elif event == SELECT:
if c: self._lcd_flash(text[:COLS], row, mode='ljust')
return c
elif event == ESCAPE:
timeout = 0
else:
return 0
def char_input(self, text=' ', row=ROWS-1, i=-1,
timeout=MENU_TIMEOUT, charset=FNCHARS):
"""Allows user to enter text with a rotary encoder and button
There are two cursor modes, which are toggled using SELECT. The
blinking square allows the current character to be changed using
DEC/INC. The underline cursor changes position with DEC/INC. ESCAPE
ends edit mode, and allows the user to confirm or cancel the input.
Args:
text: the initial text to be edited
row: the row in which to show the input
i: initial cursor position, from end if negative
timeout: seconds to wait before canceling, forever if -1
charset: the set of allowed characters
Returns: the edited string, or empty string if canceled
"""
if i < 0: i = len(text) + i
c = charset.find(text[i])
mode = 'blink'
self._lcd_setcursormode(mode)
while True:
if mode == 'blink':
self.lcd_write(text[max(0, i+1-COLS) : max(COLS, i+1)],
row, mode='ljust', now=True)
else:
self.lcd_write(charset[c], row, min(i, COLS-1), now=True)
self._lcd_setcursorpos(row, min(i, COLS-1))
t = time.time()
while timeout == -1 or time.time() - t < timeout:
event = self.update(callback=False)
if event == NULL: continue
if event == INC:
if mode == 'blink':
i = min(i + 1, len(text))
if i == len(text): text += ' '
c = charset.find(text[i])
else:
c = (c + 1) % len(charset)
text = text[0:i] + charset[c] + text[i+1 :]
elif event == DEC:
if mode == 'blink':
i = max(i - 1, 0)
c = charset.find(text[i])
else:
c = (c - 1) % len(charset)
text = text[0:i] + charset[c] + text[i+1 :]
elif event == SELECT:
mode = 'blink' if mode == 'line' else 'line'
self._lcd_setcursormode(mode)
elif event == ESCAPE:
self._lcd_setcursormode('hide')
if self.confirm_choice(text.strip()[1-COLS :], row=row):
return text.strip().replace(BACKSLASH, '\\').replace(TILDE, '~')
else:
return ''
break
else:
self._lcd_setcursormode('hide')
return ''
def choose_file(self, topdir, last='', ext=None):
"""Lets user browse and select a file on the system
Finds files of a specified type on the file system and lets the
user choose one using choose_opt(). Timeout is disabled and
filenames can scroll. Can move up and down through the
directory tree up to a specified limit. First two arguments must
be pathlib.Path objects. Shows the current file on the bottom
row and the current directory on the row above it.
Args:
topdir: Path of the highest-level directory the user may see
last: Path of the file to show as the initial choice
ext: the file extensions to show, if None shows all files
Returns: Path of the chosen file or empty string if canceled
"""
cdir = topdir if last == '' else (last.parent if last.parent > topdir else topdir)
while True:
self.lcd_write(f"{str(cdir.relative_to(topdir.parent))}/:", ROWS - 2, mode='scroll')
x = sorted([p for p in cdir.glob('*') if p.is_dir() or p.suffix == ext or ext == None])
y = [f"{SUBDIR}{p.name}/" if p.is_dir() else p.name for p in x]
i = x.index(last) if last in x else 0
if cdir != topdir:
x.append(cdir.parent)
y.append("../")
j = self.choose_opt(y, ROWS - 1, i, mode='scroll', timeout=-1)
if j < 0: return ''
if x[j].is_dir():
last = cdir
cdir = x[j]
else:
return x[j]
def display_error(self, err, msg="", etype=None, tb=None):
"""Displays Exception text on the LCD
Reformats the text of an Exception so it can be displayed on one
line and scrolls it across the bottom row of the LCD, and also prints
information to stdout. Waits for the user to press a button, then
returns if possible.
Args:
err: the Exception
msg: an optional error message
"""
if etype == KeyboardInterrupt:
sys.exit()
err_oneline = msg + re.sub(' {2,}', ' ', re.sub('\n|\^', ' ', str(err)))
self.lcd_write(err_oneline, ROWS - 1, mode='scroll')
if msg: print(msg)
if tb:
traceback.print_exception(etype, err, tb)
else:
print(err)
self.waitfortap()
@staticmethod
def shell_cmd(cmd, **kwargs):
"""Executes a shell command and returns the output
Uses subprocess.run to execute a shell command and returns the output
as ascii with trailing newlines removed. Blocks until shell command
has returned.
Args:
cmd: text of the command line to execute
kwargs: additional keyword arguments passed to subprocess.run
Returns: the stripped ascii STDOUT of the command
"""
return subprocess.run(cmd, check=True, stdout=subprocess.PIPE, shell=True,
encoding='ascii', **kwargs).stdout.rstrip('\n')
@staticmethod
def gpio_set(pin, state):
"""Sets the state of a GPIO
Sets a GPIO high or low, as long as it isn't being used by something
else. PIN_OUT can be modified to add outputs, as long as they don't
conflict with those defined above for the LCD, buttons, and GPIOs
18, 19, and 21 (which are used by the DAC).
Args:
pin: pin number (BCM numbering)
state: True for high, False for low
"""
if pin in PIN_OUT:
if state: GPIO.output(pin, GPIO.HIGH)
else: GPIO.output(pin, GPIO.LOW)
def wifi_state(self, state=''):
"""Checks or sets the state of the wifi adapter
Turns the wifi adapter on or off, or simply returns its current
state. Does not determine whether it has connected to a network,
only that it is enabled or disabled.
Args:
setstate: 'on' or 'off' to set the state, empty string just
checks the current state
Returns: 'on' or 'off'
"""
if state:
self.shell_cmd(f"sudo nmcli radio wifi {state}")
else:
state = 'on' if self.shell_cmd("nmcli radio wifi") == 'enabled' else 'off'
self.wifistatus = WIFIDOWN if state == 'off' else WIFIUP
return state
def wifi_settings(self):
"""Displays a wifi settings menu
Shows the connection status and current IP address(es) of the Pi
and a list of any available wifi networks. Allows the user to
enable/disable wifi and enter passkeys for visible networks
in order to connect.
"""
while True:
self.lcd_clear()
if ip := sb.shell_cmd("hostname -I"):
self.lcd_write(f"Connected as {ip}", ROWS - 2, mode='scroll')
else:
self.lcd_write("Not connected", ROWS - 2, mode='ljust')
if self.wifi_state() == 'off':
if self.choose_opt(["Enable WiFi"], row=ROWS - 1) == 0:
self.wifi_state('on')
break
else:
nw = sb.shell_cmd("nmcli -g IN-USE,SSID dev wifi").split('\n')
ssid = [x[0].replace('*', CHECK) + x[2:] for x in nw if x[2:]]
opts = ssid + ["Scan", "Disable WiFi"]
i = max(''.join(s[0] for s in ssid).find(CHECK), 0)
j = self.choose_opt(opts, ROWS - 1, i, 'scroll', timeout=-1)
if j < 0: break
elif opts[j] == "Disable WiFi":
self.wifi_state('off')
break
elif opts[j] == "Scan":
pass
elif opts[j][0] == CHECK:
self.lcd_write(opts[j][1:], ROWS - 2, mode='ljust')
self.lcd_write("disconnecting ", ROWS - 1, mode='rjust', now=True)
self.progresswheel_start()
sb.shell_cmd(f"sudo nmcli con down {opts[j][1:]}")
self.progresswheel_stop()
continue
else:
self.lcd_write(opts[j][1:], ROWS - 2, mode='ljust')
self.lcd_write("connecting ", ROWS - 1, mode='rjust', now=True)
self.progresswheel_start()
try: x = sb.shell_cmd(f"sudo nmcli con up {opts[j][1:]}")
except subprocess.CalledProcessError: x = -1
self.progresswheel_stop()
if x == -1:
self.lcd_write("Password:", ROWS - 2, mode='ljust')
psk = self.char_input(charset = USCHARS)
if psk == '': continue
self.lcd_write("connecting ", ROWS - 1, mode='rjust', now=True)
self.progresswheel_start()
try: x = sb.shell_cmd(f"sudo nmcli dev wifi connect {opts[j][1:]} password {psk}")
except subprocess.CalledProcessError: x = -1
self.progresswheel_stop()
if x == -1:
self.lcd_write("connection fail", ROWS - 1, mode='rjust')
self.waitfortap(MENU_TIMEOUT)
continue
self.lcd_write("scanning ", ROWS - 1, mode='rjust', now=True)
self.progresswheel_start()
sb.shell_cmd("sudo nmcli -g IN-USE,SSID dev wifi")
self.progresswheel_stop()
def _button_event(self, button):
t = time.time()
self.timer[button] = t
def _encoder_event(self, channel):
for channel in ROT_L, ROT_R:
self.encstate = (self.encstate << 1) % 64
self.encstate += 1 if GPIO.input(channel) == ACTIVE else 0
if self.encstate == 0b111000:
self.encvalue += 1
elif self.encstate == 0b110100:
self.encvalue -= 1
def _progresswheel_spin(self):
while True:
for x in BACKSLASH + '|/-':
if not self.spinning: return
self._lcd_putchars(x, ROWS - 1, COLS - 1)
time.sleep(BLINK_TIME)
def _lcd_flash(self, text, row, mode='', n=3):
text = text[:COLS].rjust(COLS) if mode == 'rjust' else text[:COLS].ljust(COLS)
for _ in range(n):
self._lcd_putchars(' ' * COLS, row, 0)
time.sleep(BLINK_TIME)
self._lcd_putchars(text, row, 0)
time.sleep(BLINK_TIME)
def _lcd_putchars(self, chars, row, col):
lastcol = -2
for c in chars:
if self.written[row][col] != c:
if lastcol != col - 1:
self._lcd_setcursorpos(row, col)
self._lcd_send(ord(c), 1)
self.written[row][col] = c
lastcol = col
col += 1
def _lcd_setcursorpos(self, row, col):
if row < ROWS and col < COLS:
offset = [0x00, 0x40, COLS, 0x40 + COLS]
self._lcd_send(0x80 | offset[row] + col)
def _lcd_setcursormode(self, mode):
if mode == 'hide':
self._lcd_send(0x0c | 0x00)
elif mode == 'blink':
self._lcd_send(0x0d)
elif mode == 'line':
self._lcd_send(0x0e)
@staticmethod
def _lcd_send(val, reg=0):
if LCD_RS == 0: return
GPIO.output(LCD_RS, reg)
GPIO.output(LCD_EN, GPIO.LOW)
for nib in (val >> 4, val):
for i in range(4):
GPIO.output(LCD_DATA[i], (nib >> i) & 0x01)
GPIO.output(LCD_EN, GPIO.HIGH)
time.sleep(EXEC_TIME)
GPIO.output(LCD_EN, GPIO.LOW)
class FluidBox:
"""Manages a SquishBox interface to FluidPatcher"""
def __init__(self):
"""Creates the FluidBox"""
self.pno = 0
self.buttonstate = 0
self.shutdowntimer = 0
fp.midi_callback = self.listener
sb.buttoncallback = self.handle_buttonevent
def handle_buttonevent(self, val):
"""Handles callback events when the stompbutton state changes
Sends a momentary and toggling MIDI message, and toggles sets the LED
to match the state of the toggle.
"""
fp.send_event(type='cc', chan=STOMP_CHAN, par1=STOMP_MOMENT, par2=val)
if val:
self.buttonstate ^= 1
fp.send_event(type='cc', chan=STOMP_CHAN, par1=STOMP_TOGGLE, par2=self.buttonstate)
if PIN_LED: sb.gpio_set(PIN_LED, self.buttonstate)
def listener(self, sig):
"""Handles MidiSignals from FluidPatcher
Receives MidiSignal instances in response to incoming MIDI events
or custom events triggered by router rules. MidiSignals for custom
events have a `val` parameter that is the result of parameter
routing, and additional parameters corresponding to the rule
parameters. The following custom rules are handled:
- `patch`: a patch name or index to be selected. If `patch` has a '+' or '-'
suffix, increment the current patch index instead.
- `lcdwrite`: a string to be written to the LCD, right-justified. If `format`
is provided, the formatted `val` parameter is appended
- `setpin`: the *index* of the pin in PIN_OUT to set using `val`. If the LED
is set, set the state of the button toggle to match
"""
if 'val' in sig:
if 'patch' in sig:
if sig.patch == -1:
self.pno = (self.pno + sig.val) % len(fp.patches)
else:
self.pno = sig.patch
elif 'lcdwrite' in sig:
if 'format' in sig:
val = format(sig.val, sig.format)
self.lcdwrite = f"{sig.lcdwrite} {val}"
else:
self.lcdwrite = sig.lcdwrite
elif 'setpin' in sig:
if PIN_OUT[sig.setpin] == PIN_LED:
self.buttonstate = 1 if sig.val else 0
sb.gpio_set(PIN_OUT[sig.setpin], sig.val)
elif 'event' in sig and sig.val > 0:
sb.nextevent = sig.event
elif 'shutdown' in sig:
if sig.val > 0:
self.shutdowntimer = time.time()
else:
self.shutdowntimer = 0
else:
self.lastsig = sig
def patchmode(self):
"""Applies a patch and displays the main screen"""
if fp.patches: warn = fp.apply_patch(self.pno)
else: warn = fp.apply_patch('')
if MIDI_CTRL: self.add_midicontrols()
pno = self.pno
sb.lcd_clear(now=False)
while True:
if fp.patches:
sb.lcd_write(fp.patches[self.pno], 0, mode='scroll')
if warn:
sb.lcd_write('; '.join(warn), 1, mode='scroll')
sb.waitfortap()
sb.lcd_write(f"patch: {self.pno + 1}/{len(fp.patches)}", 1, mode='rjust')
else:
sb.lcd_write("No patches", 0, mode='ljust')
if warn:
sb.lcd_write('; '.join(warn), 1, mode='scroll')
sb.waitfortap()
sb.lcd_write("patch 0/0", 1, mode='rjust')
sb.lcd_write(sb.wifistatus, 1, 0)
warn = []
self.lastsig = None
self.lcdwrite = None
while True:
if pno != self.pno:
return
if self.shutdowntimer and time.time() - self.shutdowntimer > 5:
sb.lcd_write("Shutting down..", 0, mode='ljust')
sb.lcd_write("Wait 15s, unplug", 1, mode='ljust', now=True)
sb.shell_cmd("sudo poweroff")
sys.exit()
if self.lastsig:
sb.lcd_blink(MIDIACT, 1, 1)
self.lastsig = None
if self.lcdwrite:
sb.lcd_blink('')
sb.lcd_blink(self.lcdwrite, 1, mode='rjust', delay=MENU_TIMEOUT)
self.lcdwrite = None
event = sb.update()
if event == NULL:
continue
if event == INC and fp.patches:
self.pno = (self.pno + 1) % len(fp.patches)
return
elif event == DEC and fp.patches:
self.pno = (self.pno - 1) % len(fp.patches)
return
elif event == SELECT:
k = sb.choose_opt(['Load Bank', 'Save Bank', 'Save Patch', 'Delete Patch',
'Open Soundfont', 'Effects..', 'System Menu..'], row=1)
if k == 0:
if self.load_bank():
return
elif k == 1:
self.save_bank()
elif k == 2:
sb.lcd_write("Save patch:", 0, mode='ljust')
newname = sb.char_input(fp.patches[self.pno])
if newname != '':
if newname != fp.patches[self.pno]:
fp.add_patch(newname, addlike=self.pno)
fp.update_patch(newname)
self.pno = fp.patches.index(newname)
elif k == 3:
if sb.confirm_choice('Delete', row=1):
fp.delete_patch(self.pno)
self.pno = min(self.pno, len(fp.patches) - 1)
return
elif k == 4:
if sfont := sb.choose_file(fp.sfdir, ext='.sf2'):
self.sfmode(sfont)
sb.lcd_write("loading bank ", 1, mode='ljust', now=True)
sb.progresswheel_start()
fp.load_bank()
sb.progresswheel_stop()
return
elif k == 5:
self.effects_menu()
elif k == 6:
self.system_menu()
break
def sfmode(self, sfont):
"""Soundfont preset chooser"""
sb.lcd_write(sfont.name, 0, mode='scroll', now=True)
sb.lcd_write("loading presets ", 1, mode='rjust', now=True)
sb.progresswheel_start()
if not (presets := fp.solo_soundfont(sfont)):
sb.progresswheel_stop()
sb.lcd_write(f"Unable to load presets from {str(sfont)}", 1, mode='scroll')
sb.waitfortap()
return
sb.progresswheel_stop()
i = 0
warn = fp.select_sfpreset(sfont, *presets[i])
while True:
bank, prog, name = presets[i]
sb.lcd_write(name, 0, mode='scroll')
if warn:
sb.lcd_write('; '.join(warn), 1, mode='scroll')
sb.waitfortap()
warn = []
sb.lcd_write(f"preset {bank:03}:{prog:03}", 1, mode='rjust')
while True:
event = sb.update(callback=False)
if event == INC:
i = (i + 1) % len(presets)
warn = fp.select_sfpreset(sfont, *presets[i])
break
elif event == DEC:
i = (i - 1) % len(presets)
warn = fp.select_sfpreset(sfont, *presets[i])
break
elif event == SELECT:
sb.lcd_write("Add as Patch:", 0, mode='ljust')
newname = sb.char_input(name)
if newname:
self.pno = fp.add_patch(newname)
fp.update_patch(newname)
break
elif event == ESCAPE: return
def load_bank(self, bank=""):
"""Bank loading menu"""
lastbank = fp.currentbank
lastpatch = fp.patches[self.pno] if fp.patches else ""
if bank == "":
last = fp.bankdir / fp.currentbank if fp.currentbank else ""
bank = sb.choose_file(fp.bankdir, last, '.yaml')
if bank == "": return False
sb.lcd_write(bank.name, 0, mode='scroll', now=True)
sb.lcd_write("loading bank ", 1, mode='ljust', now=True)
sb.progresswheel_start()
try: fp.load_bank(bank)
except Exception as e:
sb.progresswheel_stop()
sb.display_error(e, "bank load error: ")
return False
sb.progresswheel_stop()
fp.update_config()
if fp.currentbank == lastbank and lastpatch in fp.patches:
self.pno = fp.patches.index(lastpatch)
else:
self.pno = 0
return True
def save_bank(self, bank=""):
"""Bank saving menu"""
if bank == "":
bank = sb.choose_file(fp.bankdir, fp.bankdir / fp.currentbank, '.yaml')
if bank == "": return
name = sb.char_input(bank.name)
if name == "": return
bank = bank.parent / name
try: fp.save_bank(bank.with_suffix('.yaml'))
except Exception as e:
sb.display_error(e, "bank save error: ")
else:
fp.update_config()
sb.lcd_write("bank saved", 1, mode='ljust')
sb.waitfortap(2)
def effects_menu(self):
"""FluidSynth effects setting menu"""
i=0
fxmenu_info = (
# Name fluidsetting min max inc format
('Reverb Size', 'synth.reverb.room-size', 0.0, 1.0, 0.1, '4.1f'),
('Reverb Damp', 'synth.reverb.damp', 0.0, 1.0, 0.1, '4.1f'),
('Rev. Width', 'synth.reverb.width', 0.0, 100.0, 0.5, '5.1f'),
('Rev. Level', 'synth.reverb.level', 0.00, 1.00, 0.01, '5.2f'),
('Chorus Voices', 'synth.chorus.nr', 0, 99, 1, '2d'),
('Chor. Level', 'synth.chorus.level', 0.0, 10.0, 0.1, '4.1f'),
('Chor. Speed', 'synth.chorus.speed', 0.1, 21.0, 0.1, '4.1f'),
('Chorus Depth', 'synth.chorus.depth', 0.3, 5.0, 0.1, '3.1f'),
('Gain', 'synth.gain', 0.0, 5.0, 0.1, '11.1f'))
vals = [fp.fluidsetting_get(info[1]) for info in fxmenu_info]
fxopts = [fxmenu_info[i][0] + ':' + format(vals[i], fxmenu_info[i][5]) for i in range(len(fxmenu_info))]
while True:
sb.lcd_write("Effects:", 0, mode='ljust')
i = sb.choose_opt(fxopts, 1, i)
if i < 0:
break
sb.lcd_write(fxopts[i], 0, mode='ljust')
newval = sb.choose_val(vals[i], *fxmenu_info[i][2:], func=lambda x: fp.fluidsetting_set(fxmenu_info[i][1], x))
if newval != None:
fp.fluidsetting_set(fxmenu_info[i][1], newval, patch=self.pno)
vals[i] = newval
fxopts[i] = fxmenu_info[i][0] + ':' + format(newval, fxmenu_info[i][5])
else:
fp.fluidsetting_set(fxmenu_info[i][1], vals[i])
def system_menu(self):
"""System functions and settings menu"""
sb.lcd_write("System Menu:", 0, mode='ljust')
k = sb.choose_opt(['Power Down', 'MIDI Devices', 'Wifi Settings', 'USB File Copy'], row=1)
if k == 0:
sb.lcd_write("Shutting down..", 0, mode='ljust')
sb.lcd_write("Wait 15s, unplug", 1, mode='ljust', now=True)
sb.shell_cmd("sudo poweroff")
sys.exit()
elif k == 1:
self.midi_devices()
elif k == 2:
sb.wifi_settings()
elif k == 3:
self.usb_filecopy()
def midi_devices(self):
"""Menu for connecting MIDI devices and monitoring"""
sb.lcd_write("MIDI Devices:", 0, mode='ljust')
readable = re.findall(" (\d+): '([^\n]*)'", sb.shell_cmd("aconnect -i"))
rports, rnames = list(zip(*readable))
p = sb.choose_opt([*rnames, "MIDI monitor.."], row=1, mode='scroll', timeout=-1)
if p < 0: return
if 0 <= p < len(rports):