#!/usr/bin/env python3 # ========================================================================== # # # # KVMD - The main PiKVM daemon. # # # # Copyright (C) 2018-2022 Maxim Devaev # # # # This program is free software: you can redistribute it and/or modify # # it under the terms of the GNU General Public License as published by # # the Free Software Foundation, either version 3 of the License, or # # (at your option) any later version. # # # # This program is distributed in the hope that it will be useful, # # but WITHOUT ANY WARRANTY; without even the implied warranty of # # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # # GNU General Public License for more details. # # # # You should have received a copy of the GNU General Public License # # along with this program. If not, see . # # # # ========================================================================== # import sys import csv import textwrap import dataclasses from typing import Set from typing import List from typing import Optional import Xlib.keysymdef.latin1 import Xlib.keysymdef.miscellany import Xlib.keysymdef.xf86 import Xlib.keysymdef.xkb import mako.template # ===== @dataclasses.dataclass(frozen=True) class _OtgKey: code: int is_modifier: bool @dataclasses.dataclass(frozen=True) class _Ps2Key: code: int type: str @dataclasses.dataclass(frozen=True) class _X11Key: name: str code: int shift: bool @dataclasses.dataclass(frozen=True) class _KeyMapping: web_name: str mcu_code: int arduino_name: str otg_key: _OtgKey ps2_key: _Ps2Key at1_code: int x11_keys: Set[_X11Key] def _resolve_keysym(name: str) -> int: code: Optional[int] = None for module in [ Xlib.keysymdef.latin1, Xlib.keysymdef.miscellany, Xlib.keysymdef.xf86, Xlib.keysymdef.xkb, ]: code = getattr(module, name, None) if code is not None: break assert code is not None, name return code def _parse_x11_names(names: str) -> Set[_X11Key]: keys: Set[_X11Key] = set() for name in filter(None, names.split(",")): shift = name.startswith("^") name = (name[1:] if shift else name) code = _resolve_keysym(name) keys.add(_X11Key(name, code, shift)) return keys def _parse_otg_key(key: str) -> _OtgKey: is_modifier = key.startswith("^") code = int((key[1:] if is_modifier else key), 16) return _OtgKey(code, is_modifier) def _parse_ps2_key(key: str) -> _Ps2Key: (code_type, raw_code) = key.split(":") return _Ps2Key( code=int(raw_code, 16), type=code_type, ) def _read_keymap_csv(path: str) -> List[_KeyMapping]: keymap: List[_KeyMapping] = [] with open(path) as keymap_file: for row in csv.DictReader(keymap_file): if len(row) >= 6: keymap.append(_KeyMapping( web_name=row["web_name"], mcu_code=int(row["mcu_code"]), arduino_name=row["arduino_name"], otg_key=_parse_otg_key(row["otg_key"]), ps2_key=_parse_ps2_key(row["ps2_key"]), at1_code=int(row["at1_code"], 16), x11_keys=_parse_x11_names(row["x11_names"] or ""), )) return keymap def _render_keymap(keymap: List[_KeyMapping], template_path: str, out_path: str) -> None: with open(template_path) as template_file: with open(out_path, "w") as out_file: template = textwrap.dedent(template_file.read()) rendered = mako.template.Template(template).render(keymap=keymap) out_file.write(rendered) # ===== def main() -> None: # https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/code/code_values # https://github.com/NicoHood/HID/blob/master/src/KeyboardLayouts/ImprovedKeylayouts.h # https://github.com/Harvie/ps2dev/blob/master/src/ps2dev.h # https://gist.github.com/MightyPork/6da26e382a7ad91b5496ee55fdc73db2 # https://github.com/qemu/keycodemapdb/blob/master/data/keymaps.csv # Hut1_12v2.pdf # Fields list: # - Web # - MCU code # - Arduino name # - OTG code (^ for mod) # - PS/2 key # - AT set1 # - X11 keysyms (^ for shift) assert len(sys.argv) == 4, f"{sys.argv[0]}