+for my $c1group (@chars) {
+ print '<tbody>';
+ for my $c1 (@$c1group) {
+ print '<tr><th>', EscapeHTML($c1);
+ for my $c2 (map {@$_} @$colchars) {
+ my $mnem = $c1 . $c2;
+ if (not defined $di->{key}->{$mnem}) {
+ print '<td>';
+ next;
+ }
+ if (ref $di->{key}->{$mnem} ne 'ARRAY') {
+ printf '<td class="X Xr" title="%s">', EscapeHTML($mnem);
+ next;
+ }
+ my ($codepoint, $name, $support, $script, $string) =
+ @{ $di->{key}->{$mnem} };
+
+ my $glyph = $string || chr $codepoint;
+ utf8::upgrade($glyph); # prevent latin1 output
+ my $desc = $mnem . ($name && " ($name)");
+ my @class = ('X', grep {$_} $script);
+ push @class, $cmp ? $support : "u-$support" if $support;
+
+ $glyph = EscapeHTML($glyph);
+ $glyph = "<span>$glyph</span>" if $script =~ /\bZs\b/;
+
+ printf "\n".'<td class="%s" title="%s">%s',
+ join(' ', @class), EscapeHTML($desc), $glyph;