logo

qmk_firmware

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

keycodes.py (6582B)


  1. """Used by the make system to generate keycodes.h from keycodes_{version}.json
  2. """
  3. from milc import cli
  4. from qmk.constants import GPL2_HEADER_C_LIKE, GENERATED_HEADER_C_LIKE
  5. from qmk.commands import dump_lines
  6. from qmk.path import normpath
  7. from qmk.keycodes import load_spec
  8. def _translate_group(group):
  9. """Fix up any issues with badly chosen values
  10. """
  11. if group == 'modifiers':
  12. return 'modifier'
  13. if group == 'media':
  14. return 'consumer'
  15. return group
  16. def _render_key(key):
  17. width = 7
  18. if 'S(' in key:
  19. width += len('S()')
  20. if 'A(' in key:
  21. width += len('A()')
  22. if 'RCTL(' in key:
  23. width += len('RCTL()')
  24. if 'ALGR(' in key:
  25. width += len('ALGR()')
  26. return key.ljust(width)
  27. def _render_label(label):
  28. label = label.replace("\\", "(backslash)")
  29. return label
  30. def _generate_ranges(lines, keycodes):
  31. lines.append('')
  32. lines.append('enum qk_keycode_ranges {')
  33. lines.append('// Ranges')
  34. for key, value in keycodes["ranges"].items():
  35. lo, mask = map(lambda x: int(x, 16), key.split("/"))
  36. hi = lo + mask
  37. define = value.get("define")
  38. lines.append(f' {define.ljust(30)} = 0x{lo:04X},')
  39. lines.append(f' {(define + "_MAX").ljust(30)} = 0x{hi:04X},')
  40. lines.append('};')
  41. def _generate_defines(lines, keycodes):
  42. lines.append('')
  43. lines.append('enum qk_keycode_defines {')
  44. lines.append('// Keycodes')
  45. for key, value in keycodes["keycodes"].items():
  46. lines.append(f' {value.get("key")} = {key},')
  47. lines.append('')
  48. lines.append('// Alias')
  49. for key, value in keycodes["keycodes"].items():
  50. temp = value.get("key")
  51. for alias in value.get("aliases", []):
  52. lines.append(f' {alias.ljust(10)} = {temp},')
  53. lines.append('};')
  54. def _generate_helpers(lines, keycodes):
  55. lines.append('')
  56. lines.append('// Range Helpers')
  57. for value in keycodes["ranges"].values():
  58. define = value.get("define")
  59. lines.append(f'#define IS_{define}(code) ((code) >= {define} && (code) <= {define + "_MAX"})')
  60. # extract min/max
  61. temp = {}
  62. for key, value in keycodes["keycodes"].items():
  63. group = value.get('group', None)
  64. if not group:
  65. continue
  66. if group not in temp:
  67. temp[group] = [0xFFFF, 0]
  68. key = int(key, 16)
  69. if key < temp[group][0]:
  70. temp[group][0] = key
  71. if key > temp[group][1]:
  72. temp[group][1] = key
  73. lines.append('')
  74. lines.append('// Group Helpers')
  75. for group, codes in temp.items():
  76. lo = keycodes["keycodes"][f'0x{codes[0]:04X}']['key']
  77. hi = keycodes["keycodes"][f'0x{codes[1]:04X}']['key']
  78. lines.append(f'#define IS_{ _translate_group(group).upper() }_KEYCODE(code) ((code) >= {lo} && (code) <= {hi})')
  79. lines.append('')
  80. lines.append('// Switch statement Helpers')
  81. for group, codes in temp.items():
  82. lo = keycodes["keycodes"][f'0x{codes[0]:04X}']['key']
  83. hi = keycodes["keycodes"][f'0x{codes[1]:04X}']['key']
  84. name = f'{ _translate_group(group).upper() }_KEYCODE_RANGE'
  85. lines.append(f'#define { name.ljust(35) } {lo} ... {hi}')
  86. def _generate_aliases(lines, keycodes):
  87. # Work around ChibiOS ch.h include guard
  88. if 'CH_H' in [value['key'] for value in keycodes['aliases'].values()]:
  89. lines.append('')
  90. lines.append('#undef CH_H')
  91. lines.append('')
  92. lines.append('// Aliases')
  93. for key, value in keycodes["aliases"].items():
  94. define = _render_key(value.get("key"))
  95. val = _render_key(key)
  96. if 'label' in value:
  97. lines.append(f'#define {define} {val} // {_render_label(value.get("label"))}')
  98. else:
  99. lines.append(f'#define {define} {val}')
  100. lines.append('')
  101. for key, value in keycodes["aliases"].items():
  102. for alias in value.get("aliases", []):
  103. lines.append(f'#define {alias} {value.get("key")}')
  104. def _generate_version(lines, keycodes, prefix=''):
  105. version = keycodes['version']
  106. major, minor, patch = map(int, version.split('.'))
  107. bcd = f'0x{major:02d}{minor:02d}{patch:04d}'
  108. lines.append('')
  109. lines.append(f'#define QMK_{prefix}KEYCODES_VERSION "{version}"')
  110. lines.append(f'#define QMK_{prefix}KEYCODES_VERSION_BCD {bcd}')
  111. lines.append(f'#define QMK_{prefix}KEYCODES_VERSION_MAJOR {major}')
  112. lines.append(f'#define QMK_{prefix}KEYCODES_VERSION_MINOR {minor}')
  113. lines.append(f'#define QMK_{prefix}KEYCODES_VERSION_PATCH {patch}')
  114. @cli.argument('-v', '--version', arg_only=True, required=True, help='Version of keycodes to generate.')
  115. @cli.argument('-o', '--output', arg_only=True, type=normpath, help='File to write to')
  116. @cli.argument('-q', '--quiet', arg_only=True, action='store_true', help="Quiet mode, only output error messages")
  117. @cli.subcommand('Used by the make system to generate keycodes.h from keycodes_{version}.json', hidden=True)
  118. def generate_keycodes(cli):
  119. """Generates the keycodes.h file.
  120. """
  121. # Build the keycodes.h file.
  122. keycodes_h_lines = [GPL2_HEADER_C_LIKE, GENERATED_HEADER_C_LIKE, '#pragma once', '// clang-format off']
  123. keycodes = load_spec(cli.args.version)
  124. _generate_version(keycodes_h_lines, keycodes)
  125. _generate_ranges(keycodes_h_lines, keycodes)
  126. _generate_defines(keycodes_h_lines, keycodes)
  127. _generate_helpers(keycodes_h_lines, keycodes)
  128. # Show the results
  129. dump_lines(cli.args.output, keycodes_h_lines, cli.args.quiet)
  130. @cli.argument('-v', '--version', arg_only=True, required=True, help='Version of keycodes to generate.')
  131. @cli.argument('-l', '--lang', arg_only=True, required=True, help='Language of keycodes to generate.')
  132. @cli.argument('-o', '--output', arg_only=True, type=normpath, help='File to write to')
  133. @cli.argument('-q', '--quiet', arg_only=True, action='store_true', help="Quiet mode, only output error messages")
  134. @cli.subcommand('Used by the make system to generate keymap_{lang}.h from keycodes_{lang}_{version}.json', hidden=True)
  135. def generate_keycode_extras(cli):
  136. """Generates the header file.
  137. """
  138. # Build the header file.
  139. keycodes_h_lines = [GPL2_HEADER_C_LIKE, GENERATED_HEADER_C_LIKE, '#pragma once', '#include "keycodes.h"', '// clang-format off']
  140. keycodes = load_spec(cli.args.version, cli.args.lang)
  141. _generate_version(keycodes_h_lines, keycodes, f'{cli.args.lang.upper()}_')
  142. _generate_aliases(keycodes_h_lines, keycodes)
  143. # Show the results
  144. dump_lines(cli.args.output, keycodes_h_lines, cli.args.quiet)