1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
|
#!/usr/bin/perl
# SPDX-License-Identifier: BSD-2-Clause
# Copyright 1996-2017 The NASM Authors - All Rights Reserved
#
# Parse AFM metric file, returns a reference to fontdata
#
sub parse_afm_file($$) {
my($filename, $filetype) = @_;
my $fontdata = {
widths => {},
kern => {}
};
my $charmetrics = 0;
my $kerndata = 0;
my $charcode, $width, $name;
my $fontfile = $filename.'.'.$filetype;
return undef unless ( -f $fontfile );
$fontdata->{file} = $fontfile;
$fontdata->{type} = $filetype;
$fontdata->{scale} = 1000; # AFM metrics always have scale 1000
return undef unless (open(my $fh, '<', $filename.'.afm'));
while ( my $line = <$fh> ) {
if ( $line =~ /^\s*FontName\s+(.*)\s*$/i ) {
$fontdata->{'name'} = $1;
} elsif ( $line =~ /^\s*StartCharMetrics\b/i ) {
$charmetrics = 1;
} elsif ( $line =~ /^\s*EndCharMetrics\b/i ) {
$charmetrics = 0;
} elsif ( $line =~ /^\s*StartKernPairs\b/i ) {
$kerndata = 1;
} elsif ( $line =~ /^\s*EndKernPairs\b/i ) {
$kerndata = 0;
} elsif ( $charmetrics ) {
my @data = split(/\s*;\s*/, $line);
undef $charcode, $width, $name;
foreach my $d ( @data ) {
my @dd = split(/\s+/, $d);
if ( $dd[0] eq 'C' ) {
$charcode = $dd[1];
} elsif ( $dd[0] eq 'WX' ) {
$width = $dd[1];
} elsif ( $dd[0] eq 'W' ) {
$width = $dd[2];
} elsif ( $dd[0] eq 'N' ) {
$name = $dd[1];
}
}
if ( defined($name) && defined($width) ) {
$fontdata->{widths}{$name} = $width;
}
} elsif ( $kerndata ) {
my($kpx, $a, $b, $adj) = split(/\s+/, $line);
if ( $kpx eq 'KPX' ) {
if (!exists($fontdata->{kern}{$a})) {
$fontdata->{kern}{$a} = {};
}
$fontdata->{kern}{$a}{$b} = $adj;
}
}
}
return $fontdata;
}
1;
|