keychord.py/code.py

439 lines
12 KiB
Python

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()