X-Git-Url: http://git.shiar.nl/sheet.git/blobdiff_plain/024fef82307d7b421c8ec3ebee930b774427a745..a6bf5822695c1e3608f1692398d47e28a9623a79:/writing-latn.inc.pl?ds=inline
diff --git a/writing-latn.inc.pl b/writing-latn.inc.pl
index e7a418d..4ad57ff 100644
--- a/writing-latn.inc.pl
+++ b/writing-latn.inc.pl
@@ -1,322 +1,803 @@
+use 5.014;
use utf8;
+use warnings;
+use List::Util qw( pairs pairmap sum min max );
+
+my %C = (
+ red => '#EC1C24',
+ blue => '#3953A3',
+ yellow => '#F9EC31',
+ black => '#231F20',
+);
+my $U = 0; # optional unicode alternatives
+
+my @wrapstyle = (
+ 'td { white-space: normal; word-spacing: 10em }',
+ # force line break between words
+ '.sample { word-spacing: 0 }',
+ '.sample svg { margin-right: 1ex; white-space: nowrap; display: inline-block }',
+ # larger space between letters
+);
+my $spacestyle = '.sample svg { margin-right: 0.5ex }'; # separate letters
+my @tapstyle = (
+ @wrapstyle,
+ '{ line-height: 1ex }',
+ 'td:not(.sample) { vertical-align: top }',
+ '.sample { font-size: 80% }',
+);
+
+my @hueorder = (
+ 2,11,20,19,18,21,24,15,6,7,8,5,13, # red .. magenta, grey
+ 1,10,9,12,3,4,0, 14,23,22,25,16,17,26, # dark, light hues
+);
+
+# Order to put similar sounds close to each other:
+# â R Y G C B M X
+# ââ¼ââââââââââââââ
+# Wâ o e y h s f -
+# âmuaixqgkdtbp l
+# Kâ w n j c z v r
+
+my @hueletters = ((26) x 27);
+@hueletters[map { ord($_) - ord('a') } qw(
+ u a i x q g k d t b p m l w n j c z v r o e y h s f
+)] = @hueorder;
sub disptap {
- return map {
- m/\A(-?)(\d)(\d)/;
- sprintf('
%s %s',
- ' class=ex' x !!$1,
- '·' x $2, '·' x $3,
- );
- } @_;
+ my $code = shift;
+ my ($prefix, @dots) = $code =~ m/\A(-?)(\d)(\d)/ or return $code;
+ if ($U) {
+ # unicode glyph alternative as DOMINO TILE HORIZONTAL-0a-0b
+ return $prefix . chr(0x1F031 + ($dots[0] * 7) + $dots[1]);
+ }
+ return $prefix . join(' ', map { '·' x $_ } @dots);
}
-my %C = qw(red #EC1C24 blue #3953A3 yellow #F9EC31 black #231F20);
-
-(
-'Uppercase' => [qw{ A B C D E F G H I J K L M N O P Q R S T U V W X Y Z }],
-'Lowercase' => [qw{ a b c d e f g h i j k l m n o p q r s t u v w x y z }],
-'Sütterlin' => [qw{ a b c d e f g h i j k l m n o p q r Å¿âs t u v w x y z }],
-'Old Roman Cursive' => [
- map { $_ && '' || '' }
- "m 9.7966963,233.64839 c 1.9349047,4.6581 17.4141427,19.56404 17.4141427,19.56404 M 8.8650753,258.0405 17.679641,245.75946",
- "m 8.2859683,233.93504 c 0,0 6.5438297,-3.56736 8.4562497,2.86653 1.38905,4.67312 0.39098,21.06633 6.59301,19.20572 2.86653,-0.85996 4.58644,-3.00985 4.58644,-3.00985 m -14.84083,-3.65482 c -11.2172001,9.19952 -0.12626,13.80702 1.43325,7.02298",
- "m 14.579584,240.38472 8.45625,-4.87309 m -9.17289,8.7429 c 0,0 -4.1564597,16.15085 6.01971,13.47267",
- "m 10.187496,230.36768 c 0,0 15.43006,23.12513 15.71671,27.09324 m -5.11056,-9.76656 c -12.0394097,4.29979 -8.88624,20.06568 2.86652,6.87966",
- "m 10.586898,246.54775 9.45953,0.28666 m 5.4464,-16.46673 c 0,0 -14.90593,2.42075 -10.60614,35.9591",
- "m 13.108831,249.98758 11.03612,-4.87309 m -7.5963,-8.31292 16.05255,-9.02956 M 3.4754013,265.23104 c 0,0 7.9135097,6.73875 11.3533397,-6.6889 3.43983,-13.42765 3.43983,-22.60053 3.43983,-22.60053",
- "m 11.344837,238.95146 15.19258,-4.87309 m -16.37398,8.58517 c -3.5915897,11.9003 8.63168,20.00759 12.55933,14.16049 l 1.6198,6.78546",
- "m 21.047523,245.54447 9.02955,-1.00328 m -19.77903,0.71663 c 10.03284,0.14332 12.46939,-2.74357 12.32606,12.89936 M 6.0024733,230.97576 c 5.4421897,-0.20269 3.8656,27.56638 3.8656,27.56638",
- "m 18.254694,242.9646 -0.42998,15.57754",
- "m 19.142047,239.22263 c -0.81078,27.76908 -2.18046,28.04132 -2.18046,28.04132",
- "m 24.298444,237.25928 -11.41234,8.25108 9.1827,8.57252 m -8.55222,-21.67887 -1.75469,28.5974",
- "m 11.205353,230.36768 c 0,0 -2.5381497,20.65982 1.39849,22.65013 10.92663,5.52433 13.05287,4.57023 13.05287,4.57023",
- "m 3.5407243,260.23445 5.87812,-21.68826 7.9050697,12.76972 4.25657,-11.75625 10.94548,9.32392",
- "m 9.5258083,261.6533 2.8377097,-21.08017 10.1347,19.05323 4.05388,-20.67478",
- "m 16.422109,242.60007 c -5.47274,0.81078 -3.47429,13.92701 1.62155,11.14817 9.84072,-5.3663 1.82425,-10.74278 1.82425,-10.74278",
- "m 15.404684,243.61354 c 0,0 -3.04041,17.43168 4.05388,12.97241 m -8.10776,-16.4182 13.3778,6.28351",
- "m 10.672413,237.73542 21.08017,26.35021 m -20.87748,-24.32327 c -9.1212296,3.2431 -6.0808197,9.32392 -6.0808197,9.32392 0,0 5.4727397,5.27004 12.5670297,-1.21616",
- "m 4.8655023,238.74889 c 6.0808197,-2.43234 6.4862097,-2.83773 12.9724097,0.40538 4.78633,2.39317 7.29698,6.89161 13.3778,3.24311 m -15.20204,-3.4458 -5.47274,24.93136",
- "m 34.256454,230.64113 c 0,0 -10.58398,5.28535 -17.9029,8.81482 -3.67044,7.38064 -2.15569,11.07749 -5.4069,20.37311 -1.1252097,3.21711 -9.12123,4.25657 -9.12123,4.25657",
- "m 16.823534,243.81623 c 0,0 -1.62155,18.24246 5.87812,11.55356 M 8.3103943,240.37044 27.769014,239.96505",
- "m 9.0086423,236.11386 c 0,0 6.6888997,11.14817 12.7697197,6.28352 6.08082,-4.86466 5.27005,-6.08082 5.27005,-6.08082",
- '',
- '',
- "m 11.948695,268.77453 c -1.62156,-14.39127 13.58049,-39.32263 13.58049,-39.32263 M 6.8813453,243.84317 c 0.20269,1.01347 22.2963397,2.83772 22.2963397,2.83772",
- "m 4.9289653,239.8092 c 4.44908,-7.43435 11.3160897,-5.14113 12.3295597,2.96663 l 0.20269,23.71519 c 0,0 -0.10134,-9.32393 -0.12668,-23.2338 -0.0163,-8.91571 9.90666,-8.94388 13.80852,-6.05549",
- "m 12.020439,237.63036 c 9.02956,-1.71991 6.59301,5.87638 6.59301,5.87638 -1.78773,5.95591 -3.91891,10.78875 -1.86323,12.18274 2.70729,1.83586 7.30963,0.28664 7.30963,0.28664",
-],
-'Sutton ASL' => [
- # American manual alphabet in Sutton (U+1D800+) notation
- map { pack 'W*', map { hex "1D$_" } unpack '(A3)*', $_ } qw{
- 8F7a9c 847a9c 86Da9c 801a9c 84Aa9c
- 8CEa9c 8F0 815aa2 892a9c 892a9c9A2aac
- 840a9c 8DCa9c 88Da9c
- 819a9c 876a9c 840a9caA1 8F0a9caA1 81Aa9c
- 903a9c 8FBa9c 815a9c 80Ea9c 887a9c
- 806a9c 89Aa9c 800a9c945aaa
-}],
-'Braille' => [qw{ â â â â â â â â â â â â â â â â â â â â â ¥ â § â º â â ½ â µ }],
-'5-point Tactile' => [
- map { '' }
- map {
- join '', map { sprintf '',
- !$_ ? 9 : $_ & 1 ? 4 : 14,
- !$_ ? 12 : $_ < 3 ? 4 : 20,
- } split //
- }
- qw{
- 4 234 012 14 0 014 023 12
- 02 024 0134 23 013 03 01 123
- 0124 13 04 1 34 0123 134 0234
- 034 124
- }
-],
-'Morse' => [map {tr/.-/â§â/r} qw{
- .- -... -.-. -.. . ..-. --. .... .. .--- -.- .-.. --
- -. --- .--. --.- .-. ... - ..- ...- .-- -..- -.-- --..
-}],
-'Tap code' => [disptap(qw{
- 11 12 13 14 15 21 22 23 24 -24 25 31 32
- 33 34 35 41 42 43 44 45 51 52 53 54 55
-})],
-'Tap simplified' => [disptap(qw{
- 11 12 13 14 21 22 23 20 31 -31 -13 32 33
- 30 41 42 -13 43 40 10 51 52 53 50 -31 -40
-})],
-'Maritime flags' => [
- # International Code of Signals, SVG fills
- map { '' }
- split /\n\n/, qq{
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- },
-],
-'Flag semaphore' => [
- map {
- local $_ = $_;
- s/[1-4]\K(?=[4-9])/Â /;
- tr/1-9/ââââââââ/;
- s{(\S)(?=.)}{$1};
- $_
- }
- qw(
- 1 2 3 4 5 6 7 21 31 46 14 51 16 17 23
- 24 25 26 27 34 35 47 56 57 36 67
- )
-],
-'Chappe semaphore' => [
- map {
- my ($r, $pr, $pl) = split //, $_;
- !$_ ? '-' : sprintf(
+sub dispdomino {
+ my $code = shift;
+ my ($prefix, @dots) = $code =~ m/\A(-?)(\d)(\d)/ or return $code;
+ # unicode glyph alternative as DOMINO TILE HORIZONTAL-0a-0b
+ return $prefix . chr(0x1F031 + ($dots[0] * 7) + $dots[1]);
+}
+
+sub dispdash {
+ my $code = shift;
+ my ($prefix, @dots) = $code =~ m/\A(-?)(\d)(\d)/ or return $code;
+ my ($w, $h) = (max(6, 4 * max(@dots)), 9);
+ my ($w0, $w1) = ($w / $dots[0], $dots[1] ? $w / $dots[1] : 1);
+ return sprintf(
+ '',
+ $prefix && ' style="opacity:.5"',
+ $w + 1, $h + 1, join(' ',
+ "m0,$h l+$w0,-$h" x $dots[0], # slashes
+ "m0,$h l-$w1,-$h" x $dots[1], # backslashes
+ )
+ );
+}
+
+sub dispblock {
+ my $code = shift;
+ my ($prefix, $shape, $rotate) = $code =~ m/\A(-?)(\w)(\d?)/
+ or return $code;
+ my %path = (
+ S => 'm0,1h1v-1h1',
+ Z => 'm0,0h1v1h1',
+ L => 'm0,0v2h1',
+ v => 'm0,0v1h1',
+ J => 'm1,0v2h-1',
+ T => 'm0,0h2.5h-1.5v1',
+ O => 'm0,0h1v1h-1',
+ I => 'm0,1h3',
+ i => 'm0,1h2',
+ o => 'm0,1h1',
+ );
+ my %col = (
+ S => 120, Z => 0, J => 240, L => 30, T => 300, O => 60, I => 180,
+ v => '45,50%', i => '165,50%', o => '165,0%',
+ );
+ s/\z(?'
+ . '',
+ $code eq 'I' ? 4 : $code =~ /T3|[LJO]$|[Iio]1/ ? 2 : 3,
+ @gaps ? qq( class="@gaps") : '',
+ $path{$shape}, "hsl($col{$shape},50%)", join('',
+ $rotate && sprintf(' transform="rotate(%s 1 1)"', $rotate * 90),
+ $prefix && ' style="opacity:.5"',
+ ),
+ );
+}
+
+sub dispbar {
+ my $code = shift or return '';
+
+ return join '', pairmap {
+ ($a =~ tr/123/âââ/r) . ($b =~ tr/321/ââ/dr)
+ } split //, $code if $U;
+
+ my @cols = split //, $code; # bar and space widths
+ my $width = sum(@cols);
+ return sprintf(
+ '',
+ $width * 2, 14, $width, 7, join(' ',
+ 'M0,0',
+ map {
+ join('m1,-7', ('v7') x $_->[0]), # line per bar width
+ (map { sprintf 'm%d,-7', $_ + 1 } $_->[1] || ()), # space forward
+ }
+ pairs @cols
+ )
+ );
+}
+
+sub disphues {
+ my ($index, $hues, $opaque) = @_;
+ $index >= 0 or $index = 26;
+ my @lum = ($index % 3, $index / 3 % 3, $index / 9); # hue opacities (0..2)x3
+ my @lumf = $opaque ? ('hsl(%s,100%%,50%%)', 'hsl(%s,100%%,25%%)') :
+ ('hsl(%s,100%%,50%%)', 'hsla(%s,100%%,50%%,.5)');
+ return sprintf(
+ '',
+ join '', map {
+ my $colf = $lumf[ $lum[$_] ];
+ !$colf ? () : sprintf('',
+ 5 + $_, $_ == 1 ? 7 : 5, 5, sprintf($colf, $hues->[$_])
+ );
+ } 0 .. 2
+ );
+}
+
++{
+default => [qw( written sign digital touch tactile sound games semaphore barcode personal )],
+written => [qw( uppercase lowercase suetterlin roman )],
+digital => [qw( stroke ita2 )],
+stroke => [qw( graffiti unistrokes edgewrite )],
+touch => [qw( moon braille )],
+sign => ['sutton'],
+sound => [qw( morse tap shorttap )],
+games => [qw( domino tetromino cards )],
+semaphore => [qw( maritime flag chappe prussian )],
+barcode => [qw( rm4scc code39 code93 code128 )],
+personal => [qw( rgbmap cmymap dni pigpen nyctographs chromacons )],
+
+order => {
+ name => '#',
+ list => [1 .. 26],
+},
+uppercase => {
+ list => [qw{ A B C D E F G H I J K L M N O P Q R S T U V W X Y Z }],
+},
+lowercase => {
+ list => [qw{ a b c d e f g h i j k l m n o p q r s t u v w x y z }],
+},
+suetterlin => {
+ name => 'Sütterlin',
+ style => [
+ '@font-face {
+ font-family: Suetterlin; /* R. G. Arens */
+ src: url("/suetterlin.ttf");
+ }',
+ 'td { font-family: Suetterlin }',
+ 'td:hover::first-letter { text-transform: uppercase }',
+ ],
+ list => [qw{ a b c d e f g h i j k l m n o p q r Å¿Â s t u v w x y z }],
+},
+roman => {
+ name => 'Old Roman Cursive',
+ style => [
+ 'svg path { stroke-linecap: round; stroke-linejoin: round }',
+ '.sample svg { margin-right: -10px }',
+ ],
+ list => [
+ map {
+ s{\A-?\K(\w.+)}
+ {}r
+ }
+ "m2,4 c1,2 8,9 8,9 M2,15 6,9",
+ "m2,4 c0,0 3,-2 4,1 1,2 0,9 3,9 1,-0 2,-1 2,-1 m-6,-2 c-5,4 -0,6 1,3",
+ "m4,7 4,-2 m-4,4 c0,0 -2,7 3,6",
+ "m3,2 c0,0 7,10 7,12 m-2,-4 c-5,2 -4,9 1,3",
+ "m3,10 4,0 m2,-7 c0,0 -7,1 -5,16",
+ "m4,11 5,-2 m-3,-4 7,-4 M0,18 c0,0 4,3 5,-3 2,-6 2,-10 2,-10",
+ "m3,6 7,-2 m-7,4 c-2,5 4,9 6,6 l1,3",
+ "m7,9 4,-0 m-8,0 c4,0 6,-1 5,6 M1,3 c2,-0 2,12 2,12",
+ '>', # i = j
+ "m6,8 -0,7",
+ "-m9,5 -5,4 4,4 m-4,-10 -1,13",
+ "m3,2 c0,0 -1,9 1,10 5,2 6,2 6,2",
+ "m0,16 3,-10 4,6 2,-5 5,4",
+ "m2,16 1,-9 5,8 2,-9",
+ "m5,8 c-2,0 -2,6 1,5 4,-2 1,-5 1,-5",
+ "m5,8 c0,0 -1,8 2,6 m-3,-7 5,3",
+ "m3,6 9,12 m-9,-11 c-4,1 -3,4 -3,4 0,0 2,2 6,-1",
+ "m0,6 c3,-1 3,-1 6,0 2,1 3,3 6,1 m-7,-1 -2,11",
+ "m13,3 c0,0 -5,2 -8,4 -2,3 -1,5 -2,9 -1,1 -4,2 -4,2",
+ "m2,7 8,0 m-4,1 c0,0 -1,8 3,5",
+ '>', # u = v
+ "m2,5 c0,0 3,5 6,3 3,-2 2,-3 2,-3",
+ '-',
+ "m3,19 c-1,-6 6,-17 6,-17 M1,8 c0,0 10,1 10,1",
+ "-m0,7 c2,-3 5,-2 5,1 l0,11 c0,0 -0,-4 -0,-10 -0,-4 4,-4 6,-3",
+ "-m3,6 c4,-1 3,3 3,3 -1,3 -2,5 -1,5 1,1 3,0 3,0",
+ ],
+},
+sutton => {
+ name => 'Sutton ASL',
+ style => $spacestyle,
+ list => [
+ # American manual alphabet in Sutton (U+1D800+) notation
+ map { !!$_ && pack 'W*', map { hex "1D$_" } unpack '(A3)*', $_ } qw{
+ 8F7a9c 847a9c 86Da9c 801a9c 84Aa9c
+ 8CEa9c 8F0 815aa2 892a9c 892a9c9A2aac
+ 840a9c 8DCa9c 88Da9c
+ 819a9c 876a9c 840a9caA1 8F0a9caA1 81Aa9c
+ 903a9c 8FBa9c 815a9c 80Ea9c 887a9c
+ 806a9c 89Aa9c 800a9c945aaa
+ 0 965aa6
+ }],
+},
+graffiti => {
+ name => 'Palm Graffiti',
+ style => [
+ '@font-face {
+ font-family: Graffiti;
+ src: url("/graffiti.ttf");
+ }',
+ 'td { font-family: Graffiti }',
+ ],
+ list => [qw{ a b c d e f g h i j k l m n o p q r s t u v w * y z }],
+},
+unistrokes => {
+ name => 'Unistrokes',
+ url => 'https://www.google.com/patents/US5596656', # by Xerox
+ style => 'svg path { stroke-linecap: round; stroke-linejoin: round }',
+ list => [
+ map { '' }
+ map {
+ my ($x, $y, $next) = m/\AM(\d+),(\d+)(.)?/;
+ sprintf('', $x, $y) . # start point
+ (defined $next && qq())
+ }
+ 'M3,8 V0',
+ 'M0,0 6,4 0,8',
+ 'M6,0 0,4 6,8',
+ 'M6,0 0,4 6,8',
+ 'M6,4 H0',
+ 'M6,0 0,0 0,8',
+ 'M0,8 6,8 6,0',
+ 'M0,0 6,0 6,8',
+ 'M3,0 V8',
+ 'M6,0 6,8 0,8',
+ 'M0,8 6,0',
+ 'M0,0 0,8 6,8',
+ 'M6,8 3,0 0,8',
+ 'M0,8 3,0 6,8',
+ 'M6,0 Q0,6 3,8 6,6 0,0',
+ 'M0,0 Q4,8 6,4 4,0 0,8',
+ 'M6,0 Q2,8 0,4 2,0 6,8',
+ 'M0,0 6,8',
+ 'M6,0 0,0 6,8 0,8',
+ 'M0,4 H6',
+ 'M6,0 3,8 0,0',
+ 'M0,0 3,8 6,0',
+ 'M0,0 0,8 6,0 6,8',
+ 'M0,0 Q6,6 3,8 0,6 6,0',
+ 'M6,0 0,8',
+ 'M0,0 6,0 0,8 6,8',
+ 'M3,4',
+ ],
+},
+edgewrite => {
+ name => 'EdgeWrite',
+ url => 'http://depts.washington.edu/ewrite/', # patented US7729542
+ style => 'svg path { stroke-linecap: round; stroke-linejoin: round }',
+ list => [
+ map { '' }
+ map {
+ my @route = split //;
+ my @coords = map { $_ % 2 << 3, $_ >> 1 << 3 } @route; # x,y,
+ sprintf('',
+ @coords[0, 1], # start point
+ join(' ', map {
+ my $pos = join(',', @coords[$_*2, $_*2 + 1]);
+ $_ > 1 && $route[$_] == $route[$_ - 2] # curve back
+ ? 'Q4,4 '.$pos.'L' : $pos
+ } 0 .. $#route),
+ )
+ }
+ # corners (0..3) clockwise from top-left in order
+ qw(
+ 213 0232 1023 1323 103 102 10132 0213 02 132 02123 023 20313 2031
+ 10231 0102 10131 201 1032 013 0231 021 02131 0312 0313 0123 01
+ )
+ ],
+},
+ita2 => {
+ name => 'ITA2',
+ style => [@wrapstyle, 'td { font-size: 50% }'],
+ list => [map { tr/01/ââ/r =~ s/..\K/ /r } qw(
+ 11000 10011 01110 10010 10000 10110 01011 00101 01100 11010 11110 01001 00111
+ 00110 00011 01101 11101 01010 10100 00001 11100 01111 11001 10111 10101 10001
+ 00100
+ )],
+},
+moon => {
+ list => [
+ map { qq() }
+ 'M0,6 3,0 6,6',
+ 'M1,0 V4 A2,2 0,0,0 5,4',
+ 'M5,0 A4.5,3 0,0,0 5,6',
+ 'M1,0 A4.5,3 0,0,1 1,6',
+ 'M0,6 V0 H6',
+ 'M1,6 V2 A2,2 0,0,1 5,2',
+ 'M5,6 V2 A2,2 0,1,0 1,2',
+ 'M1.5,3 A1.5,1.5 0,0,0 4.5,3 1.5,1.5 0,0,0 1.5,3 M3,1.5 A1,1.5 0,0,0 3,4.5',
+ 'M3,0 V6',
+ 'M5,0 V4 A2,2 0,0,1 1,4',
+ 'M6,0 0,3 6,6',
+ 'M0,0 V6 H6',
+ 'M0,0 H6 V6',
+ 'M0,6 V2 L6,6 V0',
+ 'M0,3 A3,3 0,0,0 6,3 3,3 0,0,0 0,3',
+ 'M6,4 H2 A2,1 0,0,1 2,2',
+ 'M0,4 H4 A2,1 0,0,0 4,2',
+ 'M0,0 6,6',
+ 'M0,6 6,0',
+ 'M0,3 H6',
+ 'M0,0 V3 A3,3 0,0,0 6,3 V0',
+ 'M0,0 3,6 6,0',
+ 'M0,6 V3 A3,3 0,0,1 6,3 V6',
+ 'M0,0 6,3 0,6',
+ 'M6,0 V6 H0',
+ 'M0,0 H6 L2,6 H6',
+ ],
+ style => 'svg path { stroke-linecap: round; stroke-linejoin: round }',
+},
+braille => {
+ list => [qw{ â â â â â â â â â â â â â â â â â â â â â ¥ â § â º â â ½ â µ }],
+},
+tactile => {
+ name => '5-point Tactile',
+ list => [
+ map { '' }
+ map {
+ join '', map { sprintf '',
+ !$_ ? 9 : $_ & 1 ? 4 : 14,
+ !$_ ? 12 : $_ < 3 ? 4 : 20,
+ } split //
+ }
+ qw{
+ 4 234 012 14 0 014 023 12
+ 02 024 0134 23 013 03 01 123
+ 0124 13 04 1 34 0123 134 0234
+ 034 124
+ }
+ ],
+},
+morse => {
+ style => $spacestyle,
+ list => [map {tr/.-/â§â/r} qw{
+ .- -... -.-. -.. . ..-. --. .... .. .--- -.- .-.. --
+ -. --- .--. --.- .-. ... - ..- ...- .-- -..- -.-- --..
+ }],
+},
+tap => {
+ name => 'Tap code',
+ style => \@tapstyle,
+ list => [map { disptap($_) } qw{
+ 11 12 13 14 15 21 22 23 > 24 25 31 32
+ 33 34 35 41 42 43 44 45 51 52 53 54 55
+ }],
+},
+shorttap => {
+ name => 'Short Tap',
+ style => \@tapstyle,
+ altlist => [map { disptap($_) } qw{
+ 11 12 13 14 21 22 23 20 > 31 -13 32 33
+ 30 41 42 -13 43 40 10 51 52 53 50 -31 -40
+ }],
+ list => [map { dispdash($_) } qw{
+ 10 14 -24 12 20 23 22 21 30 -34 13 33 32
+ 31 40 43 -13 42 41 11 50 -53 -44 -52 -51 -54
+ }],
+},
+domino => {
+ name => 'Domino tiles',
+ style => [
+ # enlarge single tile height to span full vertical combinations
+ 'td { font-size: 200%; line-height: .6; padding: 0 0 .3ex }',
+ ],
+ list => [map { dispdomino($_) } qw{
+ 10 11 20 21 22 30 31 32 33 40 41 42 43
+ 44 50 51 52 53 54 55 60 61 62 63 64 65
+ }],
+},
+tetromino => {
+ style => [
+ 'svg path { stroke-linecap: square }',
+ '.sample .gapl1 + .gapr1 { margin-left: -6.3px }',
+ ],
+ name => 'Tetrominos',
+ list => [map { dispblock($_) } qw{
+ T2 T1 I T3
+ i L1 J L3
+ o1 I1 L2 L -S1 Z1
+ O J2 v2 -J3 S T
+ J1 v1 v v3 i1 Z
+ }],
+},
+cards => {
+ style => 'td { font-family: Symbola, "DejaVu Sans", serif, sans }',
+ list => [(
+ map { chr(0x1F0A0 + $_), sprintf('%s', chr(0x1F0B0 + $_)) } # spades, hearts
+ 1 .. 11, 13, 14 # A 2-10 J Q K
+ ), '', chr(0x1F0CF), chr(0x1F0DF) ],
+},
+maritime => {
+ name => 'Maritime flags',
+ style => $spacestyle,
+ list => [
+ # International Code of Signals, SVG fills
+ map { !!$_ && '' }
+ split /\n\n/, qq{
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ },
+ ],
+},
+flag => {
+ name => 'Flag semaphore',
+ list => [
+ map {
+ local $_ = $_;
+ s/[1-4]\K(?=[4-9])/Â /; # prevent unwanted vertical crossing
+ tr/1-9/ââââââââ/;
+ s{(\S)(?=.)}{$1};
+ $_
+ }
+ qw(
+ 1 2 3 4 5 6 7 21 31 46 14 51 16 17 23
+ 24 25 26 27 34 35 47 56 57 36 67
+ )
+ ],
+},
+chappe => {
+ name => 'Chappe semaphore',
+ list => [
+ map {
+ my ($r, $pr, $pl) = split //, $_;
+ /^\D$/ ? $_ : sprintf(
+ join('',
+ '',
+ ),
+ ['6', '3v3', '9v-3']->[$pl],
+ [ '', 'v3', 'v-3']->[$pr],
+ $r * 45,
+ );
+ }
+ # 360° rotation (0-7) and position state (0-2) of left and right bars
+ qw(
+ 021 121 221 321 421 521 621 721
+ > 022 122 222 322 011 111 211 311
+ 001 101 201 301 401 501 601 701 020
+ )
+ ],
+},
+prussian => {
+ name => 'Prussian semaphore',
+ list => [
+ map { /^\D+$/ ? $_ : sprintf
join('',
- '