keyboard_h.py 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. """Used by the make system to generate keyboard.h from info.json.
  2. """
  3. from pathlib import Path
  4. from milc import cli
  5. from qmk.path import normpath
  6. from qmk.info import info_json
  7. from qmk.commands import dump_lines
  8. from qmk.keyboard import keyboard_completer, keyboard_folder
  9. from qmk.constants import COL_LETTERS, ROW_LETTERS, GPL2_HEADER_C_LIKE, GENERATED_HEADER_C_LIKE
  10. def _generate_layouts(keyboard):
  11. """Generates the layouts.h file.
  12. """
  13. # Build the info.json file
  14. kb_info_json = info_json(keyboard)
  15. if 'matrix_size' not in kb_info_json:
  16. cli.log.error(f'{keyboard}: Invalid matrix config.')
  17. return []
  18. col_num = kb_info_json['matrix_size']['cols']
  19. row_num = kb_info_json['matrix_size']['rows']
  20. lines = []
  21. for layout_name in kb_info_json['layouts']:
  22. if kb_info_json['layouts'][layout_name]['c_macro']:
  23. continue
  24. if 'matrix' not in kb_info_json['layouts'][layout_name]['layout'][0]:
  25. cli.log.debug(f'{keyboard}/{layout_name}: No matrix data!')
  26. continue
  27. layout_keys = []
  28. layout_matrix = [['KC_NO' for i in range(col_num)] for i in range(row_num)]
  29. for i, key in enumerate(kb_info_json['layouts'][layout_name]['layout']):
  30. row = key['matrix'][0]
  31. col = key['matrix'][1]
  32. identifier = 'k%s%s' % (ROW_LETTERS[row], COL_LETTERS[col])
  33. try:
  34. layout_matrix[row][col] = identifier
  35. layout_keys.append(identifier)
  36. except IndexError:
  37. key_name = key.get('label', identifier)
  38. cli.log.error(f'Matrix data out of bounds for layout {layout_name} at index {i} ({key_name}): [{row}, {col}]')
  39. return []
  40. lines.append('')
  41. lines.append('#define %s(%s) {\\' % (layout_name, ', '.join(layout_keys)))
  42. rows = ', \\\n'.join(['\t {' + ', '.join(row) + '}' for row in layout_matrix])
  43. rows += ' \\'
  44. lines.append(rows)
  45. lines.append('}')
  46. for alias, target in kb_info_json.get('layout_aliases', {}).items():
  47. lines.append('')
  48. lines.append(f'#ifndef {alias}')
  49. lines.append(f'# define {alias} {target}')
  50. lines.append('#endif')
  51. return lines
  52. @cli.argument('-i', '--include', nargs='?', arg_only=True, help='Optional file to include')
  53. @cli.argument('-o', '--output', arg_only=True, type=normpath, help='File to write to')
  54. @cli.argument('-q', '--quiet', arg_only=True, action='store_true', help="Quiet mode, only output error messages")
  55. @cli.argument('-kb', '--keyboard', arg_only=True, type=keyboard_folder, completer=keyboard_completer, required=True, help='Keyboard to generate keyboard.h for.')
  56. @cli.subcommand('Used by the make system to generate keyboard.h from info.json', hidden=True)
  57. def generate_keyboard_h(cli):
  58. """Generates the keyboard.h file.
  59. """
  60. keyboard_h = cli.args.include
  61. dd_layouts = _generate_layouts(cli.args.keyboard)
  62. valid_config = dd_layouts or keyboard_h
  63. # Build the layouts.h file.
  64. keyboard_h_lines = [GPL2_HEADER_C_LIKE, GENERATED_HEADER_C_LIKE, '#pragma once', '#include "quantum.h"']
  65. keyboard_h_lines.append('')
  66. keyboard_h_lines.append('// Layout content')
  67. if dd_layouts:
  68. keyboard_h_lines.extend(dd_layouts)
  69. if keyboard_h:
  70. keyboard_h_lines.append(f'#include "{Path(keyboard_h).name}"')
  71. # Protect against poorly configured keyboards
  72. if not valid_config:
  73. keyboard_h_lines.append('#error("<keyboard>.h is required unless your keyboard uses data-driven configuration. Please rename your keyboard\'s header file to <keyboard>.h")')
  74. # Show the results
  75. dump_lines(cli.args.output, keyboard_h_lines, cli.args.quiet)