mirror of
				https://github.com/mfulz/qmk_firmware.git
				synced 2025-10-25 03:29:59 +02:00 
			
		
		
		
	Refactor compiler code
This commit is contained in:
		
							parent
							
								
									20a3229faf
								
							
						
					
					
						commit
						32c7832609
					
				| @ -1,4 +1,13 @@ | |||||||
| # encoding: utf-8 | #!/usr/bin/env python | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | """Compiler for keymap.c files | ||||||
|  | 
 | ||||||
|  | This scrip will generate a keymap.c file from a simple | ||||||
|  | markdown file with a specific layout. | ||||||
|  | 
 | ||||||
|  | Usage: | ||||||
|  |     python compile_keymap.py INPUT_PATH [OUTPUT_PATH] | ||||||
|  | """ | ||||||
| from __future__ import division | from __future__ import division | ||||||
| from __future__ import print_function | from __future__ import print_function | ||||||
| from __future__ import absolute_import | from __future__ import absolute_import | ||||||
| @ -15,299 +24,423 @@ import collections | |||||||
| PY2 = sys.version_info.major == 2 | PY2 = sys.version_info.major == 2 | ||||||
| 
 | 
 | ||||||
| if PY2: | if PY2: | ||||||
| 	chr = unichr |     chr = unichr | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ONELINE_COMMENT_RE = re.compile(r"^\s*//.*$", re.MULTILINE) | BASEPATH = os.path.abspath(os.path.join( | ||||||
| INLINE_COMMENT_RE = re.compile( |     os.path.dirname(__file__), "..", ".." | ||||||
|     r"([\,\"\[\]\{\}\d])\s+//\s[^\"\]\}\{\[]*$", re.MULTILINE | )) | ||||||
| ) | 
 | ||||||
| TRAILING_COMMA_RE = re.compile( | 
 | ||||||
|     r",$\s*([\]\}])", re.MULTILINE | KEYBOARD_LAYOUTS = { | ||||||
| ) |     # These map positions in the parsed layout to | ||||||
|  |     # positions in the KEYMAP MATRIX | ||||||
|  |     'ergodox_ez': [ | ||||||
|  |         [ 0,  1,  2,  3,  4,  5,  6],  [38, 39, 40, 41, 42, 43, 44], | ||||||
|  |         [ 7,  8,  9, 10, 11, 12, 13],  [45, 46, 47, 48, 49, 50, 51], | ||||||
|  |         [14, 15, 16, 17, 18, 19    ],  [    52, 53, 54, 55, 56, 57], | ||||||
|  |         [20, 21, 22, 23, 24, 25, 26],  [58, 59, 60, 61, 62, 63, 64], | ||||||
|  |         [27, 28, 29, 30, 31        ],  [        65, 66, 67, 68, 69], | ||||||
|  |         [                    32, 33],  [70, 71                    ], | ||||||
|  |         [                        34],  [72                        ], | ||||||
|  |         [                35, 36, 37],  [73, 74, 75                ], | ||||||
|  |     ] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | BLANK_LAYOUTS = [ | ||||||
|  | # Compact Layout | ||||||
|  | """ | ||||||
|  | .------------------------------------.------------------------------------. | ||||||
|  | |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|  | !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|  | |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|  | !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|  | |     |    |    |    |    |    |-----!-----!    |    |    |    |    |     | | ||||||
|  | !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|  | |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|  | '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|  |  |    |    |    |    |    |                     !    |    |    |    |    | | ||||||
|  |  '------------------------'                     '------------------------' | ||||||
|  |                         .-----------. .-----------. | ||||||
|  |                         |     |     | !     |     | | ||||||
|  |                   .-----+-----+-----! !-----+-----+-----. | ||||||
|  |                   !     !     |     | !     |     !     ! | ||||||
|  |                   !     !     !-----! !-----!     !     ! | ||||||
|  |                   |     |     |     | !     |     |     | | ||||||
|  |                   '-----------------' '-----------------' | ||||||
|  | """, | ||||||
|  | 
 | ||||||
|  | # Wide Layout | ||||||
|  | """ | ||||||
|  | .--------------------------------------------. .--------------------------------------------. | ||||||
|  | |      |     |     |     |     |     |       | !       |     |     |     |     |     |      | | ||||||
|  | !------+-----+-----+-----+-----+-------------! !-------+-----+-----+-----+-----+-----+------! | ||||||
|  | |      |     |     |     |     |     |       | !       |     |     |     |     |     |      | | ||||||
|  | !------+-----+-----+-----x-----x-----!       ! !       !-----x-----x-----+-----+-----+------! | ||||||
|  | |      |     |     |     |     |     |-------! !-------!     |     |     |     |     |      | | ||||||
|  | !------+-----+-----+-----x-----x-----!       ! !       !-----x-----x-----+-----+-----+------! | ||||||
|  | |      |     |     |     |     |     |       | !       |     |     |     |     |     |      | | ||||||
|  | '------+-----+-----+-----+-----+-------------' '-------------+-----+-----+-----+-----+------' | ||||||
|  |  |     |     |     |     |     |                             !     |     |     |     |     | | ||||||
|  |  '-----------------------------'                             '-----------------------------' | ||||||
|  |                              .---------------. .---------------. | ||||||
|  |                              |       |       | !       |       | | ||||||
|  |                      .-------+-------+-------! !-------+-------+-------. | ||||||
|  |                      !       !       |       | !       |       !       ! | ||||||
|  |                      !       !       !-------! !-------!       !       ! | ||||||
|  |                      |       |       |       | !       |       |       | | ||||||
|  |                      '-----------------------' '-----------------------' | ||||||
|  | """, | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | DEFAULT_CONFIG = { | ||||||
|  |     "includes_basedir": "quantum/", | ||||||
|  |     "keymaps_includes": [ | ||||||
|  |         "keymap_common.h", | ||||||
|  |     ], | ||||||
|  |     'filler': "-+.':x", | ||||||
|  |     'separator': "|", | ||||||
|  |     'default_key_prefix': ["KC_"], | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | SECTIONS = [ | ||||||
|  |     'layout_config', | ||||||
|  |     'layers', | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #       Markdown Parsing | ||||||
| 
 | 
 | ||||||
| def loads(raw_data): | def loads(raw_data): | ||||||
|  |     ONELINE_COMMENT_RE = re.compile(r""" | ||||||
|  |         ^                       # comment must be at the start of the line | ||||||
|  |         \s*                     # arbitrary whitespace | ||||||
|  |         //                      # start of the comment | ||||||
|  |         (.*)                    # the comment | ||||||
|  |         $                       # until the end of line | ||||||
|  |         """, re.MULTILINE | re.VERBOSE | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     INLINE_COMMENT_RE = re.compile(r""" | ||||||
|  |         (?:[\,\"\[\]\{\}\d])    # anythig that might end a expression | ||||||
|  |         \s+                     # comment must be preceded by whitespace | ||||||
|  |         //                      # start of the comment | ||||||
|  |         \s                      # and succeded by whitespace | ||||||
|  |         ([^\"\]\}\{\[]*)        # the comment (except things which might be json) | ||||||
|  |         $                       # until the end of line | ||||||
|  |         """, re.MULTILINE | re.VERBOSE | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     TRAILING_COMMA_RE = re.compile(r""" | ||||||
|  |         ,                       # the comma | ||||||
|  |         \s*                     # arbitrary whitespace (including newlines) | ||||||
|  |         ([\]\}])                # end of an array or object | ||||||
|  |         """, re.MULTILINE | re.VERBOSE | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|     if isinstance(raw_data, bytes): |     if isinstance(raw_data, bytes): | ||||||
|         raw_data = raw_data.decode('utf-8') |         raw_data = raw_data.decode('utf-8') | ||||||
|  | 
 | ||||||
|     raw_data = ONELINE_COMMENT_RE.sub(r"", raw_data) |     raw_data = ONELINE_COMMENT_RE.sub(r"", raw_data) | ||||||
|     raw_data = INLINE_COMMENT_RE.sub(r"\1", raw_data) |     raw_data = INLINE_COMMENT_RE.sub(r"\1", raw_data) | ||||||
|     raw_data = TRAILING_COMMA_RE.sub(r"\1", raw_data) |     raw_data = TRAILING_COMMA_RE.sub(r"\1", raw_data) | ||||||
|     return json.loads(raw_data) |     return json.loads(raw_data) | ||||||
| 
 | 
 | ||||||
| with io.open("keymap.md", encoding="utf-8") as fh: |  | ||||||
| 	lines = fh.readlines() |  | ||||||
| 
 | 
 | ||||||
| SECTIONS = [ | def parse_config(path): | ||||||
| 	'layout_config', |     def reset_section(): | ||||||
| 	'layers', |         section.update({ | ||||||
| ] |             'name': section.get('name', ""), | ||||||
|  |             'sub_name': "", | ||||||
|  |             'start_line': -1, | ||||||
|  |             'end_line': -1, | ||||||
|  |             'code_lines': [], | ||||||
|  |         }) | ||||||
| 
 | 
 | ||||||
| config = { |     def start_section(line_index, line): | ||||||
| 	"includes_basedir": "quantum/", |         end_section() | ||||||
|     "keymaps_includes": [ |         if line.startswith("# "): | ||||||
|         "keymap_common.h", |             name = line[2:] | ||||||
|     ], |         elif line.startswith("## "): | ||||||
| 	'filler': "-+.':x", |             name = line[3:] | ||||||
| 	'separator': "|", |  | ||||||
|     'default_key_prefix': ["KC_"], |  | ||||||
|     'unicode_macros': [], |  | ||||||
|     'macro_ids': ['UMS'], |  | ||||||
|     'layers': collections.OrderedDict(), |  | ||||||
| 	'layer_lines': collections.OrderedDict(), |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| section_start_index = -1 |         name = name.strip().replace(" ", "_").lower() | ||||||
| current_section = None |         if name in SECTIONS: | ||||||
| current_layer_name = None |             section['name'] = name | ||||||
| current_layer_lines = [] |         else: | ||||||
| config_data = [] |             section['sub_name'] = name | ||||||
|  |         section['start_line'] = line_index | ||||||
| 
 | 
 | ||||||
| def end_section(): |     def end_section(): | ||||||
| 	global section_start_index |         if section['start_line'] >= 0: | ||||||
| 	global current_layer_lines |             if section['name'] == 'layout_config': | ||||||
| 	section_start_index = -1 |                 config.update(loads("\n".join( | ||||||
| 	if current_section == 'layout_config': |                     section['code_lines'] | ||||||
| 		config.update(loads("".join( |                 ))) | ||||||
| 			config_data |             elif section['sub_name'].startswith('layer'): | ||||||
| 		))) |                 layer_name = section['sub_name'] | ||||||
| 	elif current_section == 'layers': |                 config['layer_lines'][layer_name] = section['code_lines'] | ||||||
| 		config['layer_lines'][current_layer_name] = current_layer_lines | 
 | ||||||
| 		current_layer_lines = [] |         reset_section() | ||||||
|  | 
 | ||||||
|  |     def amend_section(line_index, line): | ||||||
|  |         section['end_line'] = line_index | ||||||
|  |         section['code_lines'].append(line) | ||||||
|  | 
 | ||||||
|  |     config = DEFAULT_CONFIG.copy() | ||||||
|  |     config.update({ | ||||||
|  |         'layer_lines': collections.OrderedDict(), | ||||||
|  |         'macro_ids': {'UM'}, | ||||||
|  |         'unicode_macros': {}, | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     section = {} | ||||||
|  |     reset_section() | ||||||
|  | 
 | ||||||
|  |     with io.open(path, encoding="utf-8") as fh: | ||||||
|  |         for i, line in enumerate(fh): | ||||||
|  |             if line.startswith("#"): | ||||||
|  |                 start_section(i, line) | ||||||
|  |             elif line.startswith("    "): | ||||||
|  |                 amend_section(i, line[4:]) | ||||||
|  |             else: | ||||||
|  |                 # TODO: maybe parse description | ||||||
|  |                 pass | ||||||
|  | 
 | ||||||
|  |     end_section() | ||||||
|  |     return config | ||||||
|  | 
 | ||||||
|  | #       header file parsing | ||||||
|  | 
 | ||||||
|  | IF0_RE = re.compile(r""" | ||||||
|  |     ^ | ||||||
|  |     #if 0 | ||||||
|  |     $.*? | ||||||
|  |     #endif | ||||||
|  |     """, re.MULTILINE | re.DOTALL | re.VERBOSE | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| for i, line in enumerate(lines): | COMMENT_RE = re.compile(r""" | ||||||
| 	if line.startswith("# "): |     /\* | ||||||
| 		section = line[2:].strip().replace(" ", "_").lower() |     .*? | ||||||
| 		if section in SECTIONS: |     \*/" | ||||||
| 			current_section = section |     """, re.MULTILINE | re.DOTALL | re.VERBOSE | ||||||
| 	elif line.startswith("## "): | ) | ||||||
| 		sub_section = line[3:] |  | ||||||
| 		if current_section == 'layers': |  | ||||||
| 			current_layer_name = sub_section.strip() |  | ||||||
| 			# TODO: parse descriptio |  | ||||||
| 			config['layers'][current_layer_name] = "" |  | ||||||
| 	elif line.startswith("    "): |  | ||||||
| 		if section_start_index < 0: |  | ||||||
| 			section_start_index = i |  | ||||||
| 		if current_section == 'layout_config': |  | ||||||
| 			config_data.append(line) |  | ||||||
| 		elif current_section == 'layers': |  | ||||||
| 			if not line.strip(): |  | ||||||
| 				continue |  | ||||||
| 			current_layer_lines.append(line) |  | ||||||
| 	elif section_start_index > 0: |  | ||||||
| 		end_section() |  | ||||||
| 
 | 
 | ||||||
| end_section() | def read_header_file(path): | ||||||
| 
 |     with io.open(path, encoding="utf-8") as fh: | ||||||
| KEYDEF_RE = re.compile(r"#define ((?:{})(?:\w+))".format( |         data = fh.read() | ||||||
| 	"|".join(config['key_prefixes']) |     data, _ = COMMENT_RE.subn("", data) | ||||||
| )) |     data, _ = IF0_RE.subn("", data) | ||||||
| IF0_RE = re.compile(r"^#if 0$.*?#endif", re.MULTILINE | re.DOTALL) |     return data | ||||||
| COMMENT_RE = re.compile(r"/\*.*?\*/", re.MULTILINE | re.DOTALL) |  | ||||||
| ENUM_RE = re.compile(r"(enum\s\w+\s\{.*?\};)", re.MULTILINE | re.DOTALL) |  | ||||||
| ENUM_KEY_RE = re.compile(r"({}\w+)".format( |  | ||||||
| 	"|".join(config['key_prefixes']) |  | ||||||
| )) |  | ||||||
| 
 |  | ||||||
| def parse_keydefs(path): |  | ||||||
| 	with io.open(path, encoding="utf-8") as fh: |  | ||||||
| 		data = fh.read() |  | ||||||
| 	data, _ = COMMENT_RE.subn("", data) |  | ||||||
| 	data, _ = IF0_RE.subn("", data) |  | ||||||
| 
 |  | ||||||
| 	for match in KEYDEF_RE.finditer(data): |  | ||||||
| 		yield match.groups()[0] |  | ||||||
| 
 |  | ||||||
| 	for enum_match in ENUM_RE.finditer(data): |  | ||||||
| 		enum = enum_match.groups()[0] |  | ||||||
| 		for key_match in ENUM_KEY_RE.finditer(enum): |  | ||||||
| 			yield key_match.groups()[0] |  | ||||||
| 
 |  | ||||||
| valid_keycodes = set() |  | ||||||
| basepath = os.path.abspath(os.path.join( |  | ||||||
| 	os.path.dirname(__file__), "..", "..", "..", ".." |  | ||||||
| )) |  | ||||||
| 
 |  | ||||||
| valid_keycodes.update(parse_keydefs(os.path.join( |  | ||||||
| 	basepath, "tmk_core", "common", "keycode.h" |  | ||||||
| ))) |  | ||||||
| 
 |  | ||||||
| for include_path in config['keymaps_includes']: |  | ||||||
| 	path = os.path.join(basepath, config['includes_dir'], include_path) |  | ||||||
| 	path = path.replace("/", os.sep) |  | ||||||
| 	if os.path.exists(path): |  | ||||||
| 		valid_keycodes.update(parse_keydefs(path)) |  | ||||||
| 
 |  | ||||||
| LAYER_CHANGE_RE = re.compile(r"(DF|TG|MO)\(\d+\)") |  | ||||||
| MACRO_RE = re.compile(r"M\(\w+\)") |  | ||||||
| UNICODE_RE = re.compile(r"U[0-9A-F]{4}") |  | ||||||
| NON_CODE = re.compile(r"^[^A-Z0-9_]$") |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def UNICODE_MACRO(config, c): | def regex_partial(re_str_fmt, flags=re.MULTILINE | re.DOTALL | re.VERBOSE): | ||||||
| 	# TODO: don't use macro for codepoints below 0x2000 |     def partial(*args, **kwargs): | ||||||
| 	macro_id = "UC_" + ( |         re_str = re_str_fmt.format(*args, **kwargs) | ||||||
| 		unicodedata.name(c) |         return re.compile(re_str, flags) | ||||||
| 		.replace(" ", "_") |     return partial | ||||||
| 		.replace("-", "_") |  | ||||||
| 		.replace("SUPERSCRIPT_", "SUP_") |  | ||||||
| 		.replace("SUBSCRIPT_", "SUB_") |  | ||||||
| 		.replace("GREEK_SMALL_LETTER", "GR_LC") |  | ||||||
| 		.replace("GREEK_CAPITAL_LETTER", "GR_UC") |  | ||||||
| 		.replace("VULGAR_FRACTION_", "FR_") |  | ||||||
| 	) |  | ||||||
| 	if macro_id not in config['macro_ids']: |  | ||||||
| 		config['macro_ids'].append(macro_id) |  | ||||||
| 	code = "{:04X}".format(ord(c)) |  | ||||||
| 	if (macro_id, code) not in config['unicode_macros']: |  | ||||||
| 		config['unicode_macros'].append((macro_id, code)) |  | ||||||
| 	return "M({})".format(macro_id) |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def MACRO(config, code): | KEYDEF_REP = regex_partial(r""" | ||||||
| 	macro_id = code[2:-1] |     #define | ||||||
| 	if macro_id not in config['macro_ids']: |     \s | ||||||
| 		config['macro_ids'].append(macro_id) |     ( | ||||||
| 	return code |         (?:{})          # the prefixes | ||||||
| 
 |         (?:\w+)         # the key name | ||||||
| # TODO: presumably we can have a macro or function which takes |     )                   # capture group end | ||||||
| #		the hex code and produces much smaller code. |     """ | ||||||
| 
 | ) | ||||||
| WIN_UNICODE_MACRO_TEMPLATE = """ |  | ||||||
| case {0}: |  | ||||||
| 	return MACRODOWN( |  | ||||||
| 		D(LALT), T(KP_PLUS), {1}, U(LALT), END |  | ||||||
| 	); |  | ||||||
| """ |  | ||||||
| 
 |  | ||||||
| LINUX_UNICODE_MACRO_TEMPLATE = """ |  | ||||||
| case {0}: |  | ||||||
| 	return MACRODOWN( |  | ||||||
| 		D(LCTRL), D(LSHIFT), T(U), U(LCTRL), U(LSHIFT), {1}, T(KP_ENTER), END |  | ||||||
| 	); |  | ||||||
| """ |  | ||||||
| 
 |  | ||||||
| def macro_cases(config, mode): |  | ||||||
| 	if mode == 'win': |  | ||||||
| 		template = WIN_UNICODE_MACRO_TEMPLATE |  | ||||||
| 	elif mode == 'linux': |  | ||||||
| 		template = LINUX_UNICODE_MACRO_TEMPLATE |  | ||||||
| 	else: |  | ||||||
| 		raise ValueError("Invalid mode: ", mode) |  | ||||||
| 	template = template.strip() |  | ||||||
| 
 |  | ||||||
| 	for macro_id, unimacro_chars in config['unicode_macros']: |  | ||||||
| 		unimacro_keys = ", ".join( |  | ||||||
| 			"T({})".format( |  | ||||||
| 				"KP_" + char if char.isdigit() else char |  | ||||||
| 			) for char in unimacro_chars |  | ||||||
| 		) |  | ||||||
| 		yield template.format(macro_id, unimacro_keys) |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| MACROCODE = """ | ENUM_RE = re.compile(r""" | ||||||
| #define UC_MODE_WIN 0 |     ( | ||||||
| #define UC_MODE_LINUX 1 |         enum | ||||||
| 
 |         \s\w+\s | ||||||
| static uint16_t unicode_mode = UC_MODE_WIN; |         \{ | ||||||
| 
 |         .*?             # the enum content | ||||||
| const macro_t *action_get_macro(keyrecord_t *record, uint8_t id, uint8_t opt) {{ |         \} | ||||||
| 	if (!record->event.pressed) {{ |         ; | ||||||
| 		return MACRO_NONE; |     )                   # capture group end | ||||||
| 	}} |     """, re.MULTILINE | re.DOTALL | re.VERBOSE | ||||||
|   	// MACRODOWN only works in this function | ) | ||||||
|     switch(id) {{ |  | ||||||
|     	case UMS: |  | ||||||
|     		unicode_mode = (unicode_mode + 1) % 2; |  | ||||||
|     		break; |  | ||||||
|     	{macro_cases} |  | ||||||
|     	default: |  | ||||||
|     		break; |  | ||||||
|     }} |  | ||||||
| 	if (unicode_mode == UC_MODE_WIN) {{ |  | ||||||
| 		switch(id) {{ |  | ||||||
| 			{win_macro_cases} |  | ||||||
| 	    	default: |  | ||||||
| 	    		break; |  | ||||||
| 		}} |  | ||||||
| 	}} else if (unicode_mode == UC_MODE_LINUX) {{ |  | ||||||
| 		switch(id) {{ |  | ||||||
| 			{linux_macro_cases} |  | ||||||
| 	    	default: |  | ||||||
| 	    		break; |  | ||||||
| 		}} |  | ||||||
| 	}} |  | ||||||
|     return MACRO_NONE; |  | ||||||
| }}; |  | ||||||
| """ |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def iter_keycodes(layer_lines, config): | ENUM_KEY_REP = regex_partial(r""" | ||||||
| 	filler_re = re.compile("[" + |     ( | ||||||
| 		config['filler'] + " " + |         {}              # the prefixes | ||||||
| 	"]") |         \w+             # the key name | ||||||
|  |     )                   # capture group end | ||||||
|  |     """ | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| 	all_codes = [] | def parse_keydefs(config, data): | ||||||
| 	for line in layer_lines: |     prefix_options = "|".join(config['key_prefixes']) | ||||||
| 		line, _ = filler_re.subn("", line.strip()) |     keydef_re = KEYDEF_REP(prefix_options) | ||||||
| 		if not line: |     enum_key_re = ENUM_KEY_REP(prefix_options) | ||||||
| 			continue |     for match in keydef_re.finditer(data): | ||||||
| 		codes = line.split(config['separator']) |         yield match.groups()[0] | ||||||
| 		all_codes.extend(codes[1:-1]) |  | ||||||
| 
 | 
 | ||||||
| 	key_groups = {} |     for enum_match in ENUM_RE.finditer(data): | ||||||
| 	for group_index, key_indexes in enumerate(config['keymap_indexes']): |         enum = enum_match.groups()[0] | ||||||
| 		for key_index in key_indexes: |         for key_match in enum_key_re.finditer(enum): | ||||||
| 			key_groups[key_index] = group_index |             yield key_match.groups()[0] | ||||||
| 
 | 
 | ||||||
| 	keymap_indexes = sum(config['keymap_indexes'], []) |  | ||||||
| 	assert len(all_codes) == len(keymap_indexes) |  | ||||||
| 	code_index_pairs = zip(all_codes, keymap_indexes) |  | ||||||
| 	prev_index = None |  | ||||||
| 	for i, (code, key_index) in enumerate(code_index_pairs): |  | ||||||
| 		code = code.strip() |  | ||||||
| 		layer_match = LAYER_CHANGE_RE.match(code) |  | ||||||
| 		unicode_match = UNICODE_RE.match(code) |  | ||||||
| 		noncode_match = NON_CODE.match(code) |  | ||||||
| 		macro_match = MACRO_RE.match(code) |  | ||||||
| 
 | 
 | ||||||
| 		ws = "\n" if key_groups[key_index] != prev_index else "" | def parse_valid_keys(config): | ||||||
| 		prev_index = key_groups[key_index] |     valid_keycodes = set() | ||||||
|  |     paths = [ | ||||||
|  |         os.path.join(BASEPATH, "tmk_core", "common", "keycode.h") | ||||||
|  |     ] + [ | ||||||
|  |         os.path.join( | ||||||
|  |             BASEPATH, config['includes_dir'], include_path | ||||||
|  |         ) for include_path in config['keymaps_includes'] | ||||||
|  |     ] | ||||||
| 
 | 
 | ||||||
| 		try: |     for path in paths: | ||||||
| 			if not code: |         path = path.replace("/", os.sep) | ||||||
| 				code = 'KC_TRNS' |         # the config always uses forward slashe | ||||||
| 			elif layer_match: |         if os.path.exists(path): | ||||||
| 				pass |             header_data = read_header_file(path) | ||||||
| 			elif macro_match: |             valid_keycodes.update( | ||||||
| 				code = MACRO(config, code) |                 parse_keydefs(config, header_data) | ||||||
| 			elif unicode_match: |             ) | ||||||
| 				hex_code = code[1:] |     return valid_keycodes | ||||||
| 				code = UNICODE_MACRO(config, chr(int(hex_code, 16))) | 
 | ||||||
| 			elif noncode_match: | #       Keymap Parsing | ||||||
| 				code = UNICODE_MACRO(config, code) | 
 | ||||||
| 			elif "_" in code: | def iter_raw_codes(layer_lines, filler, separator): | ||||||
| 				assert code in valid_keycodes, "unknown code '{}'".format(code) |     filler_re = re.compile("[" + filler + " ]") | ||||||
| 			else: |     for line in layer_lines: | ||||||
| 				for prefix in config['key_prefixes']: |         line, _ = filler_re.subn("", line.strip()) | ||||||
| 					if prefix + code in valid_keycodes: |         if not line: | ||||||
| 						code = prefix + code |             continue | ||||||
| 						break |         codes = line.split(separator) | ||||||
| 				assert code in valid_keycodes, "unknown code '{}'".format(code) |         for code in codes[1:-1]: | ||||||
| 			yield code, key_index, ws |             yield code | ||||||
| 		except AssertionError: | 
 | ||||||
| 			print("Error processing code", repr(code).encode("utf-8")) | 
 | ||||||
| 			raise | def iter_indexed_codes(raw_codes, key_indexes): | ||||||
|  |     key_rows = {} | ||||||
|  |     key_indexes_flat = [] | ||||||
|  |     for row_index, key_indexes in enumerate(key_indexes): | ||||||
|  |         for key_index in key_indexes: | ||||||
|  |             key_rows[key_index] = row_index | ||||||
|  |         key_indexes_flat.extend(key_indexes) | ||||||
|  |     assert len(raw_codes) == len(key_indexes_flat) | ||||||
|  |     for raw_code, key_index in zip(raw_codes, key_indexes_flat): | ||||||
|  |         # we keep track of the row mostly for layout purposes | ||||||
|  |         yield raw_code, key_index, key_rows[key_index] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | LAYER_CHANGE_RE = re.compile(r""" | ||||||
|  |     (DF|TG|MO)\(\d+\) | ||||||
|  | """, re.VERBOSE) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | MACRO_RE = re.compile(r""" | ||||||
|  |     M\(\w+\) | ||||||
|  | """, re.VERBOSE) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | UNICODE_RE = re.compile(r""" | ||||||
|  |     U[0-9A-F]{4} | ||||||
|  | """, re.VERBOSE) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | NON_CODE = re.compile(r""" | ||||||
|  |     ^[^A-Z0-9_]$ | ||||||
|  | """, re.VERBOSE) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def parse_uni_code(raw_code): | ||||||
|  |     macro_id = "UC_" + ( | ||||||
|  |         unicodedata.name(raw_code) | ||||||
|  |         .replace(" ", "_") | ||||||
|  |         .replace("-", "_") | ||||||
|  |     ) | ||||||
|  |     code = "M({})".format(macro_id) | ||||||
|  |     uc_hex = "{:04X}".format(ord(raw_code)) | ||||||
|  |     return code, macro_id, uc_hex | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def parse_key_code(raw_code, key_prefixes, valid_keycodes): | ||||||
|  |     if raw_code in valid_keycodes: | ||||||
|  |         return raw_code | ||||||
|  | 
 | ||||||
|  |     for prefix in key_prefixes: | ||||||
|  |         code = prefix + raw_code | ||||||
|  |         if code in valid_keycodes: | ||||||
|  |             return code | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def parse_code(raw_code, key_prefixes, valid_keycodes): | ||||||
|  |     if not raw_code: | ||||||
|  |         return 'KC_TRNS', None, None | ||||||
|  | 
 | ||||||
|  |     if LAYER_CHANGE_RE.match(raw_code): | ||||||
|  |         return raw_code, None, None | ||||||
|  | 
 | ||||||
|  |     if MACRO_RE.match(raw_code): | ||||||
|  |         code = macro_id = raw_code[2:-1] | ||||||
|  |         return code, macro_id, None | ||||||
|  | 
 | ||||||
|  |     if UNICODE_RE.match(raw_code): | ||||||
|  |         hex_code = raw_code[1:] | ||||||
|  |         return parse_uni_code(chr(int(hex_code, 16))) | ||||||
|  | 
 | ||||||
|  |     if NON_CODE.match(raw_code): | ||||||
|  |         return parse_uni_code(raw_code) | ||||||
|  | 
 | ||||||
|  |     code = parse_key_code(raw_code, key_prefixes, valid_keycodes) | ||||||
|  |     return code, None, None | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def parse_keymap(config, key_indexes, layer_lines, valid_keycodes): | ||||||
|  |     keymap = {} | ||||||
|  |     raw_codes = list(iter_raw_codes( | ||||||
|  |         layer_lines, config['filler'], config['separator'] | ||||||
|  |     )) | ||||||
|  |     indexed_codes = iter_indexed_codes(raw_codes, key_indexes) | ||||||
|  |     for raw_code, key_index, row_index in indexed_codes: | ||||||
|  |         code, macro_id, uc_hex = parse_code( | ||||||
|  |             raw_code, config['key_prefixes'], valid_keycodes | ||||||
|  |         ) | ||||||
|  |         if macro_id: | ||||||
|  |             config['macro_ids'].add(macro_id) | ||||||
|  |         if uc_hex: | ||||||
|  |             config['unicode_macros'][macro_id] = uc_hex | ||||||
|  |         keymap[key_index] = (code, row_index) | ||||||
|  |     return keymap | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def parse_keymaps(config, valid_keycodes): | ||||||
|  |     keymaps = collections.OrderedDict() | ||||||
|  |     key_indexes = config.get( | ||||||
|  |         'key_indexes', KEYBOARD_LAYOUTS[config['layout']] | ||||||
|  |     ) | ||||||
|  |     # TODO: maybe validate key_indexes | ||||||
|  | 
 | ||||||
|  |     for layer_name, layer_lines, in config['layer_lines'].items(): | ||||||
|  |         keymaps[layer_name] = parse_keymap( | ||||||
|  |             config, key_indexes, layer_lines, valid_keycodes | ||||||
|  |         ) | ||||||
|  |     return keymaps | ||||||
|  | 
 | ||||||
|  | #       keymap.c output | ||||||
| 
 | 
 | ||||||
| USERCODE = """ | USERCODE = """ | ||||||
| // Runs just one time when the keyboard initializes. | // Runs just one time when the keyboard initializes. | ||||||
| void * matrix_init_user(void) { | void matrix_init_user(void) { | ||||||
| 
 | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // Runs constantly in the background, in a loop. | // Runs constantly in the background, in a loop. | ||||||
| void * matrix_scan_user(void) { | void matrix_scan_user(void) { | ||||||
|     uint8_t layer = biton32(layer_state); |     uint8_t layer = biton32(layer_state); | ||||||
| 
 | 
 | ||||||
|     ergodox_board_led_off(); |     ergodox_board_led_off(); | ||||||
| @ -348,71 +481,158 @@ void * matrix_scan_user(void) { | |||||||
| }; | }; | ||||||
| """ | """ | ||||||
| 
 | 
 | ||||||
| def parse_keymaps(config): | MACROCODE = """ | ||||||
| 	keymaps = {} | #define UC_MODE_WIN 0 | ||||||
| 	layer_line_items = config['layer_lines'].items() | #define UC_MODE_LINUX 1 | ||||||
| 	for i, (layer_name, layer_lines) in enumerate(layer_line_items): | 
 | ||||||
| 		print("parseing layer", layer_name) | static uint16_t unicode_mode = UC_MODE_WIN; | ||||||
| 		keymap = {} | 
 | ||||||
| 		for code, key_index, ws in iter_keycodes(layer_lines, config): | const macro_t *action_get_macro(keyrecord_t *record, uint8_t id, uint8_t opt) {{ | ||||||
| 			keymap[key_index] = (code, ws) |     if (!record->event.pressed) {{ | ||||||
| 		keymaps[layer_name] = [v for k, v in sorted(keymap.items())] |         return MACRO_NONE; | ||||||
| 	return keymaps |     }} | ||||||
|  |     // MACRODOWN only works in this function | ||||||
|  |     switch(id) {{ | ||||||
|  |         case UM: | ||||||
|  |             unicode_mode = (unicode_mode + 1) % 2; | ||||||
|  |             break; | ||||||
|  |         {macro_cases} | ||||||
|  |         default: | ||||||
|  |             break; | ||||||
|  |     }} | ||||||
|  |     if (unicode_mode == UC_MODE_WIN) {{ | ||||||
|  |         switch(id) {{ | ||||||
|  |             {win_macro_cases} | ||||||
|  |             default: | ||||||
|  |                 break; | ||||||
|  |         }} | ||||||
|  |     }} else if (unicode_mode == UC_MODE_LINUX) {{ | ||||||
|  |         switch(id) {{ | ||||||
|  |             {linux_macro_cases} | ||||||
|  |             default: | ||||||
|  |                 break; | ||||||
|  |         }} | ||||||
|  |     }} | ||||||
|  |     return MACRO_NONE; | ||||||
|  | }}; | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | WIN_UNICODE_MACRO_TEMPLATE = """ | ||||||
|  | case {0}: | ||||||
|  |     return MACRODOWN( | ||||||
|  |         D(LALT), T(KP_PLUS), {1}, U(LALT), END | ||||||
|  |     ); | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | LINUX_UNICODE_MACRO_TEMPLATE = """ | ||||||
|  | case {0}: | ||||||
|  |     return MACRODOWN( | ||||||
|  |         D(LCTRL), D(LSHIFT), T(U), U(LCTRL), U(LSHIFT), {1}, T(KP_ENTER), END | ||||||
|  |     ); | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | def macro_cases(config, mode): | ||||||
|  |     if mode == 'win': | ||||||
|  |         template = WIN_UNICODE_MACRO_TEMPLATE | ||||||
|  |     elif mode == 'linux': | ||||||
|  |         template = LINUX_UNICODE_MACRO_TEMPLATE | ||||||
|  |     else: | ||||||
|  |         raise ValueError("Invalid mode: ", mode) | ||||||
|  |     template = template.strip() | ||||||
|  | 
 | ||||||
|  |     for macro_id, uc_hex in config['unicode_macros'].items(): | ||||||
|  |         unimacro_keys = ", ".join( | ||||||
|  |             "T({})".format( | ||||||
|  |                 "KP_" + digit if digit.isdigit() else digit | ||||||
|  |             ) for digit in uc_hex | ||||||
|  |         ) | ||||||
|  |         yield template.format(macro_id, unimacro_keys) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def iter_keymap_lines(config, keymaps): | def iter_keymap_lines(keymap): | ||||||
| 	for include_path in config['keymaps_includes']: |     prev_row_index = None | ||||||
| 		yield '#include "{}"\n'.format(include_path) |     for key_index in sorted(keymap): | ||||||
| 
 |         code, row_index = keymap[key_index] | ||||||
| 	yield "\n" |         if row_index != prev_row_index: | ||||||
| 
 |             yield "\n" | ||||||
| 	layer_items = config['layers'].items() |         yield " {}".format(code) | ||||||
| 	for i, (layer_name, description) in enumerate(layer_items): |         if key_index < len(keymap) - 1: | ||||||
| 		yield '#define L{0:<3} {0:<5}  // {1}\n'.format(i, layer_name) |             yield "," | ||||||
| 
 |         prev_row_index = row_index | ||||||
| 	for i, macro_id in enumerate(config['macro_ids']): |  | ||||||
| 		yield "#define {} {}\n".format(macro_id, i) |  | ||||||
| 
 |  | ||||||
| 	yield "\n" |  | ||||||
| 
 |  | ||||||
| 	yield "const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {\n" |  | ||||||
| 
 |  | ||||||
| 	layer_line_items = config['layer_lines'].items() |  | ||||||
| 	last_index = config['keymap_indexes'][-1] |  | ||||||
| 	for i, (layer_name, layer_lines) in enumerate(layer_line_items): |  | ||||||
| 		keymap = keymaps[layer_name] |  | ||||||
| 		yield "/*\n" |  | ||||||
| 		for line in layer_lines: |  | ||||||
| 			yield " *{}".format(line) |  | ||||||
| 		yield "*/\n" |  | ||||||
| 
 |  | ||||||
| 		yield "[L{0}] = KEYMAP(\n".format(i) |  | ||||||
| 
 |  | ||||||
| 		for key_index, (code, ws) in enumerate(keymap): |  | ||||||
| 			yield "\t{}".format(code) |  | ||||||
| 			if key_index < len(keymap) - 1: |  | ||||||
| 				yield "," |  | ||||||
| 			yield ws |  | ||||||
| 		yield "),\n" |  | ||||||
| 
 |  | ||||||
| 	yield "};\n\n" |  | ||||||
| 
 |  | ||||||
| 	yield "const uint16_t PROGMEM fn_actions[] = {\n" |  | ||||||
| 	yield "};\n" |  | ||||||
| 
 |  | ||||||
| 	yield MACROCODE.format( |  | ||||||
| 		macro_cases="", |  | ||||||
| 		win_macro_cases="\n".join(macro_cases(config, mode='win')), |  | ||||||
| 		linux_macro_cases="\n".join(macro_cases(config, mode='linux')), |  | ||||||
| 	) |  | ||||||
| 
 |  | ||||||
| 	yield USERCODE |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| with io.open("keymap.c", mode="w", encoding="utf-8") as fh: | def iter_keymap_parts(config, keymaps): | ||||||
| 	for data in iter_keymap_lines(config, parse_keymaps(config)): |     # includes | ||||||
| 		fh.write(data) |     for include_path in config['keymaps_includes']: | ||||||
|  |         yield '#include "{}"\n'.format(include_path) | ||||||
|  | 
 | ||||||
|  |     yield "\n" | ||||||
|  | 
 | ||||||
|  |     # definitions | ||||||
|  |     for i, macro_id in enumerate(sorted(config['macro_ids'])): | ||||||
|  |         yield "#define {} {}\n".format(macro_id, i) | ||||||
|  | 
 | ||||||
|  |     yield "\n" | ||||||
|  | 
 | ||||||
|  |     for i, layer_name in enumerate(config['layer_lines']): | ||||||
|  |         yield '#define L{0:<3} {0:<5}  // {1}\n'.format(i, layer_name) | ||||||
|  | 
 | ||||||
|  |     yield "\n" | ||||||
|  | 
 | ||||||
|  |     # keymaps | ||||||
|  |     yield "const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = {\n" | ||||||
|  | 
 | ||||||
|  |     for i, layer_name in enumerate(config['layer_lines']): | ||||||
|  |         # comment | ||||||
|  |         layer_lines = config['layer_lines'][layer_name] | ||||||
|  |         prefixed_lines = " * " + " * ".join(layer_lines) | ||||||
|  |         yield "/*\n{}*/\n".format(prefixed_lines) | ||||||
|  | 
 | ||||||
|  |         # keymap codes | ||||||
|  |         keymap = keymaps[layer_name] | ||||||
|  |         keymap_lines = "".join(iter_keymap_lines(keymap)) | ||||||
|  |         yield "[L{0}] = KEYMAP({1}\n),\n".format(i, keymap_lines) | ||||||
|  | 
 | ||||||
|  |     yield "};\n\n" | ||||||
|  | 
 | ||||||
|  |     # no idea what this is for | ||||||
|  |     yield "const uint16_t PROGMEM fn_actions[] = {};\n" | ||||||
|  | 
 | ||||||
|  |     # macros | ||||||
|  |     yield MACROCODE.format( | ||||||
|  |         macro_cases="", | ||||||
|  |         win_macro_cases="\n".join(macro_cases(config, mode='win')), | ||||||
|  |         linux_macro_cases="\n".join(macro_cases(config, mode='linux')), | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     # TODO: dynamically create blinking lights | ||||||
|  |     yield USERCODE | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| # print("\n".join(sorted(valid_keycodes))) | def main(argv=sys.argv[1:]): | ||||||
| # print(json.dumps(config, indent=4)) |     if not argv or '-h' in argv or '--help' in argv: | ||||||
|  |         print(__doc__) | ||||||
|  |         return 0 | ||||||
|  | 
 | ||||||
|  |     in_path = os.path.abspath(argv[0]) | ||||||
|  |     if not os.path.exists(in_path): | ||||||
|  |         print("No such file '{}'".format(in_path)) | ||||||
|  |         return 1 | ||||||
|  | 
 | ||||||
|  |     if len(argv) > 1: | ||||||
|  |         out_path = os.path.abspath(argv[1]) | ||||||
|  |     else: | ||||||
|  |         dirname = os.path.dirname(in_path) | ||||||
|  |         out_path = os.path.join(dirname, "keymap.c") | ||||||
|  | 
 | ||||||
|  |     config = parse_config(in_path) | ||||||
|  |     valid_keys = parse_valid_keys(config) | ||||||
|  |     keymaps = parse_keymaps(config, valid_keys) | ||||||
|  | 
 | ||||||
|  |     with io.open(out_path, mode="w", encoding="utf-8") as fh: | ||||||
|  |         for part in iter_keymap_parts(config, keymaps): | ||||||
|  |             fh.write(part) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     sys.exit(main()) | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -1,8 +1,14 @@ | |||||||
| # ManuNeo Ergodox Keyboard Layout | # ManuNeo Ergodox Keyboard Layout | ||||||
| 
 | 
 | ||||||
|  | Compile this file to a `keymap.c` file using `compile_keymap.py` | ||||||
|  | 
 | ||||||
|  |     compile_keymap.py keymaps/german-manuneo/keymap.md | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| # Layout Config | # Layout Config | ||||||
| 
 | 
 | ||||||
|     { |     { | ||||||
|  |         "layout": "ergodox_ez", | ||||||
|         "includes_dir": "quantum/", |         "includes_dir": "quantum/", | ||||||
|         "keymaps_includes": [ |         "keymaps_includes": [ | ||||||
|             "ergodox_ez.h", |             "ergodox_ez.h", | ||||||
| @ -10,22 +16,14 @@ | |||||||
|             "keymap_common.h", |             "keymap_common.h", | ||||||
|             "keymap_extras/keymap_german.h", |             "keymap_extras/keymap_german.h", | ||||||
|         ], |         ], | ||||||
|         "keymap_indexes": [ |  | ||||||
|             [ 0,  1,  2,  3,  4,  5,  6],  [38, 39, 40, 41, 42, 43, 44], |  | ||||||
|             [ 7,  8,  9, 10, 11, 12, 13],  [45, 46, 47, 48, 49, 50, 51], |  | ||||||
|             [14, 15, 16, 17, 18, 19    ],  [    52, 53, 54, 55, 56, 57], |  | ||||||
|             [20, 21, 22, 23, 24, 25, 26],  [58, 59, 60, 61, 62, 63, 64], |  | ||||||
|             [27, 28, 29, 30, 31        ],  [        65, 66, 67, 68, 69], |  | ||||||
|             [                    32, 33],  [70, 71                    ], |  | ||||||
|             [                        34],  [72                        ], |  | ||||||
|             [                35, 36, 37],  [73, 74, 75                ], |  | ||||||
|         ], |  | ||||||
|         "key_prefixes": ["DE_", "KC_"], |         "key_prefixes": ["DE_", "KC_"], | ||||||
|         "filler": "-+.'!x", |         "filler": "-+.'!x", | ||||||
|         "separator": "|", |         "separator": "|", | ||||||
|         "macros": { |         "macros": { | ||||||
|             "MUC": "", |             // TODO: implement macros | ||||||
|  |             // "MUC": "", | ||||||
|         }, |         }, | ||||||
|  |         // TODO: implement default unicode mode | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -34,189 +32,201 @@ | |||||||
| 
 | 
 | ||||||
| ## Layer 0 | ## Layer 0 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |MO(5)| 1  | 2  | 3  | 4  | 5  | ACUT |       ! GRV  | 6  | 7  | 8  | 9  | 0  |CIRC | |     |MO(5)| 1  | 2  | 3  | 4  | 5  |ACUT | GRV | 6  | 7  | 8  | 9  | 0  |CIRC | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |MO(4)| X  | P  | F  | W  | G  | HOME |       !TG(2) | H  | J  | K  | L  | Q  |  Z  | |     |MO(4)| X  | P  | F  | W  | G  |HOME |TG(2)| H  | J  | K  | L  | Q  |  Z  | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |MO(1)| U  | I  | A  | E  | O  |------!       !------! S  | N  | R  | T  | D  | SS  | |     |MO(1)| U  | I  | A  | E  | O  |-----!-----! S  | N  | R  | T  | D  | SS  | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |MO(3)| UE | OE | AE | C  | V  | END  |       ! TAB  | B  | M  |COMM| DOT| UP |  Y  | |     |MO(3)| UE | OE | AE | C  | V  |END  | TAB | B  | M  |COMM| DOT| UP |  Y  | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |LGUI|LALT|LCTL|                               !LALT|    |LEFT|DOWN|RGHT| |      |    |    |LGUI|LALT|LCTL|                     !LALT|    |LEFT|DOWN|RGHT| | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                | INS  |TG(2) | !M(UMS)| DELT | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |INS  |TG(2)| !M(UM)|DELT | | ||||||
|                         !      !      | APP  | ! PGUP |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     | APP | ! PGUP|     !     ! | ||||||
|                         | BSPC | LSFT | ESC  | ! PGDN |ENTER |SPACE | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |BSPC |LSFT | ESC | ! PGDN|ENTER|SPACE| | ||||||
|  |                       '-----------------' '-----------------' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 1 | ## Layer 1 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |     |EXLM|DQOT|PARA|    |    |      |       !      |    |    |    |    |RING|     | |     |     |EXLM|DQOT|PARA|    |    |     |     |    |    |    |    |RING|     | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |     |ASTR|PIPE|SLSH|LCBR|RCBR|      |       !      |HASH|LESS|MORE|    |DQOT|     | |     |     |ASTR|PIPE|SLSH|LCBR|RCBR|     |     |HASH|LESS|MORE|    |DQOT|     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |UNDS|MINS|AMPR|LBRC|RBRC|------!       !------!DLR |LPRN|RPRN|TILD|QUOT|QST  | |     |     |UNDS|MINS|AMPR|LBRC|RBRC|-----!-----!DLR |LPRN|RPRN|TILD|QUOT| QST | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |PLUS| EQL|    |    |      |       !      |BSLS|PERC|SCLN|COLN| ↑  |     | |     |     |    |PLUS|EQL |    |    |     |     |BSLS|PERC|SCLN|COLN| ↑  |     | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |    |    |    |                               !    |    | ←  | ↓  | →  | |      |    |    |    |    |    |                     !    |    | ←  | ↓  | →  | | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                |      |      | !      |      | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |     |     | !     |     | | ||||||
|                         !      !      |      | !      |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     |     | !     |     !     ! | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 2 | ## Layer 2 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |     | F1 | F2 | F3 | F4 | F5 | F11  |       ! F12  | F6 | F7 | F8 | F9 |F10 |PEQL | |     |     | F1 | F2 | F3 | F4 | F5 | F11 | F12 | F6 | F7 | F8 | F9 |F10 |PEQL | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       !      |    | P7 | P8 | P9 |PAST|PSLS | |     |     |    |    |    |    |    |     |     |    | P7 | P8 | P9 |PAST|PSLS | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |------!       !------!    | P4 | P5 | P6 |PMNS|PMNS | |     |     |    |    |    |    |    |-----!-----!    | P4 | P5 | P6 |PMNS|PMNS | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       ! NLCK |    | P1 | P2 | P3 |PPLS|PPLS | |     |     |    |    |    |    |    |     | NLCK|    | P1 | P2 | P3 |PPLS|PPLS | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |    |    |    |                               ! P0 |PCMM|PDOT|PENT|PENT| |      |    |    |    |    |    |                     ! P0 |PCMM|PDOT|PENT|PENT| | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                |      |      | !      |      | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |     |     | !     |     | | ||||||
|                         !      !      |      | !      |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     |     | !     |     !     ! | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 3 | ## Layer 3 | ||||||
| 
 | 
 | ||||||
| http://symbolcodes.tlt.psu.edu/bylanguage/mathchart.html | http://symbolcodes.tlt.psu.edu/bylanguage/mathchart.html | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |     | ¹  | ²  | ³  | ⁴  | ⁵  |  ∀   |       !      | ⁶  | ⁷  | ⁸  | ⁹  |  ⁰ |     | |     |     | ¹  | ²  | ³  | ⁴  | ⁵  |  ∀  |     | ⁶  | ⁷  | ⁸  | ⁹  |  ⁰ |     | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |     | ×  | ½  | ÷  | ¼  |  ⅕ |      |       !      | ⅙  |    | ⅛  |    |    |     | |     |     | ×  | ½  | ÷  | ¼  |  ⅕ |     |     | ⅙  |    | ⅛  |    |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    | ±  | AT |EURO| ∅  |------!       !------! ∞  | ⁿ  | ∃  | ∈  |    |     | |     |     |    | ±  | AT |EURO| ∅  |-----!-----! ∞  | ⁿ  | ∃  | ∈  |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     | ⅓  | ≠  | ⅔  | ¾  | ≃  |      |       !      |EXLM|    | ∄  | ∉  |    |     | |     |     | ⅓  | ≠  | ⅔  | ¾  | ≃  |     |     |EXLM|    | ∄  | ∉  |    |     | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |    |    |    |                               !    |    |    |    |    | |      |    |    |    |    |    |                     !    |    |    |    |    | | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                |      |      | !      |      | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |     |     | !     |     | | ||||||
|                         !      !      |      | !      |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     |     | !     |     !     ! | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 4 | ## Layer 4 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. | 
 | ||||||
|     |     | ₁  | ₂  | ₃  | ₄  | ₅  |      |       !      | ₆  | ₇  | ₈  | ₉  | ₀  |     | |     .------------------------------------.------------------------------------. | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     |     | ₁  | ₂  | ₃  | ₄  | ₅  |     |     | ₆  | ₇  | ₈  | ₉  | ₀  |     | | ||||||
|     |     | χ  | π  | φ  | ω  | γ  |      |       !      | η  | ξ  | κ  | λ  |    |  ζ  | |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     |     | χ  | π  | φ  | ω  | γ  |     |     | η  | ξ  | κ  | λ  |    |  ζ  | | ||||||
|     |     | υ  | ι  | α  | ε  | ο  |------!       !------! σ  | ν  | ρ  | τ  | δ  |  ς  | |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     |     | υ  | ι  | α  | ε  | ο  |-----!-----! σ  | ν  | ρ  | τ  | δ  |  ς  | | ||||||
|     |     |    | θ  |    |    |    |      |       !      | β  | μ  |    |    |    |  ψ  | |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     |     |    | θ  |    |    |    |     |     | β  | μ  |    |    |    |  ψ  | | ||||||
|      |    |    |    |    |    |                               !    |    |    |    |    | |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      |    |    |    |    |    |                     !    |    |    |    |    | | ||||||
|                                |      |      | !      |      | |      '------------------------'                     '------------------------' | ||||||
|                         .------+------+------! !------+------+------. |                             .-----------. .-----------. | ||||||
|                         !      !      |      | !      |      !      ! |                             |     |     | !     |     | | ||||||
|                         !      !      !------! !------!      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     |     | !     |     !     ! | ||||||
|                         '--------------------' '--------------------' |                       !     !     !-----! !-----!     !     ! | ||||||
|  |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 5 | ## Layer 5 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |     | Χ  | Π  | Φ  | Ω  | Γ  |      |       !      | Η  | Ξ  | Κ  | Λ  |    |  Ζ  | |     |     | Χ  | Π  | Φ  | Ω  | Γ  |     |     | Η  | Ξ  | Κ  | Λ  |    |  Ζ  | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     | Υ  | Ι  | Α  | Ε  | Ο  |------!       !------! Σ  | Ν  | Ρ  | Τ  | Δ  |     | |     |     | Υ  | Ι  | Α  | Ε  | Ο  |-----!-----! Σ  | Ν  | Ρ  | Τ  | Δ  |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    | Θ  |    |    |    |      |       !      | Β  | Μ  |    |    |    |  Ψ  | |     |     |    | Θ  |    |    |    |     |     | Β  | Μ  |    |    |    |  Ψ  | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |    |    |    |                               !    |    |    |    |    | |      |    |    |    |    |    |                     !    |    |    |    |    | | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                |      |      | !      |      | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |     |     | !     |     | | ||||||
|                         !      !      |      | !      |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     |     | !     |     !     ! | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 6 | ## Layer 6 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |------!       !------!    |    |    |    |    |     | |     |     |    |    |    |    |    |-----!-----!    |    |    |    |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |    |    |    |                               !    |    |    |    |    | |      |    |    |    |    |    |                     !    |    |    |    |    | | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                |      |      | !      |      | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |     |     | !     |     | | ||||||
|                         !      !      |      | !      |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     |     | !     |     !     ! | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 7 | ## Layer 7 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |------!       !------!    |    |    |    |    |     | |     |     |    |    |    |    |    |-----!-----!    |    |    |    |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |    |    |    |                               !    |    |    |    |    | |      |    |    |    |    |    |                     !    |    |    |    |    | | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                |      |      | !      |      | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |     |     | !     |     | | ||||||
|                         !      !      |      | !      |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     |     | !     |     !     ! | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Layer 8 | ## Layer 8 | ||||||
| 
 | 
 | ||||||
|     .-------------------------------------.       .-------------------------------------. |     .------------------------------------.------------------------------------. | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----! |     !-----+----+----+----+----+----------!-----+----+----+----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |------!       !------!    |    |    |    |    |     | |     |     |    |    |    |    |    |-----!-----!    |    |    |    |    |     | | ||||||
|     !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----! |     !-----+----+----+----x----x----!     !     !----x----x----+----+----+-----! | ||||||
|     |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     | |     |     |    |    |    |    |    |     |     |    |    |    |    |    |     | | ||||||
|     '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----' |     '-----+----+----+----+----+----------'----------+----+----+----+----+-----' | ||||||
|      |    |    |    |    |    |                               !    |    |    |    |    | |      |    |    |    |    |    |                     !    |    |    |    |    | | ||||||
|      '------------------------'.-------------. .-------------.'------------------------' |      '------------------------'                     '------------------------' | ||||||
|                                |      |      | !      |      | |                             .-----------. .-----------. | ||||||
|                         .------+------+------! !------+------+------. |                             |     |     | !     |     | | ||||||
|                         !      !      |      | !      |      !      ! |                       .-----+-----+-----! !-----+-----+-----. | ||||||
|                         !      !      !------! !------!      !      ! |                       !     !     |     | !     |     !     ! | ||||||
|                         |      |      |      | !      |      |      | |                       !     !     !-----! !-----!     !     ! | ||||||
|                         '--------------------' '--------------------' |                       |     |     |     | !     |     |     | | ||||||
|  |                       '-----------------' '-----------------' | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Manuel Barkhau
						Manuel Barkhau