Html({
title => 'character support sheet',
- version => 'v1.0',
+ version => '1.2',
keywords => [qw'
unicode glyph char character reference common ipa symbol sign mark table digraph
'],
stylesheet => [qw'light dark mono circus red'],
- data => [qw( unicode-cover.inc.pl ttfsupport unicode-char.inc.pl )],
+ data => [qw( data/unicode-cover.inc.pl data/font data/unicode-char.inc.pl )],
+ raw => <<'EOT',
+<style>
+ tbody tr:hover th {
+ font-size: 300%;
+ min-width: 1.2em;
+ border-width: 1px;
+ }
+</style>
+EOT
});
-:>
-<h1>Character support</h1>
-
-<p>
-Selected characters from Unicode <a href="/unicode">preset</a>
-or <a href="/charset">range</a>.
-</p>
-
-<div>
-
-<:
-use 5.010;
use Shiar_Sheet::FormatChar;
my $glyphs = Shiar_Sheet::FormatChar->new;
-my %oslist = (
- win95 => [qw( arial ariuni verdana times )], # microsoft
- mac10 => [qw( )], # apple
- android => [qw( droidsans )], # google
- oss => [qw( dvsans c2k unifont )],
-);
-my @ossel = qw( win95 oss android );
-
-my (%font, @fontlist);
-for my $os (@ossel) {
- my $osfonts = $oslist{$os};
- for my $fontid (@{$osfonts}) {
- push @fontlist, $fontid;
- my ($fontmeta, @fontrange) = do "ttfsupport/$fontid.inc.pl";
- $fontmeta or next;
+my $groupinfo = Data('data/unicode-cover');
+
+my @ossel = @{ $groupinfo->{osdefault} };
+my @fontlist = map { $_->{file} }
+ @{ $groupinfo->{fonts} }[ map { @{ $groupinfo->{os}->{$_} } } @ossel ];
+
+my %font;
+for my $fontid (@fontlist) {
+ my $fontmeta = eval { Data("data/font/$fontid") } or next;
$font{$fontid} = {
- -id => $fontmeta->{id} || $fontid,
- -name => $fontmeta->{name},
- map { (chr $_ => 1) } @fontrange
+ (map { (-$_ => $fontmeta->{$_}) } keys %{$fontmeta}),
+ map { (chr $_ => 1) } @{ $fontmeta->{cover} }
};
- }
}
# parse input
-my @chars;
-my @querydesc;
-
-if (my $query = $ENV{PATH_INFO} || $get{q} || 'ipa') {
- my $groupinfo = do 'unicode-cover.inc.pl' or die $@ || $!;
- for (split /[\s+]/, $query) {
- s{^/}{};
- when (qr{^[\d,;\s+-]+$}) {
- push @querydesc, "character codepoints $_";
- for (map { split /[^\d-]/ } $_) {
- my ($charnum, $range) = split /-/, $_;
- push @chars, chr $_ for $charnum .. ($range // $charnum);
- }
- }
- when ($_) {
- my $row = $groupinfo->{$_} or do {
- warn "group $_ not found";
- next;
- };
- push @querydesc, $row->{-name} // $_;
- push @chars, map { chr } @{ $row->{-chars} };
- }
- default {
- die "unknown parameter: $_\n";
+my ($title, $parent) = ('Character overview');
+my $query = eval {
+ for ($Request || ()) {
+ return $_ if m{^[0-9 +-]+$};
+
+ my ($cat, $name) = split m{/}, $_, 2 or die "invalid query\n";
+ if (!$name) {
+ ($cat, $name) = ('table', $cat);
}
+
+ my $row = $groupinfo->{$cat}->{$name}
+ or die "unknown character group $cat/$name\n";
+
+ $title = ucfirst EscapeHTML($name).' characters';
+ $parent = $cat;
+ return EscapeHTML($row->{query});
}
+} || $get{q};
+
+say "<h1>$title</h1>";
+
+if (!$query) {
+ Abort(["Unicode group not found", $@], '404 no matches');
+};
+
+for ($parent || 'Unicode range') {
+ my %CATDESC = (
+ block => '<a href="/charset/unicode">Unicode block</a>',
+ script => 'Unicode script',
+ category => 'Unicode category',
+ table => '<a href="/unicode">Unicode preset group</a>',
+ );
+ say sprintf('<p>List %s in selected %s.</p>',
+ 'characters and <a href="/font">font support</a>',
+ $CATDESC{$parent} || $parent,
+ );
+}
+
+my @chars;
+for (map { split /[^\d-]/ } $query) {
+ my @range = split /-/, $_, 2;
+ m/^[0-9]+$/ or Abort("Invalid code point $_ in query $query", 400)
+ for @range;
+ push @chars, chr $_ for $range[0] .. ($range[1] // $range[0]);
}
-@chars <= 1500 or die sprintf(
- 'too many matches (%d) for %s'."\n",
- scalar @chars, join(', ', @querydesc),
+@chars or Abort("No match for query $query", '404 no results');
+
+@chars <= 1500 or Abort(
+ sprintf('Too many matches (%d) for query', scalar @chars),
+ '403 not allowed', $query
);
# output character list
-print '<table class=mapped>';
-say '<caption>'.EscapeHTML(join ', ', @querydesc).'</caption>';
+say '<div>';
+print '<table class="mapped cover">';
print '<col>' x 3;
-print "<colgroup span=$_>" for 2, map { scalar @{$oslist{$_}} } @ossel;
+print "<colgroup span=$_>"
+ for 2, map { scalar @{ $groupinfo->{os}->{$_} } } @ossel;
print '<thead><tr>';
print '<td colspan=3>character';
print '<td colspan=2>input';
-printf '<td colspan=%d>%s fonts', scalar @{ $oslist{$_} }, $_
+printf '<td colspan=%d>%s', scalar @{ $groupinfo->{os}->{$_} }, $_
for @ossel;
print '<tr>';
print '<td colspan=2>unicode';
print '<td>name';
print '<td><a href="/digraphs" title="digraph">di</a><td>html';
-printf '<td title="%s">%s', $font{$_}->{-name}, $font{$_}->{-id} // $_
- for @fontlist;
+printf('<td title="%s">%s', map { EscapeHTML($_) }
+ join("\n", $font{$_}->{-name}, $font{$_}->{-description}),
+ $font{$_}->{-abbr},
+) for @fontlist;
say '</thead>';
for my $chr (@chars) {
my $codepoint = ord $chr;
my $ascii = $codepoint <= 127;
- print "<tr><th>$chr\n";
+ say '<tr><th>', $chr;
my $info = $glyphs->glyph_info($codepoint);
- my ($class, $name, $mnem, $html, $string) = @$info;
+ my ($class, $name, $mnem, $entity, $string) = @$info;
print "<td>$_" for sprintf('%X', $codepoint), EscapeHTML($name || '?');
printf '<td class="%s">%s', @$_ for (
- [$ascii ? 'l0' : defined $mnem ? $class =~ /\bu-di\b/ ? 'l4' : 'l3' : 'l1', $mnem // ''],
- [$ascii ? 'l0' : defined $html ? 'l4' : 'l1', $html // ''],
+ [$ascii ? 'l0' : defined $mnem ? $class =~ /\bu-di\b/ ? 'l4' : 'l3' : 'l1',
+ EscapeHTML($mnem) // ''],
+ [$ascii ? 'l0' : defined $entity ? 'l4' : 'l1', $entity // ''],
(map {
- !$font{$_}->{-id} ? [l0 => '?'] :
+ !defined $font{$_}->{-name} ? [l0 => '?'] :
$font{$_}->{$chr} ? [l4 => '✔'] : [l1 => '✘']
} @fontlist),
);
}
say "</table>\n";
-
-:></div>
+say "</div>\n";