browser,chars: error formatting for invalid input
[sheet.git] / chars.plp
1 <(common.inc.plp)><:
2
3 Html({
4         title => 'character support sheet',
5         version => '1.0',
6         keywords => [qw'
7                 unicode glyph char character reference common ipa symbol sign mark table digraph
8         '],
9         stylesheet => [qw'light dark mono circus red'],
10         data => [qw( unicode-cover.inc.pl ttfsupport unicode-char.inc.pl )],
11 });
12
13 use Shiar_Sheet::FormatChar;
14 my $glyphs = Shiar_Sheet::FormatChar->new;
15
16 my $groupinfo = do 'unicode-cover.inc.pl' or die $@ || $!;
17
18 my @ossel = @{ $groupinfo->{osdefault} };
19 my @fontlist = map { $_->{file} }
20         @{ $groupinfo->{fonts} }[ map { @{ $groupinfo->{os}->{$_} } } @ossel ];
21
22 my %font;
23 for my $fontid (@fontlist) {
24                 my ($fontmeta, @fontrange) = do "ttfsupport/$fontid.inc.pl";
25                 $fontmeta or next;
26                 $font{$fontid} = {
27                         (map { (-$_ => $fontmeta->{$_}) } keys %{$fontmeta}),
28                         map { (chr $_ => 1) } @fontrange
29                 };
30 }
31
32 # parse input
33
34 my ($title, $parent) = ('Character overview');
35 my $query = eval {
36         for ($Request || ()) {
37                 return $_ if m{^[0-9 +-]+$};
38
39                 my ($cat, $name) = split m{/}, $_, 2 or die "invalid query\n";
40                 if (!$name) {
41                         ($cat, $name) = ('table', $cat);
42                 }
43
44                 my $row = $groupinfo->{$cat}->{$name}
45                         or die "unknown character group $cat/$name\n";
46
47                 $title = ucfirst EscapeHTML($name).' characters';
48                 $parent = $cat;
49                 return EscapeHTML($row->{query});
50         }
51 } || $get{q};
52
53 say "<h1>$title</h1>";
54
55 if (!$query) {
56         Alert('Unicode group not specified', $@);
57         exit;
58 };
59
60 for ($parent || 'Unicode range') {
61         my %CATDESC = (
62                 block    => '<a href="/charset/unicode">Unicode block</a>',
63                 script   => 'Unicode script',
64                 category => 'Unicode category',
65                 table    => '<a href="/unicode">Unicode preset group</a>',
66         );
67         say sprintf('<p>List %s in selected %s.</p>',
68                 'characters and <a href="/font">font support</a>',
69                 $CATDESC{$parent} || $parent,
70         );
71 }
72
73 my @chars;
74 for (map { split /[^\d-]/ } $query) {
75         my @range = split /-/, $_, 2;
76         m/^[0-9]+$/ or die "Invalid code point $_ in query $query\n" for @range;
77         push @chars, chr $_ for $range[0] .. ($range[1] // $range[0]);
78 }
79
80 @chars or die "No match for query $query\n";
81
82 @chars <= 1500 or die sprintf(
83         'Too many matches (%d) for query %s'."\n",
84         scalar @chars, $query,
85 );
86
87 # output character list
88
89 say '<div>';
90 print '<table class="mapped cover">';
91 print '<col>' x 3;
92 print "<colgroup span=$_>"
93         for 2, map { scalar @{ $groupinfo->{os}->{$_} } } @ossel;
94
95 print '<thead><tr>';
96 print '<td colspan=3>character';
97 print '<td colspan=2>input';
98 printf '<td colspan=%d>%s', scalar @{ $groupinfo->{os}->{$_} }, $_
99         for @ossel;
100
101 print '<tr>';
102 print '<td colspan=2>unicode';
103 print '<td>name';
104 print '<td><a href="/digraphs" title="digraph">di</a><td>html';
105 printf('<td title="%s">%s', map { EscapeHTML($_) }
106         join("\n", $font{$_}->{-name}, $font{$_}->{-description}),
107         $font{$_}->{-abbr},
108 ) for @fontlist;
109 say '</thead>';
110
111 for my $chr (@chars) {
112         my $codepoint = ord $chr;
113         my $ascii = $codepoint <= 127;
114
115         say '<tr><th>', $chr;
116         my $info = $glyphs->glyph_info($codepoint);
117         my ($class, $name, $mnem, $entity, $string) = @$info;
118         print "<td>$_" for sprintf('%X', $codepoint), EscapeHTML($name || '?');
119         printf '<td class="%s">%s', @$_ for (
120                 [$ascii ? 'l0' : defined $mnem ? $class =~ /\bu-di\b/ ? 'l4' : 'l3' : 'l1',
121                         EscapeHTML($mnem) // ''],
122                 [$ascii ? 'l0' : defined $entity ? 'l4' : 'l1', $entity // ''],
123                 (map {
124                         !defined $font{$_}->{-name} ? [l0 => '?'] :
125                         $font{$_}->{$chr} ? [l4 => '✔'] : [l1 => '✘']
126                 } @fontlist),
127         );
128 }
129
130 say "</table>\n";
131 say "</div>\n";
132