logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

keymap.py (5570B)


  1. """This script automates the copying of the default keymap into your own keymap.
  2. """
  3. import re
  4. import json
  5. import shutil
  6. from pathlib import Path
  7. from milc import cli
  8. from milc.questions import question, choice
  9. from qmk.constants import HAS_QMK_USERSPACE, QMK_USERSPACE
  10. from qmk.errors import NoSuchKeyboardError
  11. from qmk.path import is_keyboard, keymaps, keymap
  12. from qmk.git import git_get_username
  13. from qmk.decorators import automagic_keyboard, automagic_keymap
  14. from qmk.keyboard import keyboard_completer, keyboard_folder
  15. from qmk.userspace import UserspaceDefs
  16. from qmk.json_schema import json_load
  17. from qmk.json_encoders import KeymapJSONEncoder
  18. from qmk.info import info_json
  19. def _list_available_converters(kb_name):
  20. """Search for converters that can be applied to a given keyboard
  21. """
  22. if not is_keyboard(kb_name):
  23. return None
  24. info = info_json(kb_name)
  25. pin_compatible = info.get('pin_compatible')
  26. if not pin_compatible:
  27. return None
  28. return sorted(folder.name.split('_to_')[-1] for folder in Path('platforms').glob(f'*/converters/{pin_compatible}_to_*'))
  29. def _set_converter(file, converter):
  30. """add/overwrite any existing converter specified in keymap.json
  31. """
  32. json_data = json_load(file) if file.exists() else {}
  33. json_data['converter'] = converter
  34. output = json.dumps(json_data, cls=KeymapJSONEncoder, sort_keys=True)
  35. file.write_text(output + '\n', encoding='utf-8')
  36. def validate_keymap_name(name):
  37. """Returns True if the given keymap name contains only a-z, 0-9 and underscore characters.
  38. """
  39. regex = re.compile(r'^[a-zA-Z0-9][a-zA-Z0-9_]+$')
  40. return bool(regex.match(name))
  41. def prompt_keyboard():
  42. prompt = """{fg_yellow}Select Keyboard{style_reset_all}
  43. If you're unsure you can view a full list of supported keyboards with {fg_yellow}qmk list-keyboards{style_reset_all}.
  44. Keyboard Name? """
  45. return question(prompt)
  46. def prompt_user():
  47. prompt = """
  48. {fg_yellow}Name Your Keymap{style_reset_all}
  49. Keymap name? """
  50. return question(prompt, default=git_get_username())
  51. def prompt_converter(kb_name):
  52. prompt = """
  53. {fg_yellow}Configure Development Board{style_reset_all}
  54. For more information, see:
  55. https://docs.qmk.fm/feature_converters
  56. Use converter? """
  57. converters = _list_available_converters(kb_name)
  58. if not converters:
  59. return None
  60. choices = ['No (default)', *converters]
  61. answer = choice(prompt, options=choices, default=0)
  62. return None if choices.index(answer) == 0 else answer
  63. @cli.argument('-kb', '--keyboard', type=keyboard_folder, completer=keyboard_completer, help='Specify keyboard name. Example: 1upkeyboards/1up60hse')
  64. @cli.argument('-km', '--keymap', help='Specify the name for the new keymap directory')
  65. @cli.argument('--converter', help='Specify the name of a converter to configure')
  66. @cli.argument('--skip-converter', arg_only=True, action='store_true', help='Skip converter')
  67. @cli.subcommand('Creates a new keymap for the keyboard of your choosing')
  68. @automagic_keyboard
  69. @automagic_keymap
  70. def new_keymap(cli):
  71. """Creates a new keymap for the keyboard of your choosing.
  72. """
  73. cli.log.info('{style_bright}Generating a new keymap{style_normal}')
  74. cli.echo('')
  75. # ask for user input if keyboard or keymap was not provided in the command line
  76. kb_name = cli.config.new_keymap.keyboard if cli.config.new_keymap.keyboard else prompt_keyboard()
  77. user_name = cli.config.new_keymap.keymap if cli.config.new_keymap.keymap else prompt_user()
  78. converter = cli.config.new_keymap.converter if cli.args.skip_converter or cli.config.new_keymap.converter else prompt_converter(kb_name)
  79. # check directories
  80. try:
  81. kb_name = keyboard_folder(kb_name)
  82. except ValueError:
  83. cli.log.error(f'Keyboard {{fg_cyan}}{kb_name}{{fg_reset}} does not exist! Please choose a valid name.')
  84. return False
  85. # validate before any keymap ops
  86. try:
  87. keymaps_dirs = keymaps(kb_name)
  88. keymap_path_new = keymaps_dirs[0] / user_name
  89. except NoSuchKeyboardError:
  90. cli.log.error(f'Keymap folder for {{fg_cyan}}{kb_name}{{fg_reset}} does not exist!')
  91. return False
  92. keymap_path_default = keymap(kb_name, 'default')
  93. if not keymap_path_default:
  94. cli.log.error(f'Default keymap for {{fg_cyan}}{kb_name}{{fg_reset}} does not exist!')
  95. return False
  96. if not validate_keymap_name(user_name):
  97. cli.log.error('Keymap names must contain only {fg_cyan}a-z{fg_reset}, {fg_cyan}0-9{fg_reset} and {fg_cyan}_{fg_reset}! Please choose a different name.')
  98. return False
  99. if keymap_path_new.exists():
  100. cli.log.error(f'Keymap {{fg_cyan}}{user_name}{{fg_reset}} already exists! Please choose a different name.')
  101. return False
  102. # create user directory with default keymap files
  103. shutil.copytree(keymap_path_default, keymap_path_new, symlinks=True)
  104. if converter:
  105. _set_converter(keymap_path_new / 'keymap.json', converter)
  106. # end message to user
  107. cli.log.info(f'{{fg_green}}Created a new keymap called {{fg_cyan}}{user_name}{{fg_green}} in: {{fg_cyan}}{keymap_path_new}{{fg_reset}}.')
  108. cli.log.info(f"Compile a firmware with your new keymap by typing: {{fg_yellow}}qmk compile -kb {kb_name} -km {user_name}{{fg_reset}}.")
  109. # Add to userspace compile if we have userspace available
  110. if HAS_QMK_USERSPACE:
  111. userspace = UserspaceDefs(QMK_USERSPACE / 'qmk.json')
  112. userspace.add_target(keyboard=kb_name, keymap=user_name, do_print=False)
  113. return userspace.save()