my
$DEFFAMILY
=
'serif'
;
my
$DEFSERIES
=
'medium'
;
my
$DEFSHAPE
=
'upright'
;
my
$DEFCOLOR
=
'black'
;
my
$DEFBACKGROUND
=
undef
;
my
$DEFOPACITY
=
'1'
;
my
$DEFENCODING
=
'OT1'
;
my
$DEFLANGUAGE
=
undef
;
sub
DEFSIZE {
return
$STATE
->lookupValue(
'NOMINAL_FONT_SIZE'
) || 10; }
my
$FLAG_FORCE_FAMILY
= 0x1;
my
$FLAG_FORCE_SERIES
= 0x2;
my
$FLAG_FORCE_SHAPE
= 0x4;
my
$FLAG_EMPH
= 0x10;
my
%font_family
= (
cmr
=> {
family
=>
'serif'
},
cmss
=> {
family
=>
'sansserif'
},
cmtt
=> {
family
=>
'typewriter'
},
cmvtt
=> {
family
=>
'typewriter'
},
cmti
=> {
family
=>
'typewriter'
,
shape
=>
'italic'
},
cmfib
=> {
family
=>
'serif'
},
cmfr
=> {
family
=>
'serif'
},
cmdh
=> {
family
=>
'serif'
},
cm
=> {
family
=>
'serif'
},
ptm
=> {
family
=>
'serif'
},
ppl
=> {
family
=>
'serif'
},
pnc
=> {
family
=>
'serif'
},
pbk
=> {
family
=>
'serif'
},
phv
=> {
family
=>
'sansserif'
},
pag
=> {
family
=>
'serif'
},
pcr
=> {
family
=>
'typewriter'
},
pzc
=> {
family
=>
'script'
},
put
=> {
family
=>
'serif'
},
bch
=> {
family
=>
'serif'
},
psy
=> {
family
=>
'symbol'
},
pzd
=> {
family
=>
'dingbats'
},
ccr
=> {
family
=>
'serif'
},
ccy
=> {
family
=>
'symbol'
},
cmbr
=> {
family
=>
'sansserif'
},
cmtl
=> {
family
=>
'typewriter'
},
cmbrs
=> {
family
=>
'symbol'
},
ul9
=> {
family
=>
'typewriter'
},
txr
=> {
family
=>
'serif'
},
txss
=> {
family
=>
'sansserif'
},
txtt
=> {
family
=>
'typewriter'
},
txms
=> {
family
=>
'symbol'
},
txsya
=> {
family
=>
'symbol'
},
txsyb
=> {
family
=>
'symbol'
},
pxr
=> {
family
=>
'serif'
},
pxms
=> {
family
=>
'symbol'
},
pxsya
=> {
family
=>
'symbol'
},
pxsyb
=> {
family
=>
'symbol'
},
futs
=> {
family
=>
'serif'
},
uaq
=> {
family
=>
'serif'
},
ugq
=> {
family
=>
'sansserif'
},
eur
=> {
family
=>
'serif'
},
eus
=> {
family
=>
'script'
},
euf
=> {
family
=>
'fraktur'
},
euex
=> {
family
=>
'symbol'
},
ms
=> {
family
=>
'symbol'
},
ccm
=> {
family
=>
'serif'
,
shape
=>
'italic'
},
cmm
=> {
family
=>
'italic'
,
encoding
=>
'OML'
},
cmex
=> {
family
=>
'symbol'
,
encoding
=>
'OMX'
},
cmsy
=> {
family
=>
'symbol'
,
encoding
=>
'OMS'
},
ccitt
=> {
family
=>
'typewriter'
,
shape
=>
'italic'
},
cmbrm
=> {
family
=>
'sansserif'
,
shape
=>
'italic'
},
futm
=> {
family
=>
'serif'
,
shape
=>
'italic'
},
futmi
=> {
family
=>
'serif'
,
shape
=>
'italic'
},
txmi
=> {
family
=>
'serif'
,
shape
=>
'italic'
},
pxmi
=> {
family
=>
'serif'
,
shape
=>
'italic'
},
bbm
=> {
family
=>
'blackboard'
},
bbold
=> {
family
=>
'blackboard'
},
bbmss
=> {
family
=>
'blackboard'
},
cmmib
=> {
family
=>
'italic'
,
series
=>
'bold'
},
cmbsy
=> {
family
=>
'symbol'
,
series
=>
'bold'
},
msa
=> {
family
=>
'symbol'
,
encoding
=>
'AMSa'
},
msb
=> {
family
=>
'symbol'
,
encoding
=>
'AMSb'
},
msx
=> {
family
=>
'symbol'
,
encoding
=>
'AMSa'
},
msy
=> {
family
=>
'symbol'
,
encoding
=>
'AMSb'
},
);
my
%font_series
= (
''
=> {
series
=>
'medium'
},
m
=> {
series
=>
'medium'
},
mc
=> {
series
=>
'medium'
},
b
=> {
series
=>
'bold'
},
bc
=> {
series
=>
'bold'
},
bx
=> {
series
=>
'bold'
},
sb
=> {
series
=>
'bold'
},
sbc
=> {
series
=>
'bold'
},
bm
=> {
series
=>
'bold'
});
my
%font_shape
= (
''
=> {
shape
=>
'upright'
},
n
=> {
shape
=>
'upright'
},
i
=> {
shape
=>
'italic'
},
it
=> {
shape
=>
'italic'
},
sl
=> {
shape
=>
'slanted'
},
sc
=> {
shape
=>
'smallcaps'
},
csc
=> {
shape
=>
'smallcaps'
});
sub
lookupFontFamily {
my
(
$familycode
) =
@_
;
return
$font_family
{ ToString(
$familycode
) }; }
sub
lookupFontSeries {
my
(
$seriescode
) =
@_
;
return
$font_series
{ ToString(
$seriescode
) }; }
sub
lookupFontShape {
my
(
$shapecode
) =
@_
;
return
$font_shape
{ ToString(
$shapecode
) }; }
my
%font_size
= (
tiny
=> 0.5,
SMALL
=> 0.7,
Small
=> 0.8,
small
=> 0.9,
normal
=> 1.0,
large
=> 1.2,
Large
=> 1.44,
LARGE
=> 1.728,
huge
=> 2.074,
Huge
=> 2.488,
big
=> 1.2,
Big
=> 1.6,
bigg
=> 2.1,
Bigg
=> 2.6,
);
sub
rationalizeFontSize {
my
(
$size
) =
@_
;
return
unless
defined
$size
;
if
(
my
$symbolic
=
$font_size
{
$size
}) {
return
$symbolic
* DEFSIZE(); }
return
$size
; }
sub
relativeFontSize {
my
(
$newsize
,
$oldsize
) =
@_
;
return
int
(0.5 + 100 *
$newsize
/
$oldsize
) .
'%'
; }
my
$FONTREGEXP
=
'('
.
join
(
'|'
,
sort
{ -(
$a
cmp
$b
) }
keys
%font_family
) .
')'
.
'('
.
join
(
'|'
,
sort
{ -(
$a
cmp
$b
) }
keys
%font_series
) .
')'
.
'('
.
join
(
'|'
,
sort
{ -(
$a
cmp
$b
) }
keys
%font_shape
) .
')'
.
'(\d*)'
;
sub
decodeFontname {
my
(
$name
,
$at
,
$scaled
) =
@_
;
if
(
$name
=~ /^
$FONTREGEXP
$/o) {
my
%props
;
my
(
$fam
,
$ser
,
$shp
,
$size
) = ($1, $2, $3, $4);
if
(
my
$ffam
= lookupFontFamily(
$fam
)) {
map
{
$props
{
$_
} =
$$ffam
{
$_
} }
keys
%$ffam
; }
if
(
my
$fser
= lookupFontSeries(
$ser
)) {
map
{
$props
{
$_
} =
$$fser
{
$_
} }
keys
%$fser
; }
if
(
my
$fsh
= lookupFontShape(
$shp
)) {
map
{
$props
{
$_
} =
$$fsh
{
$_
} }
keys
%$fsh
; }
$size
= 1
unless
$size
;
$size
=
$at
if
defined
$at
;
$size
*=
$scaled
if
defined
$scaled
;
$props
{size} =
$size
;
$props
{encoding} =
'OT1'
unless
defined
$props
{encoding};
$props
{at} =
$at
.
"pt"
if
defined
$at
;
return
%props
; }
else
{
return
; } }
sub
lookupTeXFont {
my
(
$fontname
,
$seriescode
,
$shapecode
) =
@_
;
my
%props
;
if
(
my
$ffam
= lookupFontFamily(
$fontname
)) {
map
{
$props
{
$_
} =
$$ffam
{
$_
} }
keys
%$ffam
; }
if
(
my
$fser
= lookupFontSeries(
$seriescode
)) {
map
{
$props
{
$_
} =
$$fser
{
$_
} }
keys
%$fser
; }
if
(
my
$fsh
= lookupFontShape(
$shapecode
)) {
map
{
$props
{
$_
} =
$$fsh
{
$_
} }
keys
%$fsh
; }
return
%props
; }
sub
new {
my
(
$class
,
%options
) =
@_
;
my
$family
=
$options
{family};
my
$series
=
$options
{series};
my
$shape
=
$options
{shape};
my
$size
=
$options
{size};
my
$color
=
$options
{color};
my
$bg
=
$options
{background};
my
$opacity
=
$options
{opacity};
my
$encoding
=
$options
{encoding};
my
$language
=
$options
{language};
my
$mathstyle
=
$options
{mathstyle};
if
(
$options
{forcebold}) {
$series
=
'bold'
;
$options
{forceseries} = 1; }
my
$flags
= 0
| (
$options
{forcefamily} ?
$FLAG_FORCE_FAMILY
: 0)
| (
$options
{forceseries} ?
$FLAG_FORCE_SERIES
: 0)
| (
$options
{forceshape} ?
$FLAG_FORCE_SHAPE
: 0);
return
$class
->new_internal(
$family
,
$series
,
$shape
, rationalizeFontSize(
$size
),
$color
,
$bg
,
$opacity
,
$encoding
,
$language
,
$mathstyle
,
$flags
); }
sub
new_internal {
my
(
$class
,
@components
) =
@_
;
return
bless
[
@components
],
$class
; }
sub
textDefault {
my
(
$self
) =
@_
;
return
$self
->new_internal(
$DEFFAMILY
,
$DEFSERIES
,
$DEFSHAPE
, DEFSIZE(),
$DEFCOLOR
,
$DEFBACKGROUND
,
$DEFOPACITY
,
$DEFENCODING
,
$DEFLANGUAGE
,
undef
, 0); }
sub
mathDefault {
my
(
$self
) =
@_
;
return
$self
->new_internal(
'math'
,
$DEFSERIES
,
'italic'
, DEFSIZE(),
$DEFCOLOR
,
$DEFBACKGROUND
,
$DEFOPACITY
,
undef
,
$DEFLANGUAGE
,
'text'
, 0); }
sub
getFamily {
my
(
$self
) =
@_
;
return
$$self
[0]; }
sub
getSeries {
my
(
$self
) =
@_
;
return
$$self
[1]; }
sub
getShape {
my
(
$self
) =
@_
;
return
$$self
[2]; }
sub
getSize {
my
(
$self
) =
@_
;
return
$$self
[3]; }
sub
getColor {
my
(
$self
) =
@_
;
return
$$self
[4]; }
sub
getBackground {
my
(
$self
) =
@_
;
return
$$self
[5]; }
sub
getOpacity {
my
(
$self
) =
@_
;
return
$$self
[6]; }
sub
getEncoding {
my
(
$self
) =
@_
;
return
$$self
[7]; }
sub
getLanguage {
my
(
$self
) =
@_
;
return
$$self
[8]; }
sub
getMathstyle {
my
(
$self
) =
@_
;
return
$$self
[9]; }
sub
getFlags {
my
(
$self
) =
@_
;
return
$$self
[10]; }
sub
toString {
my
(
$self
) =
@_
;
return
"Font["
.
join
(
','
,
map
{ (
defined
$_
? ToString(
$_
) :
'*'
) } @{
$self
}) .
"]"
; }
sub
stringify {
my
(
$self
) =
@_
;
my
(
$fam
,
$ser
,
$shp
,
$siz
,
$col
,
$bkg
,
$opa
,
$enc
,
$lang
,
$mstyle
,
$flags
) =
@$self
;
$fam
=
'serif'
if
$fam
&& (
$fam
eq
'math'
);
return
'Font['
.
join
(
','
,
map
{ Stringify(
$_
) }
grep
{
$_
}
(isDiff(
$fam
,
$DEFFAMILY
) ? (
$fam
) : ()),
(isDiff(
$ser
,
$DEFSERIES
) ? (
$ser
) : ()),
(isDiff(
$shp
,
$DEFSHAPE
) ? (
$shp
) : ()),
(isDiff(
$siz
, DEFSIZE()) ? (
$siz
) : ()),
(isDiff(
$col
,
$DEFCOLOR
) ? (
$col
) : ()),
(isDiff(
$bkg
,
$DEFBACKGROUND
) ? (
$bkg
) : ()),
(isDiff(
$opa
,
$DEFOPACITY
) ? (
$opa
) : ()),
(
$mstyle
? (
$mstyle
) : ()),
(
$flags
? (
$flags
) : ()),
)
.
']'
; }
sub
equals {
my
(
$self
,
$other
) =
@_
;
return
(
defined
$other
) && ((
ref
$self
) eq (
ref
$other
))
&& (
join
(
'|'
,
map
{ (
defined
$_
?
$_
:
'*'
) }
@$self
)
eq
join
(
'|'
,
map
{ (
defined
$_
?
$_
:
'*'
) }
@$other
)); }
sub
match {
my
(
$self
,
$other
) =
@_
;
return
1
unless
defined
$other
;
return
0
unless
(
ref
$self
) eq (
ref
$other
);
my
@comp
=
@$self
;
my
@ocomp
=
@$other
;
while
(
@comp
) {
my
$c
=
shift
@comp
;
my
$oc
=
shift
@ocomp
;
return
0
if
(
defined
$c
) && (
defined
$oc
) && (
$c
ne
$oc
); }
return
1; }
sub
makeConcrete {
my
(
$self
,
$concrete
) =
@_
;
my
(
$family
,
$series
,
$shape
,
$size
,
$color
,
$bg
,
$opacity
,
$encoding
,
$lang
,
$mstyle
,
$flags
) =
@$self
;
my
(
$ofamily
,
$oseries
,
$oshape
,
$osize
,
$ocolor
,
$obg
,
$oopacity
,
$oencoding
,
$olang
,
$omstyle
,
$oflags
) =
@$concrete
;
return
(
ref
$self
)->new_internal(
$family
||
$ofamily
,
$series
||
$oseries
,
$shape
||
$oshape
,
$size
||
$osize
,
$color
||
$ocolor
,
$bg
||
$obg
, (
defined
$opacity
?
$opacity
:
$oopacity
),
$encoding
||
$oencoding
,
$lang
||
$olang
,
$mstyle
||
$omstyle
,
(
$flags
|| 0) | (
$oflags
|| 0)); }
sub
isDiff {
my
(
$x
,
$y
) =
@_
;
return
(
defined
$x
) && (!(
defined
$y
) || (
$x
ne
$y
)); }
sub
relativeTo {
my
(
$self
,
$other
) =
@_
;
my
(
$fam
,
$ser
,
$shp
,
$siz
,
$col
,
$bkg
,
$opa
,
$enc
,
$lang
,
$mstyle
,
$flags
) =
@$self
;
my
(
$ofam
,
$oser
,
$oshp
,
$osiz
,
$ocol
,
$obkg
,
$oopa
,
$oenc
,
$olang
,
$omstyle
,
$oflags
) =
@$other
;
$fam
=
'serif'
if
$fam
&& (
$fam
eq
'math'
);
$ofam
=
'serif'
if
$ofam
&& (
$ofam
eq
'math'
);
my
@diffs
= (
(isDiff(
$fam
,
$ofam
) ? (
$fam
) : ()),
(isDiff(
$ser
,
$oser
) ? (
$ser
) : ()),
(isDiff(
$shp
,
$oshp
) ? (
$shp
) : ()));
return
(
(
@diffs
?
(
font
=> {
value
=>
join
(
' '
,
@diffs
),
properties
=> { (isDiff(
$fam
,
$ofam
) ? (
family
=>
$fam
) : ()),
(isDiff(
$ser
,
$oser
) ? (
series
=>
$ser
) : ()),
(isDiff(
$shp
,
$oshp
) ? (
shape
=>
$shp
) : ()) } })
: ()),
(isDiff(
$siz
,
$osiz
)
? (
fontsize
=> {
value
=> relativeFontSize(
$siz
,
$osiz
),
properties
=> {
size
=>
$siz
} })
: ()),
(isDiff(
$col
,
$ocol
)
? (
color
=> {
value
=>
$col
,
properties
=> {
color
=>
$col
} })
: ()),
(isDiff(
$bkg
,
$obkg
)
? (
backgroundcolor
=> {
value
=>
$bkg
,
properties
=> {
background
=>
$bkg
} })
: ()),
(isDiff(
$opa
,
$oopa
)
? (
opacity
=> {
value
=>
$opa
,
properties
=> {
opacity
=>
$opa
} })
: ()),
(isDiff(
$lang
,
$olang
)
? (
'xml:lang'
=> {
value
=>
$lang
,
properties
=> {
language
=>
$lang
} })
: ()),
(!
$mstyle
&&
$flags
&& (
$flags
&
$FLAG_EMPH
) && (!
$oflags
|| !(
$oflags
&
$FLAG_EMPH
))
? (
element
=> {
value
=>
'ltx:emph'
}
)
: ()),
); }
sub
distance {
my
(
$self
,
$other
) =
@_
;
my
(
$fam
,
$ser
,
$shp
,
$siz
,
$col
,
$bkg
,
$opa
,
$enc
,
$lang
,
$mstyle
,
$flags
) =
@$self
;
my
(
$ofam
,
$oser
,
$oshp
,
$osiz
,
$ocol
,
$obkg
,
$oopa
,
$oenc
,
$olang
,
$omstyle
,
$oflags
) =
@$other
;
$fam
=
'serif'
if
$fam
&& (
$fam
eq
'math'
);
$ofam
=
'serif'
if
$ofam
&& (
$ofam
eq
'math'
);
return
(isDiff(
$fam
,
$ofam
) ? 1 : 0)
+ (isDiff(
$ser
,
$oser
) ? 1 : 0)
+ (isDiff(
$shp
,
$oshp
) ? 1 : 0)
+ (isDiff(
$siz
,
$osiz
) ? 1 : 0)
+ (isDiff(
$col
,
$ocol
) ? 1 : 0)
+ (isDiff(
$bkg
,
$obkg
) ? 1 : 0)
+ (isDiff(
$opa
,
$oopa
) ? 1 : 0)
+ (isDiff(
$lang
,
$olang
) ? 1 : 0)
+ ((
$flags
&
$FLAG_EMPH
) ^ (
$oflags
&
$FLAG_EMPH
) ? 1 : 0)
; }
our
%FONT_REGEXP_CACHE
= ();
sub
match_font {
my
(
$font1
,
$font2
) =
@_
;
my
$regexp
=
$FONT_REGEXP_CACHE
{
$font1
};
if
(!
$regexp
) {
if
(
$font1
=~ /^Font\[(.*)\]$/) {
my
@comp
=
split
(
','
, $1);
my
$re
=
'^Font\['
.
join
(
','
,
map
{ (
$_
eq
'*'
?
"[^,]+"
:
"\Q$_\E"
) }
@comp
)
.
'\]$'
;
print
STDERR
"\nCreating re for \"$font1\" => $re\n"
;
$regexp
=
$FONT_REGEXP_CACHE
{
$font1
} =
qr/$re/
; } }
return
$font2
=~ /
$regexp
/; }
sub
XXXfont_match_xpaths {
my
(
$font
) =
@_
;
if
(
$font
=~ /^Font\[(.*)\]$/) {
my
@comps
=
split
(
','
, $1);
my
(
$frag
,
@frags
) = ();
for
(
my
$i
= 0 ;
$i
<=
$#comps
;
$i
++) {
my
$comp
=
$comps
[
$i
];
if
(
$comp
eq
'*'
) {
push
(
@frags
,
$frag
)
if
$frag
;
$frag
=
undef
; }
else
{
my
$post
= (
$i
==
$#comps
?
']'
:
','
);
if
(
$frag
) {
$frag
.=
$comp
.
$post
; }
else
{
$frag
= (
$i
== 0 ?
'Font['
:
','
) .
$comp
.
$post
; } } }
push
(
@frags
,
$frag
)
if
$frag
;
return
join
(
' and '
,
'@_font'
,
map
{
"contains(\@_font,'$_')"
}
@frags
); } }
sub
font_match_xpaths {
my
(
$font
) =
@_
;
if
(
$font
=~ /^Font\[(.*)\]$/) {
my
(
$family
,
$series
,
$shape
,
$size
,
$color
,
$bg
,
$opacity
,
$encoding
,
$language
,
$mstyle
,
$force
) =
split
(
','
, $1);
my
@frags
= ();
push
(
@frags
,
'['
.
$family
.
','
)
if
(
$family
ne
'*'
);
push
(
@frags
,
','
.
$series
.
','
)
if
(
$series
ne
'*'
);
push
(
@frags
,
','
.
$shape
.
','
)
if
(
$shape
ne
'*'
);
return
join
(
' and '
,
'@_font'
,
map
{
"contains(\@_font,'$_')"
}
@frags
); } }
our
%mathstylesize
= (
display
=> 1,
text
=> 1,
script
=> 0.7,
scriptscript
=> 0.5);
sub
computeStringSize {
my
(
$self
,
$string
) =
@_
;
my
$size
= (
$self
->getSize || DEFSIZE() || 10);
my
$l
= (
defined
$string
?
length
(
$string
) : 0);
my
$u
=
$size
* 65535;
return
(Dimension(0.75 *
$u
*
$l
), Dimension(0.7 *
$u
), Dimension(0.2 *
$u
)); }
sub
getNominalSize {
my
(
$self
) =
@_
;
my
$size
= (
$self
->getSize || DEFSIZE() || 10);
my
$u
=
$size
* 65535;
return
(Dimension(0.75 *
$u
), Dimension(0.7 *
$u
), Dimension(0.2 *
$u
)); }
sub
computeBoxesSize {
my
(
$self
,
$boxes
,
%options
) =
@_
;
my
$font
= (
ref
$self
?
$self
:
$STATE
->lookupValue(
'font'
));
my
$fillwidth
=
$options
{width};
if
((!
defined
$fillwidth
) && (
$fillwidth
=
$STATE
->lookupDefinition(T_CS(
'\textwidth'
)))) {
$fillwidth
=
$fillwidth
->valueOf; }
my
$maxwidth
=
$fillwidth
&&
$fillwidth
->valueOf;
my
@lines
= ();
my
(
$wd
,
$ht
,
$dp
) = (0, 0, 0);
my
$vattach
=
$options
{vattach} ||
'baseline'
;
foreach
my
$box
(
@$boxes
) {
next
unless
defined
$box
;
next
if
ref
$box
&& !
$box
->can(
'getSize'
);
my
(
$w
,
$h
,
$d
) = (
ref
$box
?
$box
->getSize(
%options
) :
$font
->computeStringSize(
$box
));
if
(
ref
$w
) {
$wd
+=
$w
->valueOf; }
else
{
Warn(
'expected'
,
'Dimension'
,
undef
,
"Width of "
. Stringify(
$box
) .
" yielded a non-dimension: "
. Stringify(
$w
)); }
if
(
ref
$h
) {
$ht
= max(
$ht
,
$h
->valueOf); }
else
{
Warn(
'expected'
,
'Dimension'
,
undef
,
"Height of "
. Stringify(
$box
) .
" yielded a non-dimension: "
. Stringify(
$h
)); }
if
(
ref
$d
) {
$dp
= max(
$dp
,
$d
->valueOf); }
else
{
Warn(
'expected'
,
'Dimension'
,
undef
,
"Depth of "
. Stringify(
$box
) .
" yielded a non-dimension: "
. Stringify(
$d
)); }
if
(((
$options
{layout} ||
''
) eq
'vertical'
)
) {
push
(
@lines
, [
$wd
,
$ht
,
$dp
]);
$wd
=
$ht
=
$dp
= 0; }
elsif
((
defined
$maxwidth
) && (
$wd
>=
$maxwidth
)) {
push
(
@lines
, [
$wd
,
$ht
,
$dp
]);
$wd
=
$ht
=
$dp
= 0;
}
}
if
(
$wd
) {
push
(
@lines
, [
$wd
,
$ht
,
$dp
]); }
my
$nlines
=
scalar
(
@lines
);
if
(
$nlines
== 0) {
$wd
=
$ht
=
$dp
= 0; }
else
{
$wd
= max(
map
{
$$_
[0] }
@lines
);
$ht
= sum(
map
{
$$_
[1] }
@lines
);
$dp
= sum(
map
{
$$_
[2] }
@lines
);
if
(
$vattach
eq
'top'
) {
my
(
$w
,
$h
,
$d
) =
$font
->getNominalSize;
$h
=
$h
->valueOf;
$dp
=
$ht
+
$dp
-
$h
;
$ht
=
$h
; }
elsif
(
$vattach
eq
'bottom'
) {
$ht
=
$ht
+
$dp
;
$dp
= 0; }
elsif
(
$vattach
eq
'middle'
) {
my
(
$w
,
$h
,
$d
) =
$font
->getNominalSize;
$h
=
$h
->valueOf;
my
$c
= (
$ht
+
$dp
) / 2;
$ht
=
$c
+
$h
/ 2;
$dp
=
$c
-
$h
/ 2; }
else
{
my
$h
=
$lines
[0][1];
$dp
=
$ht
+
$dp
-
$h
;
$ht
=
$h
; } }
return
(Dimension(
$wd
), Dimension(
$ht
), Dimension(
$dp
)); }
sub
isSticky {
my
(
$self
) =
@_
;
return
$$self
[0] && (
$$self
[0] =~ /^(?:serif|sansserif|typewriter)$/); }
our
%scriptstylemap
= (
display
=>
'script'
,
text
=>
'script'
,
script
=>
'scriptscript'
,
scriptscript
=>
'scriptscript'
);
our
%fracstylemap
= (
display
=>
'text'
,
text
=>
'script'
,
script
=>
'scriptscript'
,
scriptscript
=>
'scriptscript'
);
our
%stylesize
= (
display
=> 10,
text
=> 10,
script
=> 7,
scriptscript
=> 5);
sub
merge {
my
(
$self
,
%options
) =
@_
;
foreach
my
$k
(
keys
%options
) {
$options
{
$k
} = &{
$options
{
$k
} }()
if
ref
$options
{
$k
} eq
'CODE'
; }
my
$family
=
$options
{family};
my
$series
=
$options
{series};
my
$shape
=
$options
{shape};
my
$size
= rationalizeFontSize(
$options
{size});
my
$color
=
$options
{color};
my
$bg
=
$options
{background};
my
$opacity
=
$options
{opacity};
my
$encoding
=
$options
{encoding};
my
$language
=
$options
{language};
my
$mathstyle
=
$options
{mathstyle};
if
(
$options
{forcebold}) {
$series
=
'bold'
;
$options
{forceseries} = 1; }
my
$flags
= 0
| (
$options
{forcefamily} ?
$FLAG_FORCE_FAMILY
: 0)
| (
$options
{forceseries} ?
$FLAG_FORCE_SERIES
: 0)
| (
$options
{forceshape} ?
$FLAG_FORCE_SHAPE
: 0);
my
$oflags
=
$$self
[10];
$family
=
$$self
[0]
if
(!
defined
$family
) || (
$oflags
&
$FLAG_FORCE_FAMILY
);
$series
=
$$self
[1]
if
(!
defined
$series
) || (
$oflags
&
$FLAG_FORCE_SERIES
);
$shape
=
$$self
[2]
if
(!
defined
$shape
) || (
$oflags
&
$FLAG_FORCE_SHAPE
);
$size
=
$$self
[3]
if
(!
defined
$size
);
$color
=
$$self
[4]
if
(!
defined
$color
);
$bg
=
$$self
[5]
if
(!
defined
$bg
);
$opacity
=
$$self
[6]
if
(!
defined
$opacity
);
$encoding
=
$$self
[7]
if
(!
defined
$encoding
);
$language
=
$$self
[8]
if
(!
defined
$language
);
$mathstyle
=
$$self
[9]
if
(!
defined
$mathstyle
);
$flags
= (
$$self
[10] || 0) |
$flags
;
if
(
my
$scale
=
$options
{scale}) {
$size
=
$scale
*
$size
; }
my
$stylescale
= (
$$self
[3] ?
$$self
[3] /
$stylesize
{
$$self
[9] ||
'display'
} : 1);
if
(
$options
{size}) { }
elsif
(
$options
{mathstyle}) {
$size
=
$stylescale
*
$stylesize
{
$mathstyle
}; }
elsif
(
$options
{scripted}) {
$mathstyle
=
$scriptstylemap
{
$mathstyle
||
'display'
};
$size
=
$stylescale
*
$stylesize
{
$mathstyle
||
'display'
}; }
elsif
(
$options
{fraction}) {
$mathstyle
=
$fracstylemap
{
$mathstyle
||
'display'
};
$size
=
$stylescale
*
$stylesize
{
$mathstyle
||
'display'
}; }
if
(
$options
{emph}) {
$shape
= (
$shape
eq
'italic'
?
'upright'
:
'italic'
);
$flags
|=
$FLAG_EMPH
; }
$flags
&= ~
$FLAG_EMPH
if
$mathstyle
;
my
$newfont
= (
ref
$self
)->new_internal(
$family
,
$series
,
$shape
,
$size
,
$color
,
$bg
,
$opacity
,
$encoding
,
$language
,
$mathstyle
,
$flags
);
if
(
my
$specialize
=
$options
{specialize}) {
$newfont
=
$newfont
->specialize(
$specialize
); }
return
$newfont
; }
sub
specialize {
my
(
$self
,
$string
) =
@_
;
return
$self
if
!(
defined
$string
) ||
ref
$string
;
my
(
$family
,
$series
,
$shape
,
$size
,
$color
,
$bg
,
$opacity
,
$encoding
,
$language
,
$mathstyle
,
$flags
) =
@$self
;
my
$deffamily
= (
$flags
&
$FLAG_FORCE_FAMILY
?
$family
||
$DEFFAMILY
:
$DEFFAMILY
);
my
$defseries
= (
$flags
&
$FLAG_FORCE_SERIES
?
$series
||
$DEFSERIES
:
$DEFSERIES
);
my
$defshape
= (
$flags
&
$FLAG_FORCE_SHAPE
?
$shape
||
$DEFSHAPE
:
$DEFSHAPE
);
if
((
$string
=~ /^\p{Latin}$/) && (
$string
=~ /^\p{L}$/)) {
$shape
=
'italic'
if
!
$shape
&& !
$family
; }
elsif
(
$string
=~ /^\p{Greek}$/) {
if
(
$string
=~ /^\p{Lu}$/) {
if
(!
$family
|| (
$family
eq
'math'
)) {
$family
=
$deffamily
;
$shape
=
$defshape
if
$shape
&& (
$shape
ne
$DEFSHAPE
); } }
else
{
$family
=
$deffamily
if
!
$family
|| (
$family
ne
$DEFFAMILY
);
$shape
=
'italic'
if
!
$shape
|| !(
$flags
&
$FLAG_FORCE_SHAPE
);
if
(
$series
&& (
$series
ne
$DEFSERIES
)) {
$series
=
$defseries
; }
} }
elsif
(
$string
=~ /^\p{N}$/) {
if
(!
$family
|| (
$family
eq
'math'
)) {
$family
=
$deffamily
;
$shape
=
$defshape
; } }
else
{
$family
=
$deffamily
;
$shape
=
$defshape
;
if
(
$series
&& (
$series
ne
$DEFSERIES
)) {
$series
=
$defseries
; } }
return
(
ref
$self
)->new_internal(
$family
,
$series
,
$shape
,
$size
,
$color
,
$bg
,
$opacity
,
$encoding
,
$language
,
$mathstyle
,
$flags
); }
our
%mathstylestep
= (
display
=> {
display
=> 0,
text
=> 1,
script
=> 2,
scriptscript
=> 3 },
text
=> {
display
=> -1,
text
=> 0,
script
=> 1,
scriptscript
=> 2 },
script
=> {
display
=> -2,
text
=> -1,
script
=> 0,
scriptscript
=> 1 },
scriptscript
=> {
display
=> -3,
text
=> -2,
script
=> -1,
scriptscript
=> 0 });
our
%stepmathstyle
= (
display
=> {
-3
=>
'display'
,
-2
=>
'display'
,
-1
=>
'display'
,
0
=>
'display'
,
1
=>
'text'
,
2
=>
'script'
,
3
=>
'scriptscript'
},
text
=> {
-3
=>
'display'
,
-2
=>
'display'
,
-1
=>
'display'
,
0
=>
'text'
,
1
=>
'script'
,
2
=>
'scriptscript'
,
3
=>
'scriptscript'
},
script
=> {
-3
=>
'display'
,
-2
=>
'display'
,
-1
=>
'text'
,
0
=>
'script'
,
1
=>
'scriptscript'
,
2
=>
'scriptscript'
,
3
=>
'scriptscript'
},
scriptscript
=> {
-3
=>
'display'
,
-2
=>
'text'
,
-1
=>
'script'
,
0
=>
'scriptscript'
,
1
=>
'scriptscript'
,
2
=>
'scriptscript'
,
3
=>
'scriptscript'
});
sub
purestyleChanges {
my
(
$self
,
$other
) =
@_
;
my
$mathstyle
=
$self
->getMathstyle;
my
$othermathstyle
=
$other
->getMathstyle;
my
$othercolor
=
$other
->getColor;
return
(
scale
=>
$other
->getSize /
$self
->getSize,
(isDiff(
$othercolor
,
$DEFCOLOR
) ? (
color
=>
$othercolor
) : ()),
background
=>
$other
->getBackground,
opacity
=>
$other
->getOpacity,
(
$mathstyle
&&
$othermathstyle
? (
mathstylestep
=>
$mathstylestep
{
$mathstyle
}{
$othermathstyle
})
: ()),
); }
sub
mergePurestyle {
my
(
$self
,
%stylechanges
) =
@_
;
my
$new
=
$self
->new_internal(
@$self
);
$$new
[3] =
$$self
[3] *
$stylechanges
{scale}
if
$stylechanges
{scale};
$$new
[4] =
$stylechanges
{color}
if
$stylechanges
{color};
$$new
[5] =
$stylechanges
{background}
if
$stylechanges
{background};
$$new
[6] =
$stylechanges
{opacity}
if
$stylechanges
{opacity};
$$new
[9] =
$stepmathstyle
{
$$self
[9] }{
$stylechanges
{mathstylestep} }
if
$stylechanges
{mathstylestep};
return
new; }
1;