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