X-Git-Url: http://git.shiar.nl/sheet.git/blobdiff_plain/38dcfca045c69d0e303e8e5a4ca4915202e19970..3e2e19818a6102eff505984ca2430ae3ac901d08:/charset.plp diff --git a/charset.plp b/charset.plp index c83508b..4f60765 100644 --- a/charset.plp +++ b/charset.plp @@ -1,111 +1,440 @@ -<: -use utf8; -use strict; -use warnings; -use open IO => ':utf8'; - -our $VERSION = '1.0'; +<(common.inc.plp)><: -$header{content_type} = 'text/html; charset=utf-8'; +my $mode = exists $get{compare}; +my @tablist = split m{/+}, $Request || 'default'; -:> - +Html({ + title => 'charset cheat sheet', + version => '1.0', + description => [ + "Reference sheet with all glyphs in common character encoding tables,", + "and an overview of Unicode ranges and UTF-8 bytes.", + ], + keywords => [qw' + charset codepage unicode ascii utf8 latin glyph character encoding + reference common overview table + '], + stylesheet => [qw'light'], + data => [qw( + charset-encoding.inc.pl + charset-unicode.inc.pl charset-ucplanes.inc.pl charset-utf8.inc.pl + )], +}); - -charset cheat sheet - - - +use List::Util qw( first pairmap pairfirst pairs ); - -

Character encoding

+:> +

Character encodings

+

<: -my $diinfo = do 'digraphs.inc.pl'; -my %di = map { $diinfo->{$_}->[0] => $_ } grep { ref $diinfo->{$_} } - keys %$diinfo; - -use Encode qw(decode); -# generate character table(s) -# (~16x faster than decoding in loop; -# substr strings is twice as fast as splitting to an array) -my @request = ('iso-8859-1', 'cp437'); -my @tables = map { decode($_, pack 'C*', 0..255) } @request; -my $NOCHAR = chr 0xFFFD; +if ($tablist[0] eq 'default') { + say "Overview of Unicode allocation and common latin code pages."; + say "Compare alternate charsets:"; +} +else { + say "Charset comparison:"; +} -for my $cp437 (grep {$request[$_] eq 'cp437'} 0 .. $#request) { - substr($tables[$cp437], 237, 1) = pack 'U*', 0x3D5; # phi sign - substr($tables[$cp437], 0, 32) = pack 'U*', map {hex} qw( - 2007 263A 263B 2665 2666 2663 2660 2022 25D8 25CB 25D9 2642 2640 266A 266B 263C - 25BA 25C4 2195 203C 00B6 00A7 25AC 21A8 2191 2193 2192 2190 221F 2194 25B2 25BC +sub optionlink { + my ($title, $href, $selected) = @_; + return sprintf( + $selected ? '%s' : '%s', + EscapeHTML($title), $href ); } -sub quote { - local $_ = shift; - s/"/"/g; - s//>/g; - return $_; +print join " •\n", ( + map { + join " ·\n", pairmap { + optionlink($b || ucfirst $a, '/charset'.($a && "/$a?compare"), $a eq $Request); + } @{$_} + } + [ + iso => 'ISO', + win => 'Windows', + dos => 'DOS', + mac => 'Apple', + ebcdic => 'EBCDIC', + $tablist[0] eq 'default' ? () : ('' => 'common'), + ], + [ + westeur => 'West', + centeur => 'Central', + norteur => 'North European', + turkish => 0, + greek => 0, + cyrillic => 0, + hebrew => 0, + ], +); +:>. +

+ +<: +use POSIX qw( ceil ); +use Shiar_Sheet::FormatChar; +my $glyphs = Shiar_Sheet::FormatChar->new; +my @request; + +my $charsets = do 'charset-encoding.inc.pl' + or Alert('Encoding metadata could not be read', $@ || $!); + +sub tabinput { + # generate character table(s) + my $input = shift or return; + my $params = $input =~ s/[+](.*)\z// ? $1 : undef; + my $charset = $charsets->{lc $input} || {}; + + if (ref $charset ne 'HASH') { + $params and Alert("Parameters ignored for $input", + "Cannot apply $params to multiple charsets.", + ); + tabinput($_) for ref $charset ? @{$charset} : $charset; + return; + } + + state $visible = {'' => 1}; # all present tables + my %row = (offset => 0, cols => 16); + + if (not defined $params) { + my @parents = @{ $charset->{inherit} || [] }; + + if (my ($parent, $part) = pairfirst { defined $visible->{$a} } @parents) { + $row{parent} = $parent; + $params = $part; + $params = 80 unless $visible->{$parent} + or ($input eq 'MacCroatian' and defined $visible->{MacRomanian}); + } + elsif (defined $visible->{ascii}) { + $row{parent} = $parents[0]; + $params = $parents[1] // 80; + $params = 80 if hex $params >= 0x80; # ascii offset at most + } + elsif (@parents) { + $row{parent} = $parents[0]; + $params = $parents[1] if hex $parents[1] == 0; # apply ascii end + } + $visible->{$_} //= 0 for $row{parent} || (); + } + + for my $param (split /[+]+/, $params // '') { + if ($param eq 'realsize') { + $row{realsize}++; + } + elsif ($param =~ m{ \A cols = (\d+) \z }x) { + $row{cols} = $1; + } + elsif ($param =~ m{ \A (? \p{AHex}+) (?: [-] (? \p{AHex}+) )? \z }x) { + if (defined $row{endpoint}) { + # extend earlier range + my $skip = int(($row{endpoint} || $row{startpoint}) / $row{cols}); + for ($skip + 1 .. (hex($+{start}) / $row{cols}) - 1) { + $row{skip}->{ $_ * $row{cols} - $row{startpoint} }++; + } + } + else { + $row{startpoint} = hex $+{start}; + } + $row{endpoint} = hex($+{end} || 0); + } + else { + Alert("Unknown option $param for charset $input"); + } + } + + if ($input =~ m{ \A (?:wing|web)dings \d* \z }ix) { + eval "require Encode::\u$input"; + } + + if ($input eq '') { + $row{offset} = delete $row{startpoint}; + $row{set} = 'Unicode characters'; + my $block = $row{offset} >> 8; + $row{endpoint} ||= ($block + 1 << 8) - 1; + $block == ($row{endpoint} >> 8) or undef $block; + + $row{table} = join '', map { chr } $row{offset} .. $row{endpoint}; + utf8::upgrade($row{table}); # prevent latin1 output + + $row{endpoint} -= $row{offset}; + + if (defined $block) { + $row{set} = sprintf 'Unicode block U+%02Xxx', $block; + $row{offset} %= 0x100; + } + } + elsif (lc $input eq 'uu') { + $row{cell} = do 'charset-ucplanes.inc.pl' + or Alert('Table data could not be read', $@ || $!); + $row{endpoint} ||= 0x3FF; + $row{set} = 'Unicode planes'; + } + elsif (lc $input eq 'u') { + $row{cell} = do 'charset-unicode.inc.pl' + or Alert('Table data could not be read', $@ || $!); + + $row{endpoint} ||= 0x1FFF; + $row{set} = 'Unicode ' . ( + $row{startpoint} < 0x1000 && $row{endpoint} < 0x1000 ? 'BMP' : + $row{startpoint} >= 0x1000 && $row{endpoint} < 0x2000 ? 'SMP' : + 'allocations' + ); + } + elsif ($input =~ m/^utf-*8$/i) { + $row{set} = 'UTF-8'; + $row{cell} = do 'charset-utf8.inc.pl' + or Alert('Table data could not be read', $@ || $!); + $row{endpoint} = 0xFF; + } + elsif ($row{set} = Encode::resolve_alias($input)) { + $row{offset} = delete $row{startpoint}; + $row{endpoint} ||= 0xFF; + if ($row{set} eq 'MacHebrew' or $row{set} eq 'MacThai') { + # array of possibly multiple characters per code point + $row{table} = [ + map { Encode::decode($row{set}, pack 'C*', $_) } $row{offset} .. $row{endpoint} + ]; + } + else { + # ~16x faster than decoding in loop; + # substr strings is twice as fast as splitting to an array + $row{table} = Encode::decode($row{set}, pack 'C*', $row{offset} .. $row{endpoint}); + } + + if ($row{set} eq 'cp437') { + if ($row{offset} <= 0xED and $row{endpoint} >= 0xED) { + # replace phi glyph + substr($row{table}, 0xED - $row{offset}, 1) = 'ϕ'; + } + if ($row{offset} < 0x20) { + # replace control characters by visible variants + my $sub = substr ' ☺☻♥♦♣♠•◘○◙♂♀♪♫☼►◄↕‼¶§▬↨↑↓→←∟↔▲▼', $row{offset}; + substr($row{table}, 0, length $sub) = $sub; + } + } + elsif ($row{set} eq 'symbol') { + if ($row{offset} <= 0x60 and $row{endpoint} >= 0x60) { + # replace radical extender by closest unicode equivalent + substr($row{table}, 0x60 - $row{offset}, 1) = '│'; + } + if ($row{offset} <= 0xBD and $row{endpoint} >= 0xFF) { + substr($row{table}, 0xBD - $row{offset}, 2) = '⏐⎯'; # arrow extenders + substr($row{table}, 0xD2 - $row{offset}, 3) = '®©™'; # serif variants + substr($row{table}, 0xE0 - $row{offset}, 1) = '◊'; # replace lookalike, should match AdobeSymbol + substr($row{table}, 0xE2 - $row{offset}, 3) = '®©™'; # sans-serif variants + substr($row{table}, 0xE6 - $row{offset}, 10) = '⎛⎜⎝⎡⎢⎣⎧⎨⎩⎪'; + substr($row{table}, 0xF0 - $row{offset}, 1) = '€'; + substr($row{table}, 0xF4 - $row{offset}, 11) = '⎮⌡⎞⎟⎠⎤⎥⎦⎫⎬⎭'; + } + } + + $row{endpoint} -= $row{offset}; + + $visible->{ascii} = # assume common base + $visible->{ $row{set} } = 1; + } + else { + Alert("Encoding $input unknown"); + return; + } + push @request, \%row; } +tabinput($_) for @tablist; + +my $NOCHAR = chr 0xFFFD; + +sub range_cell { + my ($info, $offset) = @_; + my $table = $info->{cell} or return; + my $def = $table->{$offset} or return; + my ($len, $class, $name, $title) = @{$def}; -print "\n"; - :>
- +
+ <: if ($mode) { :> +
unicode + inherited + existing + original + unassigned +<: } else { :> + control + whitespace + diacritic +
letter +
+
punctuation +
quote +
+
symbol +
math + currency +
+
numeric + greek +
latin + cyrillic +
+
aramaic +
brahmic + arabic +
+
syllabic +
african + japanese + cjk + chinese +
+
alphabetic +
+ + +
unicode 7.0 + proposed + deprecated + unassigned + invalid +<: } :>
+
-