Manuel Barkhau 9 роки тому
батько
коміт
20a3229faf

+ 418 - 0
keyboard/ergodox_ez/keymaps/german-manuneo/compile_keymap.py

@@ -0,0 +1,418 @@
+# encoding: utf-8
+from __future__ import division
+from __future__ import print_function
+from __future__ import absolute_import
+from __future__ import unicode_literals
+
+import os
+import io
+import re
+import sys
+import json
+import unicodedata
+import collections
+
+PY2 = sys.version_info.major == 2
+
+if PY2:
+	chr = unichr
+
+
+ONELINE_COMMENT_RE = re.compile(r"^\s*//.*$", re.MULTILINE)
+INLINE_COMMENT_RE = re.compile(
+    r"([\,\"\[\]\{\}\d])\s+//\s[^\"\]\}\{\[]*$", re.MULTILINE
+)
+TRAILING_COMMA_RE = re.compile(
+    r",$\s*([\]\}])", re.MULTILINE
+)
+
+def loads(raw_data):
+    if isinstance(raw_data, bytes):
+        raw_data = raw_data.decode('utf-8')
+    raw_data = ONELINE_COMMENT_RE.sub(r"", raw_data)
+    raw_data = INLINE_COMMENT_RE.sub(r"\1", raw_data)
+    raw_data = TRAILING_COMMA_RE.sub(r"\1", raw_data)
+    return json.loads(raw_data)
+
+with io.open("keymap.md", encoding="utf-8") as fh:
+	lines = fh.readlines()
+
+SECTIONS = [
+	'layout_config',
+	'layers',
+]
+
+config = {
+	"includes_basedir": "quantum/",
+    "keymaps_includes": [
+        "keymap_common.h",
+    ],
+	'filler': "-+.':x",
+	'separator': "|",
+    'default_key_prefix': ["KC_"],
+    'unicode_macros': [],
+    'macro_ids': ['UMS'],
+    'layers': collections.OrderedDict(),
+	'layer_lines': collections.OrderedDict(),
+}
+
+section_start_index = -1
+current_section = None
+current_layer_name = None
+current_layer_lines = []
+config_data = []
+
+def end_section():
+	global section_start_index
+	global current_layer_lines
+	section_start_index = -1
+	if current_section == 'layout_config':
+		config.update(loads("".join(
+			config_data
+		)))
+	elif current_section == 'layers':
+		config['layer_lines'][current_layer_name] = current_layer_lines
+		current_layer_lines = []
+
+
+for i, line in enumerate(lines):
+	if line.startswith("# "):
+		section = line[2:].strip().replace(" ", "_").lower()
+		if section in SECTIONS:
+			current_section = section
+	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()
+
+KEYDEF_RE = re.compile(r"#define ((?:{})(?:\w+))".format(
+	"|".join(config['key_prefixes'])
+))
+IF0_RE = re.compile(r"^#if 0$.*?#endif", re.MULTILINE | re.DOTALL)
+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):
+	# TODO: don't use macro for codepoints below 0x2000
+	macro_id = "UC_" + (
+		unicodedata.name(c)
+		.replace(" ", "_")
+		.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):
+	macro_id = code[2:-1]
+	if macro_id not in config['macro_ids']:
+		config['macro_ids'].append(macro_id)
+	return code
+
+# TODO: presumably we can have a macro or function which takes
+#		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 = """
+#define UC_MODE_WIN 0
+#define UC_MODE_LINUX 1
+
+static uint16_t unicode_mode = UC_MODE_WIN;
+
+const macro_t *action_get_macro(keyrecord_t *record, uint8_t id, uint8_t opt) {{
+	if (!record->event.pressed) {{
+		return MACRO_NONE;
+	}}
+  	// 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):
+	filler_re = re.compile("[" +
+		config['filler'] + " " +
+	"]")
+
+	all_codes = []
+	for line in layer_lines:
+		line, _ = filler_re.subn("", line.strip())
+		if not line:
+			continue
+		codes = line.split(config['separator'])
+		all_codes.extend(codes[1:-1])
+
+	key_groups = {}
+	for group_index, key_indexes in enumerate(config['keymap_indexes']):
+		for key_index in key_indexes:
+			key_groups[key_index] = group_index
+
+	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 ""
+		prev_index = key_groups[key_index]
+
+		try:
+			if not code:
+				code = 'KC_TRNS'
+			elif layer_match:
+				pass
+			elif macro_match:
+				code = MACRO(config, code)
+			elif unicode_match:
+				hex_code = code[1:]
+				code = UNICODE_MACRO(config, chr(int(hex_code, 16)))
+			elif noncode_match:
+				code = UNICODE_MACRO(config, code)
+			elif "_" in code:
+				assert code in valid_keycodes, "unknown code '{}'".format(code)
+			else:
+				for prefix in config['key_prefixes']:
+					if prefix + code in valid_keycodes:
+						code = prefix + code
+						break
+				assert code in valid_keycodes, "unknown code '{}'".format(code)
+			yield code, key_index, ws
+		except AssertionError:
+			print("Error processing code", repr(code).encode("utf-8"))
+			raise
+
+USERCODE = """
+// Runs just one time when the keyboard initializes.
+void * matrix_init_user(void) {
+
+};
+
+// Runs constantly in the background, in a loop.
+void * matrix_scan_user(void) {
+    uint8_t layer = biton32(layer_state);
+
+    ergodox_board_led_off();
+    ergodox_right_led_1_off();
+    ergodox_right_led_2_off();
+    ergodox_right_led_3_off();
+    switch (layer) {
+        case L1:
+            ergodox_right_led_1_on();
+            break;
+        case L2:
+            ergodox_right_led_2_on();
+            break;
+        case L3:
+            ergodox_right_led_3_on();
+            break;
+        case L4:
+            ergodox_right_led_1_on();
+            ergodox_right_led_2_on();
+            break;
+        case L5:
+            ergodox_right_led_1_on();
+            ergodox_right_led_3_on();
+            break;
+        // case L6:
+        //     ergodox_right_led_2_on();
+        //     ergodox_right_led_3_on();
+        //     break;
+        // case L7:
+        //     ergodox_right_led_1_on();
+        //     ergodox_right_led_2_on();
+        //     ergodox_right_led_3_on();
+        //     break;
+        default:
+            ergodox_board_led_off();
+            break;
+    }
+};
+"""
+
+def parse_keymaps(config):
+	keymaps = {}
+	layer_line_items = config['layer_lines'].items()
+	for i, (layer_name, layer_lines) in enumerate(layer_line_items):
+		print("parseing layer", layer_name)
+		keymap = {}
+		for code, key_index, ws in iter_keycodes(layer_lines, config):
+			keymap[key_index] = (code, ws)
+		keymaps[layer_name] = [v for k, v in sorted(keymap.items())]
+	return keymaps
+
+
+def iter_keymap_lines(config, keymaps):
+	for include_path in config['keymaps_includes']:
+		yield '#include "{}"\n'.format(include_path)
+
+	yield "\n"
+
+	layer_items = config['layers'].items()
+	for i, (layer_name, description) in enumerate(layer_items):
+		yield '#define L{0:<3} {0:<5}  // {1}\n'.format(i, layer_name)
+
+	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:
+	for data in iter_keymap_lines(config, parse_keymaps(config)):
+		fh.write(data)
+
+
+# print("\n".join(sorted(valid_keycodes)))
+# print(json.dumps(config, indent=4))

Різницю між файлами не показано, бо вона завелика
+ 1278 - 0
keyboard/ergodox_ez/keymaps/german-manuneo/keymap.c


+ 222 - 0
keyboard/ergodox_ez/keymaps/german-manuneo/keymap.md

@@ -0,0 +1,222 @@
+# ManuNeo Ergodox Keyboard Layout
+
+# Layout Config
+
+    {
+        "includes_dir": "quantum/",
+        "keymaps_includes": [
+            "ergodox_ez.h",
+            "action_layer.h",
+            "keymap_common.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_"],
+        "filler": "-+.'!x",
+        "separator": "|",
+        "macros": {
+            "MUC": "",
+        },
+    }
+
+
+# Layers
+
+
+## Layer 0
+
+    .-------------------------------------.       .-------------------------------------.
+    |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  |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |MO(1)| U  | I  | A  | E  | O  |------!       !------! S  | N  | R  | T  | D  | SS  |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |MO(3)| UE | OE | AE | C  | V  | END  |       ! TAB  | B  | M  |COMM| DOT| UP |  Y  |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |LGUI|LALT|LCTL|                               !LALT|    |LEFT|DOWN|RGHT|
+     '------------------------'.-------------. .-------------.'------------------------'
+                               | INS  |TG(2) | !M(UMS)| DELT |
+                        .------+------+------! !------+------+------.
+                        !      !      | APP  | ! PGUP |      !      !
+                        !      !      !------! !------!      !      !
+                        | BSPC | LSFT | ESC  | ! PGDN |ENTER |SPACE |
+                        '--------------------' '--------------------'
+
+
+## Layer 1
+
+    .-------------------------------------.       .-------------------------------------.
+    |     |EXLM|DQOT|PARA|    |    |      |       !      |    |    |    |    |RING|     |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     |ASTR|PIPE|SLSH|LCBR|RCBR|      |       !      |HASH|LESS|MORE|    |DQOT|     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |UNDS|MINS|AMPR|LBRC|RBRC|------!       !------!DLR |LPRN|RPRN|TILD|QUOT|QST  |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |PLUS| EQL|    |    |      |       !      |BSLS|PERC|SCLN|COLN| ↑  |     |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               !    |    | ←  | ↓  | →  |
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'
+
+
+## Layer 2
+
+    .-------------------------------------.       .-------------------------------------.
+    |     | F1 | F2 | F3 | F4 | F5 | F11  |       ! F12  | F6 | F7 | F8 | F9 |F10 |PEQL |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     |    |    |    |    |    |      |       !      |    | P7 | P8 | P9 |PAST|PSLS |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |------!       !------!    | P4 | P5 | P6 |PMNS|PMNS |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |      |       ! NLCK |    | P1 | P2 | P3 |PPLS|PPLS |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               ! P0 |PCMM|PDOT|PENT|PENT|
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'
+
+
+## Layer 3
+
+http://symbolcodes.tlt.psu.edu/bylanguage/mathchart.html
+
+    .-------------------------------------.       .-------------------------------------.
+    |     | ¹  | ²  | ³  | ⁴  | ⁵  |  ∀   |       !      | ⁶  | ⁷  | ⁸  | ⁹  |  ⁰ |     |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     | ×  | ½  | ÷  | ¼  |  ⅕ |      |       !      | ⅙  |    | ⅛  |    |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    | ±  | AT |EURO| ∅  |------!       !------! ∞  | ⁿ  | ∃  | ∈  |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     | ⅓  | ≠  | ⅔  | ¾  | ≃  |      |       !      |EXLM|    | ∄  | ∉  |    |     |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               !    |    |    |    |    |
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'
+
+
+## Layer 4
+
+    .-------------------------------------.       .-------------------------------------.
+    |     | ₁  | ₂  | ₃  | ₄  | ₅  |      |       !      | ₆  | ₇  | ₈  | ₉  | ₀  |     |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     | χ  | π  | φ  | ω  | γ  |      |       !      | η  | ξ  | κ  | λ  |    |  ζ  |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     | υ  | ι  | α  | ε  | ο  |------!       !------! σ  | ν  | ρ  | τ  | δ  |  ς  |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    | θ  |    |    |    |      |       !      | β  | μ  |    |    |    |  ψ  |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               !    |    |    |    |    |
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'
+
+
+## Layer 5
+
+    .-------------------------------------.       .-------------------------------------.
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     | Χ  | Π  | Φ  | Ω  | Γ  |      |       !      | Η  | Ξ  | Κ  | Λ  |    |  Ζ  |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     | Υ  | Ι  | Α  | Ε  | Ο  |------!       !------! Σ  | Ν  | Ρ  | Τ  | Δ  |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    | Θ  |    |    |    |      |       !      | Β  | Μ  |    |    |    |  Ψ  |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               !    |    |    |    |    |
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'
+
+
+## Layer 6
+
+    .-------------------------------------.       .-------------------------------------.
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |------!       !------!    |    |    |    |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               !    |    |    |    |    |
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'
+
+## Layer 7
+
+    .-------------------------------------.       .-------------------------------------.
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |------!       !------!    |    |    |    |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               !    |    |    |    |    |
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'
+
+
+## Layer 8
+
+    .-------------------------------------.       .-------------------------------------.
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    !-----+----+----+----+----+-----------!       !------+----+----+----+----+----+-----!
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |------!       !------!    |    |    |    |    |     |
+    !-----+----+----+----x----x----!      !       !      !----x----x----+----+----+-----!
+    |     |    |    |    |    |    |      |       !      |    |    |    |    |    |     |
+    '-----+----+----+----+----+-----------'       '-----------+----+----+----+----+-----'
+     |    |    |    |    |    |                               !    |    |    |    |    |
+     '------------------------'.-------------. .-------------.'------------------------'
+                               |      |      | !      |      |
+                        .------+------+------! !------+------+------.
+                        !      !      |      | !      |      !      !
+                        !      !      !------! !------!      !      !
+                        |      |      |      | !      |      |      |
+                        '--------------------' '--------------------'