import time import board import math import random import busio import usb_hid from digitalio import DigitalInOut, Direction, Pull import displayio from vectorio import Rectangle, Circle, Polygon from adafruit_display_text import label from adafruit_bitmap_font import bitmap_font from adafruit_hid.keyboard import Keyboard from adafruit_hid.keyboard_layout_us import KeyboardLayoutUS import gc9a01 from characters import characters, shortcuts, words, keycodes kbd = Keyboard(usb_hid.devices) layout = KeyboardLayoutUS(kbd) font = bitmap_font.load_font("fonts/VictorMonoNerdFont-SemiBoldItalic-32.pcf") displayio.release_displays() mode_labels = [ ["Letters", "Shortcuts", "Words", "Predictive"], ["Lettres", "Raccourcis", "Mots", "Prédictif"], ] language_labels = ["English", "Français"] backlight_labels = ["Backlight", "Rétroéclairage"] tft_clk = board.GP14 tft_mosi = board.GP15 tft_rst = board.GP27 tft_dc = board.GP28 tft_cs = board.GP29 tft_bl = DigitalInOut(board.GP26) tft_bl.direction = Direction.OUTPUT tft_bl.value = True key0 = DigitalInOut(board.GP5) key0.pull = Pull.UP key1 = DigitalInOut(board.GP6) key1.pull = Pull.UP key2 = DigitalInOut(board.GP4) key2.pull = Pull.UP key3 = DigitalInOut(board.GP3) key3.pull = Pull.UP key4 = DigitalInOut(board.GP1) key4.pull = Pull.UP key5 = DigitalInOut(board.GP2) key5.pull = Pull.UP key6 = DigitalInOut(board.GP0) key6.pull = Pull.UP spi = busio.SPI(clock=tft_clk, MOSI=tft_mosi) display_bus = displayio.FourWire(spi, command=tft_dc, chip_select=tft_cs, reset=tft_rst) display = gc9a01.GC9A01(display_bus, width=240, height=240) main_display = displayio.Group() display.root_group = main_display label_position_radius = 90 circle_radius = 16 def create_settings_text(y): text = label.Label( font, text="", color=0xFFFFFF, anchor_point=(0.5, 0.5), anchored_position=(0, 0) ) text_group = displayio.Group(scale=1) text_group.x = 120 text_group.y = y text_group.append(text) main_display.append(text_group) return text top_text = create_settings_text(50) middle_text = create_settings_text(110) bottom_text = create_settings_text(170) def calculate_text_position(key_angle, label_position_radius=label_position_radius): x = int(120 + label_position_radius * math.sin(key_angle)) y = int(120 - label_position_radius * math.cos(key_angle)) return x, y def create_text_area_angle( key_angle, text_colour=0xFFFFFF, label_position_radius=label_position_radius ): return create_text_area( *calculate_text_position( key_angle, text_colour=text_colour, label_position_radius=label_position_radius, ) ) def create_text_area(x, y, text_colour=0xFFFFFF): key_display = displayio.Group(scale=1) key_display.x = x key_display.y = y key_text = label.Label( font, text=" ", color=text_colour, anchor_point=(0.5, 0.5), anchored_position=(0, 0), ) key_display.append(key_text) main_display.append(key_display) return key_text def create_circle(x, y, key_colour, circle_radius=circle_radius): key_palette = displayio.Palette(1) key_circle = Circle(pixel_shader=key_palette, radius=circle_radius, x=x, y=y) main_display.append(key_circle) return key_palette, key_circle def create_key_display( key_angle, key_colour, text_colour=0xFFFFFF, circle_radius=circle_radius, label_position_radius=label_position_radius, ): x, y = calculate_text_position( key_angle, label_position_radius=label_position_radius ) key_palette, key_circle = create_circle( x, y, key_colour, circle_radius=circle_radius ) key_text = create_text_area(x, y, text_colour=text_colour) return key_text, key_palette, key_colour, key_circle key0_text, key0_palette, key0_colour, key0_circle = create_key_display( -2.0 * math.pi / 3.0, (219, 97, 71) ) key1_text, key1_palette, key1_colour, key1_circle = create_key_display( -math.pi / 3.0, (100, 150, 40) ) key2_text, key2_palette, key2_colour, key2_circle = create_key_display( 0.0, (40, 150, 100) ) key3_text, key3_palette, key3_colour, key3_circle = create_key_display( math.pi / 3.0, (100, 40, 150) ) key4_text, key4_palette, key4_colour, key4_circle = create_key_display( 2.0 * math.pi / 3.0, (40, 40, 150) ) selected_circle_radius = 32 selected_text, selected_palette, selected_colour, selected_circle = create_key_display( math.pi, (255, 255, 255), text_colour=0x000000, circle_radius=selected_circle_radius, label_position_radius=70, ) key5_colour = (0, 100, 30) key5_palette = displayio.Palette(1) key5_circle = Circle(pixel_shader=key5_palette, radius=circle_radius, x=120 - 32, y=120) main_display.append(key5_circle) key6_colour = (15, 100, 200) key6_palette = displayio.Palette(1) key6_circle = Circle(pixel_shader=key6_palette, radius=circle_radius, x=120 + 32, y=120) main_display.append(key6_circle) def swap_to_settings_display(): tft_bl.value = True key0_text.text = "" key1_text.text = "" key2_text.text = "" key3_text.text = "" key4_text.text = "" selected_text = "" key0_circle.radius = 0 key1_circle.radius = 0 key2_circle.radius = 0 key3_circle.radius = 0 key4_circle.radius = 0 key5_circle.radius = 0 key6_circle.radius = 0 selected_circle.radius = 0 def settings_display(mode, language, backlight, location): top_text.text = mode_labels[language][mode] top_text.color = 0x1C4D2D if location == 0 else 0xFFFFFF middle_text.text = language_labels[language] middle_text.color = 0x1C4D2D if location == 1 else 0xFFFFFF bottom_text.text = backlight_labels[language] bottom_text.color = ( (0x1C4D2D if backlight else 0xFF0000) if location == 2 else 0xFFFFFF ) def swap_away_from_settings_display(): top_text.text = "" middle_text.text = "" bottom_text.text = "" def swap_to_character_display(): swap_away_from_settings_display() key0_circle.radius = circle_radius key1_circle.radius = circle_radius key2_circle.radius = circle_radius key3_circle.radius = circle_radius key4_circle.radius = circle_radius key5_circle.radius = circle_radius key6_circle.radius = circle_radius selected_circle.radius = selected_circle_radius def edit_character(pin, character, key_text, key_palette, colour): if not pin: key_palette[0] = (0, 0, 0) if character is None: key_text.text = "" else: key_text.text = character else: key_text.text = "" key_palette[0] = colour def character_display(pin0, pin1, pin2, pin3, pin4, selection): edit_character( pin0, characters[1][pin1][pin2][pin3][pin4][selection], key0_text, key0_palette, key0_colour, ) edit_character( pin1, characters[pin0][1][pin2][pin3][pin4][selection], key1_text, key1_palette, key1_colour, ) edit_character( pin2, characters[pin0][pin1][1][pin3][pin4][selection], key2_text, key2_palette, key2_colour, ) edit_character( pin3, characters[pin0][pin1][pin2][1][pin4][selection], key3_text, key3_palette, key3_colour, ) edit_character( pin4, characters[pin0][pin1][pin2][pin3][1][selection], key4_text, key4_palette, key4_colour, ) if any([pin0, pin1, pin2, pin3, pin4]): character = characters[pin0][pin1][pin2][pin3][pin4][selection] if character is None: selected_palette[0] = (0, 0, 0) selected_text.text = "" else: selected_text.text = character selected_palette[0] = selected_colour else: selected_text.text = "" selected_palette[0] = (0, 0, 0) def mode_display(selection): if selection is 0: key5_palette[0] = (0, 0, 0) key6_palette[0] = (0, 0, 0) elif selection is 1: key5_palette[0] = key5_colour key6_palette[0] = (0, 0, 0) elif selection is 2: key5_palette[0] = (0, 0, 0) key6_palette[0] = key6_colour elif selection is 3: key5_palette[0] = key5_colour key6_palette[0] = key6_colour def pressed(key): return not key.value def poll(): return [ pressed(key0), pressed(key1), pressed(key2), pressed(key3), pressed(key4), ], [pressed(key5), pressed(key6)] def toggle(_p, p, t): updates = update(_p, p) toggles = [not _t if u and __p else _t for _t, u, __p in zip(t, updates, _p)] return any(updates), toggles def update(_p, p): return [___p != __p for ___p, __p in zip(_p, p)] def get_time(p, t): return [_t + 1 if _p else 0 for _p, _t in zip(p, t)] def send_keycode(v): if v is not None: kbd.send(*keycodes[v]) def send_character(p, s): send_keycode(characters[p[0]][p[1]][p[2]][p[3]][p[4]][s]) def send_shortcut(p, s): send_keycode(shortcuts[p[0]][p[1]][p[2]][p[3]][p[4]][s]) def send_word(p, s, l): w = words[l][p[0]][p[1]][p[2]][p[3]][p[4]][s] if w is not None: layout.write(w) layout.write(" ") def release_countdown(p, t, start=3): return [max(_t - 1, 0) if not _p else start for _p, _t in zip(p, t)] def main(rate=0.05, start=3): p = [False] * 5 t = [0] * 5 m = [False] * 2 tm = [0] * 2 toggles = [False] * 2 selection = 0 settings = False mode = 0 language = 0 backlight = True counter = 0 while True: _p, _m = poll() t = release_countdown(_p, t, start=start) tm = get_time(_m, tm) if any(_p) or any(_m): counter = 0 else: if counter < 100: counter += 1 if counter == 100: tft_bl.value = False elif counter == 0: tft_bl.value = backlight if all([_tm > 60 for _tm in tm]): if not settings: settings = True toggles = [False] * 2 swap_to_settings_display() location = 0 tm = [0] * 2 elif settings: u = update(_m, m) if u[0] and _m[0]: location += 1 if location > 2: location = 0 settings = not settings if u[1] and _m[1]: if location == 0: mode += 1 if mode > 3: mode = 0 elif location == 1: language += 1 if language > 1: language = 0 elif location == 2: backlight = not backlight if settings: if any(u): settings_display(mode, language, backlight, location) else: tft_bl.value = backlight swap_to_character_display() elif not settings: u = any(update(_p, p)) um, toggles = toggle(_m, m, toggles) if um: selection = toggles[0] + 2 * toggles[1] if backlight: mode_display(selection) if (u or um) and backlight and mode == 0: character_display(*_p, selection) if u and not any(_p): pressed = [_t > 0 for _t in t] if mode == 0: send_character(pressed, selection) elif mode == 1: send_shortcut(pressed, selection) elif mode == 2: send_word(pressed, selection, language) p = _p m = _m time.sleep(rate) main()