summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVincent Hennebert <vhennebert@apache.org>2013-07-29 21:45:20 +0000
committerVincent Hennebert <vhennebert@apache.org>2013-07-29 21:45:20 +0000
commitf8e822efe1de8bd8192dbb8ff035b9a79f876614 (patch)
tree8d38f873dd101e6ed0c2ba3b4dfa29e1eb0fdc60
parentc0b99ad44d0e1409008886e2f687c46f4ac05d9d (diff)
downloadxmlgraphics-fop-f8e822efe1de8bd8192dbb8ff035b9a79f876614.tar.gz
xmlgraphics-fop-f8e822efe1de8bd8192dbb8ff035b9a79f876614.zip
Directly use FOP fonts to lay out SVG images for PDF, PS and AFP outputs.
The metrics are now taken from FOP configured fonts and no longer from AWT equivalents. That avoids discrepancies in case AWT and FOP use slightly different fonts, or if the font is not installed on the system. That actually also avoids having to install the font on the system. FOP is also used for the primary layout of text (prior to SVG-specific transforms like translation or rotation) for consistency between SVG and XSL-FO. This is a joint work from Peter Hancock and myself. git-svn-id: https://svn.apache.org/repos/asf/xmlgraphics/fop/branches/Temp_FopFontsForSVG@1508208 13f79535-47bb-0310-9956-ffa450edef68
-rw-r--r--build.xml7
-rw-r--r--findbugs-exclude.xml30
-rw-r--r--lib/batik-all-1.7.jarbin3318083 -> 0 bytes
-rw-r--r--lib/batik-all-trunk-patch1041.jarbin0 -> 3313165 bytes
-rw-r--r--src/codegen/fonts/Courier.xml501
-rw-r--r--src/codegen/fonts/CourierBold.xml501
-rw-r--r--src/codegen/fonts/CourierBoldOblique.xml499
-rw-r--r--src/codegen/fonts/CourierOblique.xml501
-rw-r--r--src/codegen/fonts/Helvetica.xml465
-rw-r--r--src/codegen/fonts/HelveticaBold.xml465
-rw-r--r--src/codegen/fonts/HelveticaBoldOblique.xml465
-rw-r--r--src/codegen/fonts/HelveticaOblique.xml467
-rw-r--r--src/codegen/fonts/Symbol.xml387
-rw-r--r--src/codegen/fonts/TimesBold.xml465
-rw-r--r--src/codegen/fonts/TimesBoldItalic.xml465
-rw-r--r--src/codegen/fonts/TimesItalic.xml467
-rw-r--r--src/codegen/fonts/TimesRoman.xml465
-rw-r--r--src/codegen/fonts/ZapfDingbats.xml412
-rw-r--r--src/codegen/fonts/font-file.xsl26
-rw-r--r--src/java/org/apache/fop/afp/AFPEventProducer.java8
-rw-r--r--src/java/org/apache/fop/afp/fonts/AFPFont.java32
-rw-r--r--src/java/org/apache/fop/afp/fonts/AbstractOutlineFont.java60
-rw-r--r--src/java/org/apache/fop/afp/fonts/CharacterSet.java72
-rw-r--r--src/java/org/apache/fop/afp/fonts/CharacterSetBuilder.java95
-rw-r--r--src/java/org/apache/fop/afp/fonts/CharacterSetOrientation.java124
-rw-r--r--src/java/org/apache/fop/afp/fonts/DoubleByteFont.java34
-rw-r--r--src/java/org/apache/fop/afp/fonts/FopCharacterSet.java50
-rw-r--r--src/java/org/apache/fop/afp/fonts/IntegerKeyStore.java71
-rw-r--r--src/java/org/apache/fop/afp/fonts/OutlineFont.java16
-rw-r--r--src/java/org/apache/fop/afp/fonts/RasterFont.java88
-rw-r--r--src/java/org/apache/fop/afp/goca/GraphicsCharacterString.java6
-rw-r--r--src/java/org/apache/fop/afp/svg/AFPBridgeContext.java42
-rw-r--r--src/java/org/apache/fop/afp/svg/AFPFontFamilyResolver.java84
-rw-r--r--src/java/org/apache/fop/afp/svg/AFPTextHandler.java31
-rw-r--r--src/java/org/apache/fop/afp/svg/AFPTextPainter.java21
-rw-r--r--src/java/org/apache/fop/fo/FOText.java94
-rw-r--r--src/java/org/apache/fop/fonts/Base14Font.java11
-rw-r--r--src/java/org/apache/fop/fonts/CIDFont.java1
-rw-r--r--src/java/org/apache/fop/fonts/CustomFont.java47
-rw-r--r--src/java/org/apache/fop/fonts/FontInfo.java37
-rw-r--r--src/java/org/apache/fop/fonts/FontMetrics.java44
-rw-r--r--src/java/org/apache/fop/fonts/GlyphMapping.java327
-rw-r--r--src/java/org/apache/fop/fonts/LazyFont.java26
-rw-r--r--src/java/org/apache/fop/fonts/MultiByteFont.java18
-rw-r--r--src/java/org/apache/fop/fonts/SingleByteFont.java40
-rw-r--r--src/java/org/apache/fop/fonts/TextFragment.java31
-rw-r--r--src/java/org/apache/fop/fonts/truetype/TTFFile.java38
-rw-r--r--src/java/org/apache/fop/fonts/truetype/TTFFontLoader.java20
-rw-r--r--src/java/org/apache/fop/fonts/type1/AFMCharMetrics.java8
-rw-r--r--src/java/org/apache/fop/fonts/type1/Type1FontLoader.java17
-rw-r--r--src/java/org/apache/fop/image/loader/batik/ImageConverterSVG2G2D.java7
-rw-r--r--src/java/org/apache/fop/image/loader/batik/PreloaderSVG.java3
-rw-r--r--src/java/org/apache/fop/layoutmgr/inline/TextLayoutManager.java719
-rw-r--r--src/java/org/apache/fop/render/AbstractGenericSVGHandler.java4
-rw-r--r--src/java/org/apache/fop/render/afp/AFPSVGHandler.java15
-rw-r--r--src/java/org/apache/fop/render/java2d/CustomFontMetricsMapper.java21
-rw-r--r--src/java/org/apache/fop/render/java2d/Java2DFontMetrics.java20
-rw-r--r--src/java/org/apache/fop/render/java2d/Java2DSVGHandler.java5
-rw-r--r--src/java/org/apache/fop/render/java2d/SystemFontMetricsMapper.java21
-rw-r--r--src/java/org/apache/fop/render/pdf/PDFImageHandlerSVG.java4
-rw-r--r--src/java/org/apache/fop/render/ps/AbstractPSTranscoder.java3
-rw-r--r--src/java/org/apache/fop/render/ps/PSImageHandlerSVG.java5
-rw-r--r--src/java/org/apache/fop/render/ps/PSSVGHandler.java3
-rw-r--r--src/java/org/apache/fop/render/ps/PSTextPainter.java361
-rw-r--r--src/java/org/apache/fop/svg/ACIUtils.java68
-rw-r--r--src/java/org/apache/fop/svg/AbstractFOPTextPainter.java331
-rw-r--r--src/java/org/apache/fop/svg/AbstractFOPTranscoder.java22
-rw-r--r--src/java/org/apache/fop/svg/NativeTextPainter.java147
-rw-r--r--src/java/org/apache/fop/svg/PDFTextPainter.java229
-rw-r--r--src/java/org/apache/fop/svg/PDFTextUtil.java41
-rw-r--r--src/java/org/apache/fop/svg/PDFTranscoder.java6
-rw-r--r--src/java/org/apache/fop/svg/SVGUserAgent.java11
-rw-r--r--src/java/org/apache/fop/svg/SimpleSVGUserAgent.java12
-rw-r--r--src/java/org/apache/fop/svg/font/AggregatingFontFamilyResolver.java70
-rw-r--r--src/java/org/apache/fop/svg/font/FOPFontFamilyResolver.java31
-rw-r--r--src/java/org/apache/fop/svg/font/FOPFontFamilyResolverImpl.java67
-rw-r--r--src/java/org/apache/fop/svg/font/FOPGVTFont.java159
-rw-r--r--src/java/org/apache/fop/svg/font/FOPGVTFontFamily.java73
-rw-r--r--src/java/org/apache/fop/svg/font/FOPGVTGlyphVector.java339
-rw-r--r--src/java/org/apache/fop/svg/font/FilteringFontFamilyResolver.java47
-rw-r--r--test/java/org/apache/fop/afp/fonts/IntegerKeyStoreTestCase.java48
-rw-r--r--test/java/org/apache/fop/fonts/DejaVuLGCSerifTestCase.java17
-rw-r--r--test/java/org/apache/fop/fonts/FontEventProcessingTestCase.java9
-rw-r--r--test/java/org/apache/fop/fonts/svg-fonts.fo37
-rw-r--r--test/java/org/apache/fop/fonts/truetype/TTFFileTestCase.java22
-rw-r--r--test/java/org/apache/fop/fonts/type1/AFMParserTestCase.java15
-rw-r--r--test/java/org/apache/fop/fonts/type1/underline.afm4
-rw-r--r--test/java/org/apache/fop/svg/NativeTextPainterTest.java78
-rw-r--r--test/java/org/apache/fop/svg/OperatorValidator.java108
-rw-r--r--test/java/org/apache/fop/svg/PDFTextPainterTestCase.java149
-rw-r--r--test/java/org/apache/fop/svg/PSTextPainterTestCase.java77
-rw-r--r--test/java/org/apache/fop/svg/baseline-shift.svg9
-rw-r--r--test/java/org/apache/fop/svg/dx-dy.svg8
-rw-r--r--test/java/org/apache/fop/svg/font/BasicGlyphVectorTestCase.java193
-rw-r--r--test/java/org/apache/fop/svg/font/FOPFontFamilyResolverTestCase.java127
-rw-r--r--test/java/org/apache/fop/svg/font/FOPGVTFontTestCase.java72
-rw-r--r--test/java/org/apache/fop/svg/font/FOPGVTGlyphVectorTest.java27
-rw-r--r--test/java/org/apache/fop/svg/font/FontInfoBuilder.java102
-rw-r--r--test/java/org/apache/fop/svg/font/GlyphLayoutTestCase.java91
-rw-r--r--test/java/org/apache/fop/svg/glyph-orientation.svg10
-rw-r--r--test/java/org/apache/fop/svg/rotated-glyph.svg5
-rw-r--r--test/java/org/apache/fop/svg/spacing.svg21
102 files changed, 7165 insertions, 5089 deletions
diff --git a/build.xml b/build.xml
index fee1ef03c..e1770ec55 100644
--- a/build.xml
+++ b/build.xml
@@ -565,11 +565,13 @@ list of possible build targets.
<include name="org/apache/fop/apps/io/**"/>
<include name="org/apache/fop/complexscripts/fonts/*.class"/>
<include name="org/apache/fop/complexscripts/util/GlyphTester.class"/>
+ <include name="org/apache/fop/events/EventProducer.class"/>
<include name="org/apache/fop/fo/Constants.class"/>
<include name="org/apache/fop/fo/FOTreeBuilder.class"/>
<include name="org/apache/fop/area/AreaTreeControl*"/>
<include name="org/apache/fop/svg/**"/>
<include name="org/apache/fop/fonts/**"/>
+ <include name="org/apache/fop/traits/MinOptMax.class"/>
<include name="org/apache/fop/image/loader/batik/BatikImageFlavors*.class"/>
<include name="org/apache/fop/util/CMYKColorSpace*.class"/>
<include name="org/apache/fop/util/Color*.class"/>
@@ -579,6 +581,7 @@ list of possible build targets.
<include name="org/apache/fop/util/Finalizable.class"/>
<include name="org/apache/fop/util/CharUtilities.class"/>
<include name="org/apache/fop/util/DecimalFormatCache*.class"/>
+ <include name="org/apache/fop/util/ImageObject.class"/>
</patternset>
<!-- PDF transcoder -->
<patternset>
@@ -647,7 +650,8 @@ list of possible build targets.
<include name="org/apache/xmlgraphics/java2d/**"/>
<include name="org/apache/xmlgraphics/ps/**"/>
<include name="org/apache/xmlgraphics/fonts/**"/>
- <include name="org/apache/xmlgraphics/util/io/**"/>
+ <include name="org/apache/xmlgraphics/util/**"/>
+ <include name="org/apache/xmlgraphics/image/loader/**"/>
</patternset>
<fileset refid="transcoder-lib-files"/>
</unjar>
@@ -709,6 +713,7 @@ list of possible build targets.
<include name="**/*.txt"/>
<include name="**/*.afm"/>
<include name="**/*.fo"/>
+ <include name="**/*.svg"/>
</fileset>
</copy>
</target>
diff --git a/findbugs-exclude.xml b/findbugs-exclude.xml
index 31edec268..0af5e4012 100644
--- a/findbugs-exclude.xml
+++ b/findbugs-exclude.xml
@@ -1,6 +1,36 @@
<?xml version="1.0" encoding="utf-8"?>
<FindBugsFilter>
<Match>
+ <Class name="org.apache.fop.layoutmgr.inline.TextLayoutManager$TextAreaBuilder"/>
+ <Method name="getMappingBidiLevels"/>
+ <Bug pattern="PZLA_PREFER_ZERO_LENGTH_ARRAYS"/>
+ </Match>
+ <Match>
+ <Class name="org.apache.fop.svg.PDFTextPainter"/>
+ <Field name="pdf"/>
+ <Bug pattern="UWF_FIELD_NOT_INITIALIZED_IN_CONSTRUCTOR"/>
+ </Match>
+ <Match>
+ <Class name="org.apache.fop.render.ps.PSTextPainter"/>
+ <Or>
+ <Field name="gen"/>
+ <Field name="ps"/>
+ <Field name="psRun"/>
+ <Field name="textUtil"/>
+ </Or>
+ <Bug pattern="UWF_FIELD_NOT_INITIALIZED_IN_CONSTRUCTOR"/>
+ </Match>
+ <Match>
+ <Class name="org.apache.fop.fonts.GlyphMapping"/>
+ <Method name="&lt;init&gt;"/>
+ <Bug pattern="EI_EXPOSE_REP2"/>
+ </Match>
+ <Match>
+ <Class name="org.apache.fop.fonts.MultiByteFont"/>
+ <Method name="setBBoxArray"/>
+ <Bug pattern="EI_EXPOSE_REP2"/>
+ </Match>
+ <Match>
<Class name="org.apache.fop.fonts.truetype.TTFFile$1"/>
<Bug pattern="SIC_INNER_SHOULD_BE_STATIC_ANON"/>
</Match>
diff --git a/lib/batik-all-1.7.jar b/lib/batik-all-1.7.jar
deleted file mode 100644
index 589328581..000000000
--- a/lib/batik-all-1.7.jar
+++ /dev/null
Binary files differ
diff --git a/lib/batik-all-trunk-patch1041.jar b/lib/batik-all-trunk-patch1041.jar
new file mode 100644
index 000000000..cf69a308d
--- /dev/null
+++ b/lib/batik-all-trunk-patch1041.jar
Binary files differ
diff --git a/src/codegen/fonts/Courier.xml b/src/codegen/fonts/Courier.xml
index be977e22e..075699427 100644
--- a/src/codegen/fonts/Courier.xml
+++ b/src/codegen/fonts/Courier.xml
@@ -22,275 +22,246 @@
<family-name>Courier</family-name>
<class-name>Courier</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>562</cap-height>
<x-height>426</x-height>
<ascender>629</ascender>
<descender>-157</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="600"/>
- <char name="AE" width="600"/>
- <char name="Aacute" width="600"/>
- <char name="Acircumflex" width="600"/>
- <char name="Adieresis" width="600"/>
- <char name="Agrave" width="600"/>
- <char name="Aring" width="600"/>
- <char name="Atilde" width="600"/>
- <char name="B" width="600"/>
- <char name="C" width="600"/>
- <char name="Ccedilla" width="600"/>
- <char name="D" width="600"/>
- <char name="E" width="600"/>
- <char name="Eacute" width="600"/>
- <char name="Ecircumflex" width="600"/>
- <char name="Edieresis" width="600"/>
- <char name="Egrave" width="600"/>
- <char name="Eth" width="600"/>
- <char name="Euro" width="600"/>
- <char name="F" width="600"/>
- <char name="G" width="600"/>
- <char name="Gcaron" width="600"/>
- <char name="H" width="600"/>
- <char name="I" width="600"/>
- <char name="IJ" width="600"/>
- <char name="Iacute" width="600"/>
- <char name="Icircumflex" width="600"/>
- <char name="Idieresis" width="600"/>
- <char name="Idot" width="600"/>
- <char name="Igrave" width="600"/>
- <char name="J" width="600"/>
- <char name="K" width="600"/>
- <char name="L" width="600"/>
- <char name="LL" width="600"/>
- <char name="Lslash" width="600"/>
- <char name="M" width="600"/>
- <char name="N" width="600"/>
- <char name="Ntilde" width="600"/>
- <char name="O" width="600"/>
- <char name="OE" width="600"/>
- <char name="Oacute" width="600"/>
- <char name="Ocircumflex" width="600"/>
- <char name="Odieresis" width="600"/>
- <char name="Ograve" width="600"/>
- <char name="Oslash" width="600"/>
- <char name="Otilde" width="600"/>
- <char name="P" width="600"/>
- <char name="Q" width="600"/>
- <char name="R" width="600"/>
- <char name="S" width="600"/>
- <char name="Scaron" width="600"/>
- <char name="Scedilla" width="600"/>
- <char name="T" width="600"/>
- <char name="Thorn" width="600"/>
- <char name="U" width="600"/>
- <char name="Uacute" width="600"/>
- <char name="Ucircumflex" width="600"/>
- <char name="Udieresis" width="600"/>
- <char name="Ugrave" width="600"/>
- <char name="V" width="600"/>
- <char name="W" width="600"/>
- <char name="X" width="600"/>
- <char name="Y" width="600"/>
- <char name="Yacute" width="600"/>
- <char name="Ydieresis" width="600"/>
- <char name="Z" width="600"/>
- <char name="Zcaron" width="600"/>
- <char name="a" width="600"/>
- <char name="aacute" width="600"/>
- <char name="acircumflex" width="600"/>
- <char name="acute" width="600"/>
- <char name="adieresis" width="600"/>
- <char name="ae" width="600"/>
- <char name="agrave" width="600"/>
- <char name="ampersand" width="600"/>
- <char name="aring" width="600"/>
- <char name="arrowboth" width="600"/>
- <char name="arrowdown" width="600"/>
- <char name="arrowleft" width="600"/>
- <char name="arrowright" width="600"/>
- <char name="arrowup" width="600"/>
- <char name="asciicircum" width="600"/>
- <char name="asciitilde" width="600"/>
- <char name="asterisk" width="600"/>
- <char name="at" width="600"/>
- <char name="atilde" width="600"/>
- <char name="b" width="600"/>
- <char name="backslash" width="600"/>
- <char name="bar" width="600"/>
- <char name="braceleft" width="600"/>
- <char name="braceright" width="600"/>
- <char name="bracketleft" width="600"/>
- <char name="bracketright" width="600"/>
- <char name="breve" width="600"/>
- <char name="brokenbar" width="600"/>
- <char name="bullet" width="600"/>
- <char name="c" width="600"/>
- <char name="caron" width="600"/>
- <char name="ccedilla" width="600"/>
- <char name="cedilla" width="600"/>
- <char name="cent" width="600"/>
- <char name="center" width="600"/>
- <char name="circumflex" width="600"/>
- <char name="colon" width="600"/>
- <char name="comma" width="600"/>
- <char name="copyright" width="600"/>
- <char name="currency" width="600"/>
- <char name="d" width="600"/>
- <char name="dagger" width="600"/>
- <char name="daggerdbl" width="600"/>
- <char name="dectab" width="600"/>
- <char name="degree" width="600"/>
- <char name="dieresis" width="600"/>
- <char name="divide" width="600"/>
- <char name="dollar" width="600"/>
- <char name="dotaccent" width="600"/>
- <char name="dotlessi" width="600"/>
- <char name="down" width="600"/>
- <char name="e" width="600"/>
- <char name="eacute" width="600"/>
- <char name="ecircumflex" width="600"/>
- <char name="edieresis" width="600"/>
- <char name="egrave" width="600"/>
- <char name="eight" width="600"/>
- <char name="ellipsis" width="600"/>
- <char name="emdash" width="600"/>
- <char name="endash" width="600"/>
- <char name="equal" width="600"/>
- <char name="eth" width="600"/>
- <char name="exclam" width="600"/>
- <char name="exclamdown" width="600"/>
- <char name="f" width="600"/>
- <char name="fi" width="600"/>
- <char name="five" width="600"/>
- <char name="fl" width="600"/>
- <char name="florin" width="600"/>
- <char name="format" width="600"/>
- <char name="four" width="600"/>
- <char name="fraction" width="600"/>
- <char name="g" width="600"/>
- <char name="gcaron" width="600"/>
- <char name="germandbls" width="600"/>
- <char name="grave" width="600"/>
- <char name="graybox" width="600"/>
- <char name="greater" width="600"/>
- <char name="guillemotleft" width="600"/>
- <char name="guillemotright" width="600"/>
- <char name="guilsinglleft" width="600"/>
- <char name="guilsinglright" width="600"/>
- <char name="h" width="600"/>
- <char name="hungarumlaut" width="600"/>
- <char name="hyphen" width="600"/>
- <char name="i" width="600"/>
- <char name="iacute" width="600"/>
- <char name="icircumflex" width="600"/>
- <char name="idieresis" width="600"/>
- <char name="igrave" width="600"/>
- <char name="ij" width="600"/>
- <char name="indent" width="600"/>
- <char name="j" width="600"/>
- <char name="k" width="600"/>
- <char name="l" width="600"/>
- <char name="largebullet" width="600"/>
- <char name="left" width="600"/>
- <char name="less" width="600"/>
- <char name="lira" width="600"/>
- <char name="ll" width="600"/>
- <char name="logicalnot" width="600"/>
- <char name="lslash" width="600"/>
- <char name="m" width="600"/>
- <char name="macron" width="600"/>
- <char name="merge" width="600"/>
- <char name="minus" width="600"/>
- <char name="mu" width="600"/>
- <char name="multiply" width="600"/>
- <char name="n" width="600"/>
- <char name="nine" width="600"/>
- <char name="notegraphic" width="600"/>
- <char name="ntilde" width="600"/>
- <char name="numbersign" width="600"/>
- <char name="o" width="600"/>
- <char name="oacute" width="600"/>
- <char name="ocircumflex" width="600"/>
- <char name="odieresis" width="600"/>
- <char name="oe" width="600"/>
- <char name="ogonek" width="600"/>
- <char name="ograve" width="600"/>
- <char name="one" width="600"/>
- <char name="onehalf" width="600"/>
- <char name="onequarter" width="600"/>
- <char name="onesuperior" width="600"/>
- <char name="ordfeminine" width="600"/>
- <char name="ordmasculine" width="600"/>
- <char name="oslash" width="600"/>
- <char name="otilde" width="600"/>
- <char name="overscore" width="600"/>
- <char name="p" width="600"/>
- <char name="paragraph" width="600"/>
- <char name="parenleft" width="600"/>
- <char name="parenright" width="600"/>
- <char name="percent" width="600"/>
- <char name="period" width="600"/>
- <char name="periodcentered" width="600"/>
- <char name="perthousand" width="600"/>
- <char name="plus" width="600"/>
- <char name="plusminus" width="600"/>
- <char name="prescription" width="600"/>
- <char name="q" width="600"/>
- <char name="question" width="600"/>
- <char name="questiondown" width="600"/>
- <char name="quotedbl" width="600"/>
- <char name="quotedblbase" width="600"/>
- <char name="quotedblleft" width="600"/>
- <char name="quotedblright" width="600"/>
- <char name="quoteleft" width="600"/>
- <char name="quoteright" width="600"/>
- <char name="quotesinglbase" width="600"/>
- <char name="quotesingle" width="600"/>
- <char name="r" width="600"/>
- <char name="registered" width="600"/>
- <char name="return" width="600"/>
- <char name="ring" width="600"/>
- <char name="s" width="600"/>
- <char name="scaron" width="600"/>
- <char name="scedilla" width="600"/>
- <char name="section" width="600"/>
- <char name="semicolon" width="600"/>
- <char name="seven" width="600"/>
- <char name="six" width="600"/>
- <char name="slash" width="600"/>
- <char name="space" width="600"/>
+ <char-metrics>
+ <char name="A" width="600" llx="3" lly="0" urx="597" ury="562"/>
+ <char name="AE" width="600" llx="3" lly="0" urx="550" ury="562"/>
+ <char name="Aacute" width="600" llx="3" lly="0" urx="597" ury="805"/>
+ <char name="Acircumflex" width="600" llx="3" lly="0" urx="597" ury="787"/>
+ <char name="Adieresis" width="600" llx="3" lly="0" urx="597" ury="753"/>
+ <char name="Agrave" width="600" llx="3" lly="0" urx="597" ury="805"/>
+ <char name="Aring" width="600" llx="3" lly="0" urx="597" ury="750"/>
+ <char name="Atilde" width="600" llx="3" lly="0" urx="597" ury="729"/>
+ <char name="B" width="600" llx="43" lly="0" urx="559" ury="562"/>
+ <char name="C" width="600" llx="41" lly="-18" urx="540" ury="580"/>
+ <char name="Ccedilla" width="600" llx="41" lly="-151" urx="540" ury="580"/>
+ <char name="D" width="600" llx="43" lly="0" urx="574" ury="562"/>
+ <char name="E" width="600" llx="53" lly="0" urx="550" ury="562"/>
+ <char name="Eacute" width="600" llx="53" lly="0" urx="550" ury="805"/>
+ <char name="Ecircumflex" width="600" llx="53" lly="0" urx="550" ury="787"/>
+ <char name="Edieresis" width="600" llx="53" lly="0" urx="550" ury="753"/>
+ <char name="Egrave" width="600" llx="53" lly="0" urx="550" ury="805"/>
+ <char name="Eth" width="600" llx="30" lly="0" urx="574" ury="562"/>
+ <char name="Euro" width="600" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="600" llx="53" lly="0" urx="545" ury="562"/>
+ <char name="G" width="600" llx="31" lly="-18" urx="575" ury="580"/>
+ <char name="H" width="600" llx="32" lly="0" urx="568" ury="562"/>
+ <char name="I" width="600" llx="96" lly="0" urx="504" ury="562"/>
+ <char name="Iacute" width="600" llx="96" lly="0" urx="504" ury="805"/>
+ <char name="Icircumflex" width="600" llx="96" lly="0" urx="504" ury="787"/>
+ <char name="Idieresis" width="600" llx="96" lly="0" urx="504" ury="753"/>
+ <char name="Igrave" width="600" llx="96" lly="0" urx="504" ury="805"/>
+ <char name="J" width="600" llx="34" lly="-18" urx="566" ury="562"/>
+ <char name="K" width="600" llx="38" lly="0" urx="582" ury="562"/>
+ <char name="L" width="600" llx="47" lly="0" urx="554" ury="562"/>
+ <char name="Lslash" width="600" llx="47" lly="0" urx="554" ury="562"/>
+ <char name="M" width="600" llx="4" lly="0" urx="596" ury="562"/>
+ <char name="N" width="600" llx="7" lly="-13" urx="593" ury="562"/>
+ <char name="Ntilde" width="600" llx="7" lly="-13" urx="593" ury="729"/>
+ <char name="O" width="600" llx="43" lly="-18" urx="557" ury="580"/>
+ <char name="OE" width="600" llx="7" lly="0" urx="567" ury="562"/>
+ <char name="Oacute" width="600" llx="43" lly="-18" urx="557" ury="805"/>
+ <char name="Ocircumflex" width="600" llx="43" lly="-18" urx="557" ury="787"/>
+ <char name="Odieresis" width="600" llx="43" lly="-18" urx="557" ury="753"/>
+ <char name="Ograve" width="600" llx="43" lly="-18" urx="557" ury="805"/>
+ <char name="Oslash" width="600" llx="43" lly="-80" urx="557" ury="629"/>
+ <char name="Otilde" width="600" llx="43" lly="-18" urx="557" ury="729"/>
+ <char name="P" width="600" llx="79" lly="0" urx="558" ury="562"/>
+ <char name="Q" width="600" llx="43" lly="-138" urx="557" ury="580"/>
+ <char name="R" width="600" llx="38" lly="0" urx="588" ury="562"/>
+ <char name="S" width="600" llx="72" lly="-20" urx="529" ury="580"/>
+ <char name="Scaron" width="600" llx="72" lly="-20" urx="529" ury="802"/>
+ <char name="Scedilla" width="600" llx="72" lly="-151" urx="529" ury="580"/>
+ <char name="T" width="600" llx="38" lly="0" urx="563" ury="562"/>
+ <char name="Thorn" width="600" llx="79" lly="0" urx="538" ury="562"/>
+ <char name="U" width="600" llx="17" lly="-18" urx="583" ury="562"/>
+ <char name="Uacute" width="600" llx="17" lly="-18" urx="583" ury="805"/>
+ <char name="Ucircumflex" width="600" llx="17" lly="-18" urx="583" ury="787"/>
+ <char name="Udieresis" width="600" llx="17" lly="-18" urx="583" ury="753"/>
+ <char name="Ugrave" width="600" llx="17" lly="-18" urx="583" ury="805"/>
+ <char name="V" width="600" llx="-4" lly="-13" urx="604" ury="562"/>
+ <char name="W" width="600" llx="-3" lly="-13" urx="603" ury="562"/>
+ <char name="X" width="600" llx="23" lly="0" urx="577" ury="562"/>
+ <char name="Y" width="600" llx="24" lly="0" urx="576" ury="562"/>
+ <char name="Yacute" width="600" llx="24" lly="0" urx="576" ury="805"/>
+ <char name="Ydieresis" width="600" llx="24" lly="0" urx="576" ury="753"/>
+ <char name="Z" width="600" llx="86" lly="0" urx="514" ury="562"/>
+ <char name="Zcaron" width="600" llx="86" lly="0" urx="514" ury="802"/>
+ <char name="a" width="600" llx="53" lly="-15" urx="559" ury="441"/>
+ <char name="aacute" width="600" llx="53" lly="-15" urx="559" ury="672"/>
+ <char name="acircumflex" width="600" llx="53" lly="-15" urx="559" ury="654"/>
+ <char name="acute" width="600" llx="242" lly="497" urx="469" ury="672"/>
+ <char name="adieresis" width="600" llx="53" lly="-15" urx="559" ury="620"/>
+ <char name="ae" width="600" llx="19" lly="-15" urx="570" ury="441"/>
+ <char name="agrave" width="600" llx="53" lly="-15" urx="559" ury="672"/>
+ <char name="ampersand" width="600" llx="63" lly="-15" urx="538" ury="543"/>
+ <char name="aring" width="600" llx="53" lly="-15" urx="559" ury="627"/>
+ <char name="asciicircum" width="600" llx="94" lly="354" urx="506" ury="622"/>
+ <char name="asciitilde" width="600" llx="63" lly="197" urx="540" ury="320"/>
+ <char name="asterisk" width="600" llx="116" lly="257" urx="484" ury="607"/>
+ <char name="at" width="600" llx="77" lly="-15" urx="533" ury="622"/>
+ <char name="atilde" width="600" llx="53" lly="-15" urx="559" ury="606"/>
+ <char name="b" width="600" llx="14" lly="-15" urx="575" ury="629"/>
+ <char name="backslash" width="600" llx="118" lly="-80" urx="482" ury="629"/>
+ <char name="bar" width="600" llx="275" lly="-250" urx="326" ury="750"/>
+ <char name="braceleft" width="600" llx="182" lly="-108" urx="437" ury="622"/>
+ <char name="braceright" width="600" llx="163" lly="-108" urx="418" ury="622"/>
+ <char name="bracketleft" width="600" llx="269" lly="-108" urx="442" ury="622"/>
+ <char name="bracketright" width="600" llx="158" lly="-108" urx="331" ury="622"/>
+ <char name="breve" width="600" llx="153" lly="501" urx="447" ury="609"/>
+ <char name="brokenbar" width="600" llx="275" lly="-175" urx="326" ury="675"/>
+ <char name="bullet" width="600" llx="172" lly="130" urx="428" ury="383"/>
+ <char name="c" width="600" llx="66" lly="-15" urx="529" ury="441"/>
+ <char name="caron" width="600" llx="124" lly="492" urx="476" ury="669"/>
+ <char name="ccedilla" width="600" llx="66" lly="-151" urx="529" ury="441"/>
+ <char name="cedilla" width="600" llx="224" lly="-151" urx="362" ury="10"/>
+ <char name="cent" width="600" llx="96" lly="-49" urx="500" ury="614"/>
+ <char name="circumflex" width="600" llx="124" lly="477" urx="476" ury="654"/>
+ <char name="colon" width="600" llx="229" lly="-15" urx="371" ury="385"/>
+ <char name="comma" width="600" llx="181" lly="-112" urx="344" ury="122"/>
+ <char name="copyright" width="600" llx="0" lly="-18" urx="600" ury="580"/>
+ <char name="currency" width="600" llx="73" lly="58" urx="527" ury="506"/>
+ <char name="d" width="600" llx="45" lly="-15" urx="591" ury="629"/>
+ <char name="dagger" width="600" llx="141" lly="-78" urx="459" ury="580"/>
+ <char name="daggerdbl" width="600" llx="141" lly="-78" urx="459" ury="580"/>
+ <char name="degree" width="600" llx="123" lly="269" urx="477" ury="622"/>
+ <char name="dieresis" width="600" llx="148" lly="537" urx="453" ury="640"/>
+ <char name="divide" width="600" llx="87" lly="48" urx="513" ury="467"/>
+ <char name="dollar" width="600" llx="105" lly="-126" urx="496" ury="662"/>
+ <char name="dotaccent" width="600" llx="249" lly="537" urx="352" ury="640"/>
+ <char name="dotlessi" width="600" llx="95" lly="0" urx="505" ury="426"/>
+ <char name="e" width="600" llx="66" lly="-15" urx="548" ury="441"/>
+ <char name="eacute" width="600" llx="66" lly="-15" urx="548" ury="672"/>
+ <char name="ecircumflex" width="600" llx="66" lly="-15" urx="548" ury="654"/>
+ <char name="edieresis" width="600" llx="66" lly="-15" urx="548" ury="620"/>
+ <char name="egrave" width="600" llx="66" lly="-15" urx="548" ury="672"/>
+ <char name="eight" width="600" llx="102" lly="-15" urx="498" ury="622"/>
+ <char name="ellipsis" width="600" llx="37" lly="-15" urx="563" ury="111"/>
+ <char name="emdash" width="600" llx="0" lly="231" urx="600" ury="285"/>
+ <char name="endash" width="600" llx="75" lly="231" urx="525" ury="285"/>
+ <char name="equal" width="600" llx="80" lly="138" urx="520" ury="376"/>
+ <char name="eth" width="600" llx="62" lly="-15" urx="538" ury="629"/>
+ <char name="exclam" width="600" llx="236" lly="-15" urx="364" ury="572"/>
+ <char name="exclamdown" width="600" llx="236" lly="-157" urx="364" ury="430"/>
+ <char name="f" width="600" llx="114" lly="0" urx="531" ury="629"/>
+ <char name="fi" width="600" llx="3" lly="0" urx="597" ury="629"/>
+ <char name="five" width="600" llx="92" lly="-15" urx="497" ury="607"/>
+ <char name="fl" width="600" llx="3" lly="0" urx="597" ury="629"/>
+ <char name="florin" width="600" llx="4" lly="-143" urx="539" ury="622"/>
+ <char name="four" width="600" llx="78" lly="0" urx="500" ury="622"/>
+ <char name="fraction" width="600" llx="92" lly="-57" urx="509" ury="665"/>
+ <char name="g" width="600" llx="45" lly="-157" urx="566" ury="441"/>
+ <char name="germandbls" width="600" llx="48" lly="-15" urx="588" ury="629"/>
+ <char name="grave" width="600" llx="151" lly="497" urx="378" ury="672"/>
+ <char name="greater" width="600" llx="66" lly="42" urx="544" ury="472"/>
+ <char name="guillemotleft" width="600" llx="37" lly="70" urx="563" ury="446"/>
+ <char name="guillemotright" width="600" llx="37" lly="70" urx="563" ury="446"/>
+ <char name="guilsinglleft" width="600" llx="149" lly="70" urx="451" ury="446"/>
+ <char name="guilsinglright" width="600" llx="149" lly="70" urx="451" ury="446"/>
+ <char name="h" width="600" llx="18" lly="0" urx="582" ury="629"/>
+ <char name="hungarumlaut" width="600" llx="133" lly="497" urx="540" ury="672"/>
+ <char name="hyphen" width="600" llx="103" lly="231" urx="497" ury="285"/>
+ <char name="i" width="600" llx="95" lly="0" urx="505" ury="657"/>
+ <char name="iacute" width="600" llx="95" lly="0" urx="505" ury="672"/>
+ <char name="icircumflex" width="600" llx="94" lly="0" urx="505" ury="654"/>
+ <char name="idieresis" width="600" llx="95" lly="0" urx="505" ury="620"/>
+ <char name="igrave" width="600" llx="95" lly="0" urx="505" ury="672"/>
+ <char name="j" width="600" llx="82" lly="-157" urx="410" ury="657"/>
+ <char name="k" width="600" llx="43" lly="0" urx="580" ury="629"/>
+ <char name="l" width="600" llx="95" lly="0" urx="505" ury="629"/>
+ <char name="less" width="600" llx="41" lly="42" urx="519" ury="472"/>
+ <char name="logicalnot" width="600" llx="87" lly="108" urx="513" ury="369"/>
+ <char name="lslash" width="600" llx="95" lly="0" urx="505" ury="629"/>
+ <char name="m" width="600" llx="-5" lly="0" urx="605" ury="441"/>
+ <char name="macron" width="600" llx="120" lly="525" urx="480" ury="565"/>
+ <char name="minus" width="600" llx="80" lly="232" urx="520" ury="283"/>
+ <char name="mu" width="600" llx="21" lly="-157" urx="562" ury="426"/>
+ <char name="multiply" width="600" llx="87" lly="43" urx="515" ury="470"/>
+ <char name="n" width="600" llx="26" lly="0" urx="575" ury="441"/>
+ <char name="nine" width="600" llx="96" lly="-15" urx="489" ury="622"/>
+ <char name="ntilde" width="600" llx="26" lly="0" urx="575" ury="606"/>
+ <char name="numbersign" width="600" llx="93" lly="-32" urx="507" ury="639"/>
+ <char name="o" width="600" llx="62" lly="-15" urx="538" ury="441"/>
+ <char name="oacute" width="600" llx="62" lly="-15" urx="538" ury="672"/>
+ <char name="ocircumflex" width="600" llx="62" lly="-15" urx="538" ury="654"/>
+ <char name="odieresis" width="600" llx="62" lly="-15" urx="538" ury="620"/>
+ <char name="oe" width="600" llx="19" lly="-15" urx="559" ury="441"/>
+ <char name="ogonek" width="600" llx="211" lly="-172" urx="407" ury="4"/>
+ <char name="ograve" width="600" llx="62" lly="-15" urx="538" ury="672"/>
+ <char name="one" width="600" llx="96" lly="0" urx="505" ury="622"/>
+ <char name="onehalf" width="600" llx="0" lly="-57" urx="611" ury="665"/>
+ <char name="onequarter" width="600" llx="0" lly="-57" urx="600" ury="665"/>
+ <char name="onesuperior" width="600" llx="172" lly="249" urx="428" ury="622"/>
+ <char name="ordfeminine" width="600" llx="156" lly="249" urx="442" ury="580"/>
+ <char name="ordmasculine" width="600" llx="157" lly="249" urx="443" ury="580"/>
+ <char name="oslash" width="600" llx="62" lly="-80" urx="538" ury="506"/>
+ <char name="otilde" width="600" llx="62" lly="-15" urx="538" ury="606"/>
+ <char name="p" width="600" llx="9" lly="-157" urx="555" ury="441"/>
+ <char name="paragraph" width="600" llx="50" lly="-78" urx="511" ury="562"/>
+ <char name="parenleft" width="600" llx="269" lly="-108" urx="440" ury="622"/>
+ <char name="parenright" width="600" llx="160" lly="-108" urx="331" ury="622"/>
+ <char name="percent" width="600" llx="81" lly="-15" urx="518" ury="622"/>
+ <char name="period" width="600" llx="229" lly="-15" urx="371" ury="109"/>
+ <char name="periodcentered" width="600" llx="222" lly="189" urx="378" ury="327"/>
+ <char name="perthousand" width="600" llx="3" lly="-15" urx="600" ury="622"/>
+ <char name="plus" width="600" llx="80" lly="44" urx="520" ury="470"/>
+ <char name="plusminus" width="600" llx="87" lly="44" urx="513" ury="558"/>
+ <char name="q" width="600" llx="45" lly="-157" urx="591" ury="441"/>
+ <char name="question" width="600" llx="129" lly="-15" urx="492" ury="572"/>
+ <char name="questiondown" width="600" llx="108" lly="-157" urx="471" ury="430"/>
+ <char name="quotedbl" width="600" llx="187" lly="328" urx="413" ury="562"/>
+ <char name="quotedblbase" width="600" llx="143" lly="-134" urx="457" ury="100"/>
+ <char name="quotedblleft" width="600" llx="143" lly="328" urx="471" ury="562"/>
+ <char name="quotedblright" width="600" llx="143" lly="328" urx="457" ury="562"/>
+ <char name="quoteleft" width="600" llx="224" lly="328" urx="387" ury="562"/>
+ <char name="quoteright" width="600" llx="213" lly="328" urx="376" ury="562"/>
+ <char name="quotesinglbase" width="600" llx="213" lly="-134" urx="376" ury="100"/>
+ <char name="quotesingle" width="600" llx="259" lly="328" urx="341" ury="562"/>
+ <char name="r" width="600" llx="60" lly="0" urx="559" ury="441"/>
+ <char name="registered" width="600" llx="0" lly="-18" urx="600" ury="580"/>
+ <char name="ring" width="600" llx="218" lly="463" urx="382" ury="627"/>
+ <char name="s" width="600" llx="80" lly="-15" urx="513" ury="441"/>
+ <char name="scaron" width="600" llx="80" lly="-15" urx="513" ury="669"/>
+ <char name="scedilla" width="600" llx="80" lly="-151" urx="513" ury="441"/>
+ <char name="section" width="600" llx="113" lly="-78" urx="488" ury="580"/>
+ <char name="semicolon" width="600" llx="181" lly="-112" urx="371" ury="385"/>
+ <char name="seven" width="600" llx="82" lly="0" urx="483" ury="607"/>
+ <char name="six" width="600" llx="111" lly="-15" urx="497" ury="622"/>
+ <char name="slash" width="600" llx="125" lly="-80" urx="475" ury="629"/>
+ <char name="space" width="600" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="600"/>
- <char name="square" width="600"/>
- <char name="sterling" width="600"/>
- <char name="stop" width="600"/>
- <char name="t" width="600"/>
- <char name="tab" width="600"/>
- <char name="thorn" width="600"/>
- <char name="three" width="600"/>
- <char name="threequarters" width="600"/>
- <char name="threesuperior" width="600"/>
- <char name="tilde" width="600"/>
- <char name="trademark" width="600"/>
- <char name="two" width="600"/>
- <char name="twosuperior" width="600"/>
- <char name="u" width="600"/>
- <char name="uacute" width="600"/>
- <char name="ucircumflex" width="600"/>
- <char name="udieresis" width="600"/>
- <char name="ugrave" width="600"/>
- <char name="underscore" width="600"/>
- <char name="up" width="600"/>
- <char name="v" width="600"/>
- <char name="w" width="600"/>
- <char name="x" width="600"/>
- <char name="y" width="600"/>
- <char name="yacute" width="600"/>
- <char name="ydieresis" width="600"/>
- <char name="yen" width="600"/>
- <char name="z" width="600"/>
- <char name="zcaron" width="600"/>
- <char name="zero" width="600"/>
- </widths>
-</font-metrics> \ No newline at end of file
+ <char name="sterling" width="600" llx="84" lly="-21" urx="521" ury="611"/>
+ <char name="t" width="600" llx="87" lly="-15" urx="530" ury="561"/>
+ <char name="thorn" width="600" llx="-6" lly="-157" urx="555" ury="629"/>
+ <char name="three" width="600" llx="75" lly="-15" urx="466" ury="622"/>
+ <char name="threequarters" width="600" llx="8" lly="-56" urx="593" ury="666"/>
+ <char name="threesuperior" width="600" llx="155" lly="240" urx="406" ury="622"/>
+ <char name="tilde" width="600" llx="105" lly="489" urx="503" ury="606"/>
+ <char name="trademark" width="600" llx="-23" lly="263" urx="623" ury="562"/>
+ <char name="two" width="600" llx="70" lly="0" urx="471" ury="622"/>
+ <char name="twosuperior" width="600" llx="177" lly="249" urx="424" ury="622"/>
+ <char name="u" width="600" llx="21" lly="-15" urx="562" ury="426"/>
+ <char name="uacute" width="600" llx="21" lly="-15" urx="562" ury="672"/>
+ <char name="ucircumflex" width="600" llx="21" lly="-15" urx="562" ury="654"/>
+ <char name="udieresis" width="600" llx="21" lly="-15" urx="562" ury="620"/>
+ <char name="ugrave" width="600" llx="21" lly="-15" urx="562" ury="672"/>
+ <char name="underscore" width="600" llx="0" lly="-125" urx="600" ury="-75"/>
+ <char name="v" width="600" llx="10" lly="-10" urx="590" ury="426"/>
+ <char name="w" width="600" llx="-4" lly="-10" urx="604" ury="426"/>
+ <char name="x" width="600" llx="20" lly="0" urx="580" ury="426"/>
+ <char name="y" width="600" llx="7" lly="-157" urx="592" ury="426"/>
+ <char name="yacute" width="600" llx="7" lly="-157" urx="592" ury="672"/>
+ <char name="ydieresis" width="600" llx="7" lly="-157" urx="592" ury="620"/>
+ <char name="yen" width="600" llx="26" lly="0" urx="574" ury="562"/>
+ <char name="z" width="600" llx="99" lly="0" urx="502" ury="426"/>
+ <char name="zcaron" width="600" llx="99" lly="0" urx="502" ury="669"/>
+ <char name="zero" width="600" llx="106" lly="-15" urx="494" ury="622"/>
+ </char-metrics>
+</font-metrics>
diff --git a/src/codegen/fonts/CourierBold.xml b/src/codegen/fonts/CourierBold.xml
index fa8bed196..c06ef5f23 100644
--- a/src/codegen/fonts/CourierBold.xml
+++ b/src/codegen/fonts/CourierBold.xml
@@ -22,275 +22,246 @@
<family-name>Courier</family-name>
<class-name>CourierBold</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>562</cap-height>
<x-height>439</x-height>
<ascender>626</ascender>
<descender>-142</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="600"/>
- <char name="AE" width="600"/>
- <char name="Aacute" width="600"/>
- <char name="Acircumflex" width="600"/>
- <char name="Adieresis" width="600"/>
- <char name="Agrave" width="600"/>
- <char name="Aring" width="600"/>
- <char name="Atilde" width="600"/>
- <char name="B" width="600"/>
- <char name="C" width="600"/>
- <char name="Ccedilla" width="600"/>
- <char name="D" width="600"/>
- <char name="E" width="600"/>
- <char name="Eacute" width="600"/>
- <char name="Ecircumflex" width="600"/>
- <char name="Edieresis" width="600"/>
- <char name="Egrave" width="600"/>
- <char name="Eth" width="600"/>
- <char name="Euro" width="600"/>
- <char name="F" width="600"/>
- <char name="G" width="600"/>
- <char name="Gcaron" width="600"/>
- <char name="H" width="600"/>
- <char name="I" width="600"/>
- <char name="IJ" width="600"/>
- <char name="Iacute" width="600"/>
- <char name="Icircumflex" width="600"/>
- <char name="Idieresis" width="600"/>
- <char name="Idot" width="600"/>
- <char name="Igrave" width="600"/>
- <char name="J" width="600"/>
- <char name="K" width="600"/>
- <char name="L" width="600"/>
- <char name="LL" width="600"/>
- <char name="Lslash" width="600"/>
- <char name="M" width="600"/>
- <char name="N" width="600"/>
- <char name="Ntilde" width="600"/>
- <char name="O" width="600"/>
- <char name="OE" width="600"/>
- <char name="Oacute" width="600"/>
- <char name="Ocircumflex" width="600"/>
- <char name="Odieresis" width="600"/>
- <char name="Ograve" width="600"/>
- <char name="Oslash" width="600"/>
- <char name="Otilde" width="600"/>
- <char name="P" width="600"/>
- <char name="Q" width="600"/>
- <char name="R" width="600"/>
- <char name="S" width="600"/>
- <char name="Scaron" width="600"/>
- <char name="Scedilla" width="600"/>
- <char name="T" width="600"/>
- <char name="Thorn" width="600"/>
- <char name="U" width="600"/>
- <char name="Uacute" width="600"/>
- <char name="Ucircumflex" width="600"/>
- <char name="Udieresis" width="600"/>
- <char name="Ugrave" width="600"/>
- <char name="V" width="600"/>
- <char name="W" width="600"/>
- <char name="X" width="600"/>
- <char name="Y" width="600"/>
- <char name="Yacute" width="600"/>
- <char name="Ydieresis" width="600"/>
- <char name="Z" width="600"/>
- <char name="Zcaron" width="600"/>
- <char name="a" width="600"/>
- <char name="aacute" width="600"/>
- <char name="acircumflex" width="600"/>
- <char name="acute" width="600"/>
- <char name="adieresis" width="600"/>
- <char name="ae" width="600"/>
- <char name="agrave" width="600"/>
- <char name="ampersand" width="600"/>
- <char name="aring" width="600"/>
- <char name="arrowboth" width="600"/>
- <char name="arrowdown" width="600"/>
- <char name="arrowleft" width="600"/>
- <char name="arrowright" width="600"/>
- <char name="arrowup" width="600"/>
- <char name="asciicircum" width="600"/>
- <char name="asciitilde" width="600"/>
- <char name="asterisk" width="600"/>
- <char name="at" width="600"/>
- <char name="atilde" width="600"/>
- <char name="b" width="600"/>
- <char name="backslash" width="600"/>
- <char name="bar" width="600"/>
- <char name="braceleft" width="600"/>
- <char name="braceright" width="600"/>
- <char name="bracketleft" width="600"/>
- <char name="bracketright" width="600"/>
- <char name="breve" width="600"/>
- <char name="brokenbar" width="600"/>
- <char name="bullet" width="600"/>
- <char name="c" width="600"/>
- <char name="caron" width="600"/>
- <char name="ccedilla" width="600"/>
- <char name="cedilla" width="600"/>
- <char name="cent" width="600"/>
- <char name="center" width="600"/>
- <char name="circumflex" width="600"/>
- <char name="colon" width="600"/>
- <char name="comma" width="600"/>
- <char name="copyright" width="600"/>
- <char name="currency" width="600"/>
- <char name="d" width="600"/>
- <char name="dagger" width="600"/>
- <char name="daggerdbl" width="600"/>
- <char name="dectab" width="600"/>
- <char name="degree" width="600"/>
- <char name="dieresis" width="600"/>
- <char name="divide" width="600"/>
- <char name="dollar" width="600"/>
- <char name="dotaccent" width="600"/>
- <char name="dotlessi" width="600"/>
- <char name="down" width="600"/>
- <char name="e" width="600"/>
- <char name="eacute" width="600"/>
- <char name="ecircumflex" width="600"/>
- <char name="edieresis" width="600"/>
- <char name="egrave" width="600"/>
- <char name="eight" width="600"/>
- <char name="ellipsis" width="600"/>
- <char name="emdash" width="600"/>
- <char name="endash" width="600"/>
- <char name="equal" width="600"/>
- <char name="eth" width="600"/>
- <char name="exclam" width="600"/>
- <char name="exclamdown" width="600"/>
- <char name="f" width="600"/>
- <char name="fi" width="600"/>
- <char name="five" width="600"/>
- <char name="fl" width="600"/>
- <char name="florin" width="600"/>
- <char name="format" width="600"/>
- <char name="four" width="600"/>
- <char name="fraction" width="600"/>
- <char name="g" width="600"/>
- <char name="gcaron" width="600"/>
- <char name="germandbls" width="600"/>
- <char name="grave" width="600"/>
- <char name="graybox" width="600"/>
- <char name="greater" width="600"/>
- <char name="guillemotleft" width="600"/>
- <char name="guillemotright" width="600"/>
- <char name="guilsinglleft" width="600"/>
- <char name="guilsinglright" width="600"/>
- <char name="h" width="600"/>
- <char name="hungarumlaut" width="600"/>
- <char name="hyphen" width="600"/>
- <char name="i" width="600"/>
- <char name="iacute" width="600"/>
- <char name="icircumflex" width="600"/>
- <char name="idieresis" width="600"/>
- <char name="igrave" width="600"/>
- <char name="ij" width="600"/>
- <char name="indent" width="600"/>
- <char name="j" width="600"/>
- <char name="k" width="600"/>
- <char name="l" width="600"/>
- <char name="largebullet" width="600"/>
- <char name="left" width="600"/>
- <char name="less" width="600"/>
- <char name="lira" width="600"/>
- <char name="ll" width="600"/>
- <char name="logicalnot" width="600"/>
- <char name="lslash" width="600"/>
- <char name="m" width="600"/>
- <char name="macron" width="600"/>
- <char name="merge" width="600"/>
- <char name="minus" width="600"/>
- <char name="mu" width="600"/>
- <char name="multiply" width="600"/>
- <char name="n" width="600"/>
- <char name="nine" width="600"/>
- <char name="notegraphic" width="600"/>
- <char name="ntilde" width="600"/>
- <char name="numbersign" width="600"/>
- <char name="o" width="600"/>
- <char name="oacute" width="600"/>
- <char name="ocircumflex" width="600"/>
- <char name="odieresis" width="600"/>
- <char name="oe" width="600"/>
- <char name="ogonek" width="600"/>
- <char name="ograve" width="600"/>
- <char name="one" width="600"/>
- <char name="onehalf" width="600"/>
- <char name="onequarter" width="600"/>
- <char name="onesuperior" width="600"/>
- <char name="ordfeminine" width="600"/>
- <char name="ordmasculine" width="600"/>
- <char name="oslash" width="600"/>
- <char name="otilde" width="600"/>
- <char name="overscore" width="600"/>
- <char name="p" width="600"/>
- <char name="paragraph" width="600"/>
- <char name="parenleft" width="600"/>
- <char name="parenright" width="600"/>
- <char name="percent" width="600"/>
- <char name="period" width="600"/>
- <char name="periodcentered" width="600"/>
- <char name="perthousand" width="600"/>
- <char name="plus" width="600"/>
- <char name="plusminus" width="600"/>
- <char name="prescription" width="600"/>
- <char name="q" width="600"/>
- <char name="question" width="600"/>
- <char name="questiondown" width="600"/>
- <char name="quotedbl" width="600"/>
- <char name="quotedblbase" width="600"/>
- <char name="quotedblleft" width="600"/>
- <char name="quotedblright" width="600"/>
- <char name="quoteleft" width="600"/>
- <char name="quoteright" width="600"/>
- <char name="quotesinglbase" width="600"/>
- <char name="quotesingle" width="600"/>
- <char name="r" width="600"/>
- <char name="registered" width="600"/>
- <char name="return" width="600"/>
- <char name="ring" width="600"/>
- <char name="s" width="600"/>
- <char name="scaron" width="600"/>
- <char name="scedilla" width="600"/>
- <char name="section" width="600"/>
- <char name="semicolon" width="600"/>
- <char name="seven" width="600"/>
- <char name="six" width="600"/>
- <char name="slash" width="600"/>
- <char name="space" width="600"/>
+ <char-metrics>
+ <char name="A" width="600" llx="-9" lly="0" urx="609" ury="562"/>
+ <char name="AE" width="600" llx="-29" lly="0" urx="602" ury="562"/>
+ <char name="Aacute" width="600" llx="-9" lly="0" urx="609" ury="784"/>
+ <char name="Acircumflex" width="600" llx="-9" lly="0" urx="609" ury="780"/>
+ <char name="Adieresis" width="600" llx="-9" lly="0" urx="609" ury="761"/>
+ <char name="Agrave" width="600" llx="-9" lly="0" urx="609" ury="784"/>
+ <char name="Aring" width="600" llx="-9" lly="0" urx="609" ury="801"/>
+ <char name="Atilde" width="600" llx="-9" lly="0" urx="609" ury="759"/>
+ <char name="B" width="600" llx="30" lly="0" urx="573" ury="562"/>
+ <char name="C" width="600" llx="22" lly="-18" urx="560" ury="580"/>
+ <char name="Ccedilla" width="600" llx="22" lly="-206" urx="560" ury="580"/>
+ <char name="D" width="600" llx="30" lly="0" urx="594" ury="562"/>
+ <char name="E" width="600" llx="25" lly="0" urx="560" ury="562"/>
+ <char name="Eacute" width="600" llx="25" lly="0" urx="560" ury="784"/>
+ <char name="Ecircumflex" width="600" llx="25" lly="0" urx="560" ury="780"/>
+ <char name="Edieresis" width="600" llx="25" lly="0" urx="560" ury="761"/>
+ <char name="Egrave" width="600" llx="25" lly="0" urx="560" ury="784"/>
+ <char name="Eth" width="600" llx="30" lly="0" urx="594" ury="562"/>
+ <char name="Euro" width="600" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="600" llx="39" lly="0" urx="570" ury="562"/>
+ <char name="G" width="600" llx="22" lly="-18" urx="594" ury="580"/>
+ <char name="H" width="600" llx="20" lly="0" urx="580" ury="562"/>
+ <char name="I" width="600" llx="77" lly="0" urx="523" ury="562"/>
+ <char name="Iacute" width="600" llx="77" lly="0" urx="523" ury="784"/>
+ <char name="Icircumflex" width="600" llx="77" lly="0" urx="523" ury="780"/>
+ <char name="Idieresis" width="600" llx="77" lly="0" urx="523" ury="761"/>
+ <char name="Igrave" width="600" llx="77" lly="0" urx="523" ury="784"/>
+ <char name="J" width="600" llx="37" lly="-18" urx="601" ury="562"/>
+ <char name="K" width="600" llx="21" lly="0" urx="599" ury="562"/>
+ <char name="L" width="600" llx="39" lly="0" urx="578" ury="562"/>
+ <char name="Lslash" width="600" llx="39" lly="0" urx="578" ury="562"/>
+ <char name="M" width="600" llx="-2" lly="0" urx="602" ury="562"/>
+ <char name="N" width="600" llx="8" lly="-12" urx="610" ury="562"/>
+ <char name="Ntilde" width="600" llx="8" lly="-12" urx="610" ury="759"/>
+ <char name="O" width="600" llx="22" lly="-18" urx="578" ury="580"/>
+ <char name="OE" width="600" llx="-25" lly="0" urx="595" ury="562"/>
+ <char name="Oacute" width="600" llx="22" lly="-18" urx="578" ury="784"/>
+ <char name="Ocircumflex" width="600" llx="22" lly="-18" urx="578" ury="780"/>
+ <char name="Odieresis" width="600" llx="22" lly="-18" urx="578" ury="761"/>
+ <char name="Ograve" width="600" llx="22" lly="-18" urx="578" ury="784"/>
+ <char name="Oslash" width="600" llx="22" lly="-22" urx="578" ury="584"/>
+ <char name="Otilde" width="600" llx="22" lly="-18" urx="578" ury="759"/>
+ <char name="P" width="600" llx="48" lly="0" urx="559" ury="562"/>
+ <char name="Q" width="600" llx="32" lly="-138" urx="578" ury="580"/>
+ <char name="R" width="600" llx="24" lly="0" urx="599" ury="562"/>
+ <char name="S" width="600" llx="47" lly="-22" urx="553" ury="582"/>
+ <char name="Scaron" width="600" llx="47" lly="-22" urx="553" ury="790"/>
+ <char name="Scedilla" width="600" llx="47" lly="-206" urx="553" ury="582"/>
+ <char name="T" width="600" llx="21" lly="0" urx="579" ury="562"/>
+ <char name="Thorn" width="600" llx="48" lly="0" urx="557" ury="562"/>
+ <char name="U" width="600" llx="4" lly="-18" urx="596" ury="562"/>
+ <char name="Uacute" width="600" llx="4" lly="-18" urx="596" ury="784"/>
+ <char name="Ucircumflex" width="600" llx="4" lly="-18" urx="596" ury="780"/>
+ <char name="Udieresis" width="600" llx="4" lly="-18" urx="596" ury="761"/>
+ <char name="Ugrave" width="600" llx="4" lly="-18" urx="596" ury="784"/>
+ <char name="V" width="600" llx="-13" lly="0" urx="613" ury="562"/>
+ <char name="W" width="600" llx="-18" lly="0" urx="618" ury="562"/>
+ <char name="X" width="600" llx="12" lly="0" urx="588" ury="562"/>
+ <char name="Y" width="600" llx="12" lly="0" urx="589" ury="562"/>
+ <char name="Yacute" width="600" llx="12" lly="0" urx="589" ury="784"/>
+ <char name="Ydieresis" width="600" llx="12" lly="0" urx="589" ury="761"/>
+ <char name="Z" width="600" llx="62" lly="0" urx="539" ury="562"/>
+ <char name="Zcaron" width="600" llx="62" lly="0" urx="539" ury="790"/>
+ <char name="a" width="600" llx="35" lly="-15" urx="570" ury="454"/>
+ <char name="aacute" width="600" llx="35" lly="-15" urx="570" ury="661"/>
+ <char name="acircumflex" width="600" llx="35" lly="-15" urx="570" ury="657"/>
+ <char name="acute" width="600" llx="205" lly="508" urx="468" ury="661"/>
+ <char name="adieresis" width="600" llx="35" lly="-15" urx="570" ury="638"/>
+ <char name="ae" width="600" llx="-4" lly="-15" urx="601" ury="454"/>
+ <char name="agrave" width="600" llx="35" lly="-15" urx="570" ury="661"/>
+ <char name="ampersand" width="600" llx="36" lly="-15" urx="546" ury="543"/>
+ <char name="aring" width="600" llx="35" lly="-15" urx="570" ury="678"/>
+ <char name="asciicircum" width="600" llx="108" lly="250" urx="492" ury="616"/>
+ <char name="asciitilde" width="600" llx="71" lly="153" urx="530" ury="356"/>
+ <char name="asterisk" width="600" llx="91" lly="219" urx="509" ury="601"/>
+ <char name="at" width="600" llx="16" lly="-15" urx="584" ury="616"/>
+ <char name="atilde" width="600" llx="35" lly="-15" urx="570" ury="636"/>
+ <char name="b" width="600" llx="0" lly="-15" urx="584" ury="626"/>
+ <char name="backslash" width="600" llx="99" lly="-77" urx="503" ury="626"/>
+ <char name="bar" width="600" llx="255" lly="-250" urx="345" ury="750"/>
+ <char name="braceleft" width="600" llx="160" lly="-102" urx="464" ury="616"/>
+ <char name="braceright" width="600" llx="136" lly="-102" urx="440" ury="616"/>
+ <char name="bracketleft" width="600" llx="245" lly="-102" urx="475" ury="616"/>
+ <char name="bracketright" width="600" llx="125" lly="-102" urx="355" ury="616"/>
+ <char name="breve" width="600" llx="83" lly="468" urx="517" ury="631"/>
+ <char name="brokenbar" width="600" llx="255" lly="-175" urx="345" ury="675"/>
+ <char name="bullet" width="600" llx="140" lly="132" urx="460" ury="430"/>
+ <char name="c" width="600" llx="40" lly="-15" urx="545" ury="459"/>
+ <char name="caron" width="600" llx="103" lly="493" urx="497" ury="667"/>
+ <char name="ccedilla" width="600" llx="40" lly="-206" urx="545" ury="459"/>
+ <char name="cedilla" width="600" llx="205" lly="-206" urx="387" ury="0"/>
+ <char name="cent" width="600" llx="66" lly="-49" urx="518" ury="614"/>
+ <char name="circumflex" width="600" llx="103" lly="483" urx="497" ury="657"/>
+ <char name="colon" width="600" llx="191" lly="-15" urx="407" ury="425"/>
+ <char name="comma" width="600" llx="123" lly="-111" urx="393" ury="174"/>
+ <char name="copyright" width="600" llx="0" lly="-18" urx="600" ury="580"/>
+ <char name="currency" width="600" llx="54" lly="49" urx="546" ury="517"/>
+ <char name="d" width="600" llx="20" lly="-15" urx="591" ury="626"/>
+ <char name="dagger" width="600" llx="106" lly="-70" urx="494" ury="580"/>
+ <char name="daggerdbl" width="600" llx="106" lly="-70" urx="494" ury="580"/>
+ <char name="degree" width="600" llx="86" lly="243" urx="474" ury="616"/>
+ <char name="dieresis" width="600" llx="128" lly="498" urx="472" ury="638"/>
+ <char name="divide" width="600" llx="71" lly="16" urx="529" ury="500"/>
+ <char name="dollar" width="600" llx="82" lly="-126" urx="519" ury="666"/>
+ <char name="dotaccent" width="600" llx="230" lly="498" urx="370" ury="638"/>
+ <char name="dotlessi" width="600" llx="77" lly="0" urx="523" ury="439"/>
+ <char name="e" width="600" llx="40" lly="-15" urx="563" ury="454"/>
+ <char name="eacute" width="600" llx="40" lly="-15" urx="563" ury="661"/>
+ <char name="ecircumflex" width="600" llx="40" lly="-15" urx="563" ury="657"/>
+ <char name="edieresis" width="600" llx="40" lly="-15" urx="563" ury="638"/>
+ <char name="egrave" width="600" llx="40" lly="-15" urx="563" ury="661"/>
+ <char name="eight" width="600" llx="83" lly="-15" urx="517" ury="616"/>
+ <char name="ellipsis" width="600" llx="26" lly="-15" urx="574" ury="116"/>
+ <char name="emdash" width="600" llx="-10" lly="203" urx="610" ury="313"/>
+ <char name="endash" width="600" llx="65" lly="203" urx="535" ury="313"/>
+ <char name="equal" width="600" llx="71" lly="118" urx="529" ury="398"/>
+ <char name="eth" width="600" llx="58" lly="-27" urx="543" ury="626"/>
+ <char name="exclam" width="600" llx="202" lly="-15" urx="398" ury="572"/>
+ <char name="exclamdown" width="600" llx="202" lly="-146" urx="398" ury="449"/>
+ <char name="f" width="600" llx="83" lly="0" urx="547" ury="626"/>
+ <char name="fi" width="600" llx="12" lly="0" urx="593" ury="626"/>
+ <char name="five" width="600" llx="70" lly="-15" urx="521" ury="601"/>
+ <char name="fl" width="600" llx="12" lly="0" urx="593" ury="626"/>
+ <char name="florin" width="600" llx="-30" lly="-131" urx="572" ury="616"/>
+ <char name="four" width="600" llx="53" lly="0" urx="507" ury="616"/>
+ <char name="fraction" width="600" llx="25" lly="-60" urx="576" ury="661"/>
+ <char name="g" width="600" llx="30" lly="-146" urx="580" ury="454"/>
+ <char name="germandbls" width="600" llx="22" lly="-15" urx="596" ury="626"/>
+ <char name="grave" width="600" llx="132" lly="508" urx="395" ury="661"/>
+ <char name="greater" width="600" llx="77" lly="15" urx="534" ury="501"/>
+ <char name="guillemotleft" width="600" llx="8" lly="70" urx="553" ury="446"/>
+ <char name="guillemotright" width="600" llx="47" lly="70" urx="592" ury="446"/>
+ <char name="guilsinglleft" width="600" llx="141" lly="70" urx="459" ury="446"/>
+ <char name="guilsinglright" width="600" llx="141" lly="70" urx="459" ury="446"/>
+ <char name="h" width="600" llx="5" lly="0" urx="592" ury="626"/>
+ <char name="hungarumlaut" width="600" llx="68" lly="488" urx="588" ury="661"/>
+ <char name="hyphen" width="600" llx="100" lly="203" urx="500" ury="313"/>
+ <char name="i" width="600" llx="77" lly="0" urx="523" ury="658"/>
+ <char name="iacute" width="600" llx="77" lly="0" urx="523" ury="661"/>
+ <char name="icircumflex" width="600" llx="73" lly="0" urx="523" ury="657"/>
+ <char name="idieresis" width="600" llx="77" lly="0" urx="523" ury="618"/>
+ <char name="igrave" width="600" llx="77" lly="0" urx="523" ury="661"/>
+ <char name="j" width="600" llx="63" lly="-146" urx="440" ury="658"/>
+ <char name="k" width="600" llx="20" lly="0" urx="585" ury="626"/>
+ <char name="l" width="600" llx="77" lly="0" urx="523" ury="626"/>
+ <char name="less" width="600" llx="66" lly="15" urx="523" ury="501"/>
+ <char name="logicalnot" width="600" llx="71" lly="103" urx="529" ury="413"/>
+ <char name="lslash" width="600" llx="77" lly="0" urx="523" ury="626"/>
+ <char name="m" width="600" llx="-22" lly="0" urx="626" ury="454"/>
+ <char name="macron" width="600" llx="88" lly="505" urx="512" ury="585"/>
+ <char name="minus" width="600" llx="71" lly="203" urx="529" ury="313"/>
+ <char name="mu" width="600" llx="-1" lly="-142" urx="569" ury="439"/>
+ <char name="multiply" width="600" llx="81" lly="39" urx="520" ury="478"/>
+ <char name="n" width="600" llx="18" lly="0" urx="592" ury="454"/>
+ <char name="nine" width="600" llx="79" lly="-15" urx="510" ury="616"/>
+ <char name="ntilde" width="600" llx="18" lly="0" urx="592" ury="636"/>
+ <char name="numbersign" width="600" llx="56" lly="-45" urx="544" ury="651"/>
+ <char name="o" width="600" llx="30" lly="-15" urx="570" ury="454"/>
+ <char name="oacute" width="600" llx="30" lly="-15" urx="570" ury="661"/>
+ <char name="ocircumflex" width="600" llx="30" lly="-15" urx="570" ury="657"/>
+ <char name="odieresis" width="600" llx="30" lly="-15" urx="570" ury="638"/>
+ <char name="oe" width="600" llx="-18" lly="-15" urx="611" ury="454"/>
+ <char name="ogonek" width="600" llx="169" lly="-199" urx="400" ury="0"/>
+ <char name="ograve" width="600" llx="30" lly="-15" urx="570" ury="661"/>
+ <char name="one" width="600" llx="81" lly="0" urx="539" ury="616"/>
+ <char name="onehalf" width="600" llx="-47" lly="-60" urx="648" ury="661"/>
+ <char name="onequarter" width="600" llx="-56" lly="-60" urx="656" ury="661"/>
+ <char name="onesuperior" width="600" llx="153" lly="230" urx="447" ury="616"/>
+ <char name="ordfeminine" width="600" llx="147" lly="196" urx="453" ury="580"/>
+ <char name="ordmasculine" width="600" llx="147" lly="196" urx="453" ury="580"/>
+ <char name="oslash" width="600" llx="30" lly="-24" urx="570" ury="463"/>
+ <char name="otilde" width="600" llx="30" lly="-15" urx="570" ury="636"/>
+ <char name="p" width="600" llx="-1" lly="-142" urx="570" ury="454"/>
+ <char name="paragraph" width="600" llx="6" lly="-70" urx="576" ury="580"/>
+ <char name="parenleft" width="600" llx="219" lly="-102" urx="461" ury="616"/>
+ <char name="parenright" width="600" llx="139" lly="-102" urx="381" ury="616"/>
+ <char name="percent" width="600" llx="5" lly="-15" urx="595" ury="616"/>
+ <char name="period" width="600" llx="192" lly="-15" urx="408" ury="171"/>
+ <char name="periodcentered" width="600" llx="196" lly="165" urx="404" ury="351"/>
+ <char name="perthousand" width="600" llx="-113" lly="-15" urx="713" ury="616"/>
+ <char name="plus" width="600" llx="71" lly="39" urx="529" ury="478"/>
+ <char name="plusminus" width="600" llx="71" lly="24" urx="529" ury="515"/>
+ <char name="q" width="600" llx="20" lly="-142" urx="591" ury="454"/>
+ <char name="question" width="600" llx="98" lly="-14" urx="501" ury="580"/>
+ <char name="questiondown" width="600" llx="99" lly="-146" urx="502" ury="449"/>
+ <char name="quotedbl" width="600" llx="135" lly="277" urx="465" ury="562"/>
+ <char name="quotedblbase" width="600" llx="65" lly="-142" urx="529" ury="143"/>
+ <char name="quotedblleft" width="600" llx="71" lly="277" urx="535" ury="562"/>
+ <char name="quotedblright" width="600" llx="61" lly="277" urx="525" ury="562"/>
+ <char name="quoteleft" width="600" llx="178" lly="277" urx="428" ury="562"/>
+ <char name="quoteright" width="600" llx="171" lly="277" urx="423" ury="562"/>
+ <char name="quotesinglbase" width="600" llx="175" lly="-142" urx="427" ury="143"/>
+ <char name="quotesingle" width="600" llx="227" lly="277" urx="373" ury="562"/>
+ <char name="r" width="600" llx="47" lly="0" urx="580" ury="454"/>
+ <char name="registered" width="600" llx="0" lly="-18" urx="600" ury="580"/>
+ <char name="ring" width="600" llx="198" lly="481" urx="402" ury="678"/>
+ <char name="s" width="600" llx="68" lly="-17" urx="535" ury="459"/>
+ <char name="scaron" width="600" llx="68" lly="-17" urx="535" ury="667"/>
+ <char name="scedilla" width="600" llx="68" lly="-206" urx="535" ury="459"/>
+ <char name="section" width="600" llx="83" lly="-70" urx="517" ury="580"/>
+ <char name="semicolon" width="600" llx="123" lly="-111" urx="408" ury="425"/>
+ <char name="seven" width="600" llx="55" lly="0" urx="494" ury="601"/>
+ <char name="six" width="600" llx="90" lly="-15" urx="521" ury="616"/>
+ <char name="slash" width="600" llx="98" lly="-77" urx="502" ury="626"/>
+ <char name="space" width="600" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="600"/>
- <char name="square" width="600"/>
- <char name="sterling" width="600"/>
- <char name="stop" width="600"/>
- <char name="t" width="600"/>
- <char name="tab" width="600"/>
- <char name="thorn" width="600"/>
- <char name="three" width="600"/>
- <char name="threequarters" width="600"/>
- <char name="threesuperior" width="600"/>
- <char name="tilde" width="600"/>
- <char name="trademark" width="600"/>
- <char name="two" width="600"/>
- <char name="twosuperior" width="600"/>
- <char name="u" width="600"/>
- <char name="uacute" width="600"/>
- <char name="ucircumflex" width="600"/>
- <char name="udieresis" width="600"/>
- <char name="ugrave" width="600"/>
- <char name="underscore" width="600"/>
- <char name="up" width="600"/>
- <char name="v" width="600"/>
- <char name="w" width="600"/>
- <char name="x" width="600"/>
- <char name="y" width="600"/>
- <char name="yacute" width="600"/>
- <char name="ydieresis" width="600"/>
- <char name="yen" width="600"/>
- <char name="z" width="600"/>
- <char name="zcaron" width="600"/>
- <char name="zero" width="600"/>
- </widths>
-</font-metrics> \ No newline at end of file
+ <char name="sterling" width="600" llx="72" lly="-28" urx="558" ury="611"/>
+ <char name="t" width="600" llx="47" lly="-15" urx="532" ury="562"/>
+ <char name="thorn" width="600" llx="-14" lly="-142" urx="570" ury="626"/>
+ <char name="three" width="600" llx="63" lly="-15" urx="501" ury="616"/>
+ <char name="threequarters" width="600" llx="-47" lly="-60" urx="648" ury="661"/>
+ <char name="threesuperior" width="600" llx="138" lly="222" urx="433" ury="616"/>
+ <char name="tilde" width="600" llx="89" lly="493" urx="512" ury="636"/>
+ <char name="trademark" width="600" llx="-9" lly="230" urx="749" ury="562"/>
+ <char name="two" width="600" llx="61" lly="0" urx="499" ury="616"/>
+ <char name="twosuperior" width="600" llx="143" lly="230" urx="436" ury="616"/>
+ <char name="u" width="600" llx="-1" lly="-15" urx="569" ury="439"/>
+ <char name="uacute" width="600" llx="-1" lly="-15" urx="569" ury="661"/>
+ <char name="ucircumflex" width="600" llx="-1" lly="-15" urx="569" ury="657"/>
+ <char name="udieresis" width="600" llx="-1" lly="-15" urx="569" ury="638"/>
+ <char name="ugrave" width="600" llx="-1" lly="-15" urx="569" ury="661"/>
+ <char name="underscore" width="600" llx="0" lly="-125" urx="600" ury="-75"/>
+ <char name="v" width="600" llx="-1" lly="0" urx="601" ury="439"/>
+ <char name="w" width="600" llx="-18" lly="0" urx="618" ury="439"/>
+ <char name="x" width="600" llx="6" lly="0" urx="594" ury="439"/>
+ <char name="y" width="600" llx="-4" lly="-142" urx="601" ury="439"/>
+ <char name="yacute" width="600" llx="-4" lly="-142" urx="601" ury="661"/>
+ <char name="ydieresis" width="600" llx="-4" lly="-142" urx="601" ury="638"/>
+ <char name="yen" width="600" llx="10" lly="0" urx="590" ury="562"/>
+ <char name="z" width="600" llx="81" lly="0" urx="520" ury="439"/>
+ <char name="zcaron" width="600" llx="81" lly="0" urx="520" ury="667"/>
+ <char name="zero" width="600" llx="87" lly="-15" urx="513" ury="616"/>
+ </char-metrics>
+</font-metrics>
diff --git a/src/codegen/fonts/CourierBoldOblique.xml b/src/codegen/fonts/CourierBoldOblique.xml
index 5bd26092e..a82a75ad2 100644
--- a/src/codegen/fonts/CourierBoldOblique.xml
+++ b/src/codegen/fonts/CourierBoldOblique.xml
@@ -22,275 +22,246 @@
<family-name>Courier</family-name>
<class-name>CourierBoldOblique</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>562</cap-height>
<x-height>439</x-height>
<ascender>626</ascender>
<descender>-142</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="600"/>
- <char name="AE" width="600"/>
- <char name="Aacute" width="600"/>
- <char name="Acircumflex" width="600"/>
- <char name="Adieresis" width="600"/>
- <char name="Agrave" width="600"/>
- <char name="Aring" width="600"/>
- <char name="Atilde" width="600"/>
- <char name="B" width="600"/>
- <char name="C" width="600"/>
- <char name="Ccedilla" width="600"/>
- <char name="D" width="600"/>
- <char name="E" width="600"/>
- <char name="Eacute" width="600"/>
- <char name="Ecircumflex" width="600"/>
- <char name="Edieresis" width="600"/>
- <char name="Egrave" width="600"/>
- <char name="Eth" width="600"/>
- <char name="Euro" width="600"/>
- <char name="F" width="600"/>
- <char name="G" width="600"/>
- <char name="Gcaron" width="600"/>
- <char name="H" width="600"/>
- <char name="I" width="600"/>
- <char name="IJ" width="600"/>
- <char name="Iacute" width="600"/>
- <char name="Icircumflex" width="600"/>
- <char name="Idieresis" width="600"/>
- <char name="Idot" width="600"/>
- <char name="Igrave" width="600"/>
- <char name="J" width="600"/>
- <char name="K" width="600"/>
- <char name="L" width="600"/>
- <char name="LL" width="600"/>
- <char name="Lslash" width="600"/>
- <char name="M" width="600"/>
- <char name="N" width="600"/>
- <char name="Ntilde" width="600"/>
- <char name="O" width="600"/>
- <char name="OE" width="600"/>
- <char name="Oacute" width="600"/>
- <char name="Ocircumflex" width="600"/>
- <char name="Odieresis" width="600"/>
- <char name="Ograve" width="600"/>
- <char name="Oslash" width="600"/>
- <char name="Otilde" width="600"/>
- <char name="P" width="600"/>
- <char name="Q" width="600"/>
- <char name="R" width="600"/>
- <char name="S" width="600"/>
- <char name="Scaron" width="600"/>
- <char name="Scedilla" width="600"/>
- <char name="T" width="600"/>
- <char name="Thorn" width="600"/>
- <char name="U" width="600"/>
- <char name="Uacute" width="600"/>
- <char name="Ucircumflex" width="600"/>
- <char name="Udieresis" width="600"/>
- <char name="Ugrave" width="600"/>
- <char name="V" width="600"/>
- <char name="W" width="600"/>
- <char name="X" width="600"/>
- <char name="Y" width="600"/>
- <char name="Yacute" width="600"/>
- <char name="Ydieresis" width="600"/>
- <char name="Z" width="600"/>
- <char name="Zcaron" width="600"/>
- <char name="a" width="600"/>
- <char name="aacute" width="600"/>
- <char name="acircumflex" width="600"/>
- <char name="acute" width="600"/>
- <char name="adieresis" width="600"/>
- <char name="ae" width="600"/>
- <char name="agrave" width="600"/>
- <char name="ampersand" width="600"/>
- <char name="aring" width="600"/>
- <char name="arrowboth" width="600"/>
- <char name="arrowdown" width="600"/>
- <char name="arrowleft" width="600"/>
- <char name="arrowright" width="600"/>
- <char name="arrowup" width="600"/>
- <char name="asciicircum" width="600"/>
- <char name="asciitilde" width="600"/>
- <char name="asterisk" width="600"/>
- <char name="at" width="600"/>
- <char name="atilde" width="600"/>
- <char name="b" width="600"/>
- <char name="backslash" width="600"/>
- <char name="bar" width="600"/>
- <char name="braceleft" width="600"/>
- <char name="braceright" width="600"/>
- <char name="bracketleft" width="600"/>
- <char name="bracketright" width="600"/>
- <char name="breve" width="600"/>
- <char name="brokenbar" width="600"/>
- <char name="bullet" width="600"/>
- <char name="c" width="600"/>
- <char name="caron" width="600"/>
- <char name="ccedilla" width="600"/>
- <char name="cedilla" width="600"/>
- <char name="cent" width="600"/>
- <char name="center" width="600"/>
- <char name="circumflex" width="600"/>
- <char name="colon" width="600"/>
- <char name="comma" width="600"/>
- <char name="copyright" width="600"/>
- <char name="currency" width="600"/>
- <char name="d" width="600"/>
- <char name="dagger" width="600"/>
- <char name="daggerdbl" width="600"/>
- <char name="dectab" width="600"/>
- <char name="degree" width="600"/>
- <char name="dieresis" width="600"/>
- <char name="divide" width="600"/>
- <char name="dollar" width="600"/>
- <char name="dotaccent" width="600"/>
- <char name="dotlessi" width="600"/>
- <char name="down" width="600"/>
- <char name="e" width="600"/>
- <char name="eacute" width="600"/>
- <char name="ecircumflex" width="600"/>
- <char name="edieresis" width="600"/>
- <char name="egrave" width="600"/>
- <char name="eight" width="600"/>
- <char name="ellipsis" width="600"/>
- <char name="emdash" width="600"/>
- <char name="endash" width="600"/>
- <char name="equal" width="600"/>
- <char name="eth" width="600"/>
- <char name="exclam" width="600"/>
- <char name="exclamdown" width="600"/>
- <char name="f" width="600"/>
- <char name="fi" width="600"/>
- <char name="five" width="600"/>
- <char name="fl" width="600"/>
- <char name="florin" width="600"/>
- <char name="format" width="600"/>
- <char name="four" width="600"/>
- <char name="fraction" width="600"/>
- <char name="g" width="600"/>
- <char name="gcaron" width="600"/>
- <char name="germandbls" width="600"/>
- <char name="grave" width="600"/>
- <char name="graybox" width="600"/>
- <char name="greater" width="600"/>
- <char name="guillemotleft" width="600"/>
- <char name="guillemotright" width="600"/>
- <char name="guilsinglleft" width="600"/>
- <char name="guilsinglright" width="600"/>
- <char name="h" width="600"/>
- <char name="hungarumlaut" width="600"/>
- <char name="hyphen" width="600"/>
- <char name="i" width="600"/>
- <char name="iacute" width="600"/>
- <char name="icircumflex" width="600"/>
- <char name="idieresis" width="600"/>
- <char name="igrave" width="600"/>
- <char name="ij" width="600"/>
- <char name="indent" width="600"/>
- <char name="j" width="600"/>
- <char name="k" width="600"/>
- <char name="l" width="600"/>
- <char name="largebullet" width="600"/>
- <char name="left" width="600"/>
- <char name="less" width="600"/>
- <char name="lira" width="600"/>
- <char name="ll" width="600"/>
- <char name="logicalnot" width="600"/>
- <char name="lslash" width="600"/>
- <char name="m" width="600"/>
- <char name="macron" width="600"/>
- <char name="merge" width="600"/>
- <char name="minus" width="600"/>
- <char name="mu" width="600"/>
- <char name="multiply" width="600"/>
- <char name="n" width="600"/>
- <char name="nine" width="600"/>
- <char name="notegraphic" width="600"/>
- <char name="ntilde" width="600"/>
- <char name="numbersign" width="600"/>
- <char name="o" width="600"/>
- <char name="oacute" width="600"/>
- <char name="ocircumflex" width="600"/>
- <char name="odieresis" width="600"/>
- <char name="oe" width="600"/>
- <char name="ogonek" width="600"/>
- <char name="ograve" width="600"/>
- <char name="one" width="600"/>
- <char name="onehalf" width="600"/>
- <char name="onequarter" width="600"/>
- <char name="onesuperior" width="600"/>
- <char name="ordfeminine" width="600"/>
- <char name="ordmasculine" width="600"/>
- <char name="oslash" width="600"/>
- <char name="otilde" width="600"/>
- <char name="overscore" width="600"/>
- <char name="p" width="600"/>
- <char name="paragraph" width="600"/>
- <char name="parenleft" width="600"/>
- <char name="parenright" width="600"/>
- <char name="percent" width="600"/>
- <char name="period" width="600"/>
- <char name="periodcentered" width="600"/>
- <char name="perthousand" width="600"/>
- <char name="plus" width="600"/>
- <char name="plusminus" width="600"/>
- <char name="prescription" width="600"/>
- <char name="q" width="600"/>
- <char name="question" width="600"/>
- <char name="questiondown" width="600"/>
- <char name="quotedbl" width="600"/>
- <char name="quotedblbase" width="600"/>
- <char name="quotedblleft" width="600"/>
- <char name="quotedblright" width="600"/>
- <char name="quoteleft" width="600"/>
- <char name="quoteright" width="600"/>
- <char name="quotesinglbase" width="600"/>
- <char name="quotesingle" width="600"/>
- <char name="r" width="600"/>
- <char name="registered" width="600"/>
- <char name="return" width="600"/>
- <char name="ring" width="600"/>
- <char name="s" width="600"/>
- <char name="scaron" width="600"/>
- <char name="scedilla" width="600"/>
- <char name="section" width="600"/>
- <char name="semicolon" width="600"/>
- <char name="seven" width="600"/>
- <char name="six" width="600"/>
- <char name="slash" width="600"/>
- <char name="space" width="600"/>
+ <char-metrics>
+ <char name="A" width="600" llx="-9" lly="0" urx="632" ury="562"/>
+ <char name="AE" width="600" llx="-29" lly="0" urx="708" ury="562"/>
+ <char name="Aacute" width="600" llx="-9" lly="0" urx="655" ury="784"/>
+ <char name="Acircumflex" width="600" llx="-9" lly="0" urx="632" ury="780"/>
+ <char name="Adieresis" width="600" llx="-9" lly="0" urx="632" ury="761"/>
+ <char name="Agrave" width="600" llx="-9" lly="0" urx="632" ury="784"/>
+ <char name="Aring" width="600" llx="-9" lly="0" urx="632" ury="801"/>
+ <char name="Atilde" width="600" llx="-9" lly="0" urx="669" ury="759"/>
+ <char name="B" width="600" llx="30" lly="0" urx="630" ury="562"/>
+ <char name="C" width="600" llx="74" lly="-18" urx="675" ury="580"/>
+ <char name="Ccedilla" width="600" llx="74" lly="-206" urx="675" ury="580"/>
+ <char name="D" width="600" llx="30" lly="0" urx="664" ury="562"/>
+ <char name="E" width="600" llx="25" lly="0" urx="670" ury="562"/>
+ <char name="Eacute" width="600" llx="25" lly="0" urx="670" ury="784"/>
+ <char name="Ecircumflex" width="600" llx="25" lly="0" urx="670" ury="780"/>
+ <char name="Edieresis" width="600" llx="25" lly="0" urx="670" ury="761"/>
+ <char name="Egrave" width="600" llx="25" lly="0" urx="670" ury="784"/>
+ <char name="Eth" width="600" llx="30" lly="0" urx="664" ury="562"/>
+ <char name="Euro" width="600" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="600" llx="39" lly="0" urx="684" ury="562"/>
+ <char name="G" width="600" llx="74" lly="-18" urx="675" ury="580"/>
+ <char name="H" width="600" llx="20" lly="0" urx="700" ury="562"/>
+ <char name="I" width="600" llx="77" lly="0" urx="643" ury="562"/>
+ <char name="Iacute" width="600" llx="77" lly="0" urx="643" ury="784"/>
+ <char name="Icircumflex" width="600" llx="77" lly="0" urx="643" ury="780"/>
+ <char name="Idieresis" width="600" llx="77" lly="0" urx="643" ury="761"/>
+ <char name="Igrave" width="600" llx="77" lly="0" urx="643" ury="784"/>
+ <char name="J" width="600" llx="58" lly="-18" urx="721" ury="562"/>
+ <char name="K" width="600" llx="21" lly="0" urx="692" ury="562"/>
+ <char name="L" width="600" llx="39" lly="0" urx="636" ury="562"/>
+ <char name="Lslash" width="600" llx="39" lly="0" urx="636" ury="562"/>
+ <char name="M" width="600" llx="-2" lly="0" urx="722" ury="562"/>
+ <char name="N" width="600" llx="8" lly="-12" urx="730" ury="562"/>
+ <char name="Ntilde" width="600" llx="8" lly="-12" urx="730" ury="759"/>
+ <char name="O" width="600" llx="74" lly="-18" urx="645" ury="580"/>
+ <char name="OE" width="600" llx="26" lly="0" urx="701" ury="562"/>
+ <char name="Oacute" width="600" llx="74" lly="-18" urx="645" ury="784"/>
+ <char name="Ocircumflex" width="600" llx="74" lly="-18" urx="645" ury="780"/>
+ <char name="Odieresis" width="600" llx="74" lly="-18" urx="645" ury="761"/>
+ <char name="Ograve" width="600" llx="74" lly="-18" urx="645" ury="784"/>
+ <char name="Oslash" width="600" llx="48" lly="-22" urx="673" ury="584"/>
+ <char name="Otilde" width="600" llx="74" lly="-18" urx="669" ury="759"/>
+ <char name="P" width="600" llx="48" lly="0" urx="643" ury="562"/>
+ <char name="Q" width="600" llx="83" lly="-138" urx="636" ury="580"/>
+ <char name="R" width="600" llx="24" lly="0" urx="617" ury="562"/>
+ <char name="S" width="600" llx="54" lly="-22" urx="673" ury="582"/>
+ <char name="Scaron" width="600" llx="54" lly="-22" urx="689" ury="790"/>
+ <char name="Scedilla" width="600" llx="54" lly="-206" urx="673" ury="582"/>
+ <char name="T" width="600" llx="86" lly="0" urx="679" ury="562"/>
+ <char name="Thorn" width="600" llx="48" lly="0" urx="620" ury="562"/>
+ <char name="U" width="600" llx="101" lly="-18" urx="716" ury="562"/>
+ <char name="Uacute" width="600" llx="101" lly="-18" urx="716" ury="784"/>
+ <char name="Ucircumflex" width="600" llx="101" lly="-18" urx="716" ury="780"/>
+ <char name="Udieresis" width="600" llx="101" lly="-18" urx="716" ury="761"/>
+ <char name="Ugrave" width="600" llx="101" lly="-18" urx="716" ury="784"/>
+ <char name="V" width="600" llx="84" lly="0" urx="733" ury="562"/>
+ <char name="W" width="600" llx="79" lly="0" urx="738" ury="562"/>
+ <char name="X" width="600" llx="12" lly="0" urx="690" ury="562"/>
+ <char name="Y" width="600" llx="109" lly="0" urx="709" ury="562"/>
+ <char name="Yacute" width="600" llx="109" lly="0" urx="709" ury="784"/>
+ <char name="Ydieresis" width="600" llx="109" lly="0" urx="709" ury="761"/>
+ <char name="Z" width="600" llx="62" lly="0" urx="637" ury="562"/>
+ <char name="Zcaron" width="600" llx="62" lly="0" urx="659" ury="790"/>
+ <char name="a" width="600" llx="61" lly="-15" urx="593" ury="454"/>
+ <char name="aacute" width="600" llx="61" lly="-15" urx="609" ury="661"/>
+ <char name="acircumflex" width="600" llx="61" lly="-15" urx="607" ury="657"/>
+ <char name="acute" width="600" llx="312" lly="508" urx="609" ury="661"/>
+ <char name="adieresis" width="600" llx="61" lly="-15" urx="595" ury="638"/>
+ <char name="ae" width="600" llx="21" lly="-15" urx="652" ury="454"/>
+ <char name="agrave" width="600" llx="61" lly="-15" urx="593" ury="661"/>
+ <char name="ampersand" width="600" llx="61" lly="-15" urx="595" ury="543"/>
+ <char name="aring" width="600" llx="61" lly="-15" urx="593" ury="678"/>
+ <char name="asciicircum" width="600" llx="171" lly="250" urx="556" ury="616"/>
+ <char name="asciitilde" width="600" llx="120" lly="153" urx="590" ury="356"/>
+ <char name="asterisk" width="600" llx="179" lly="219" urx="598" ury="601"/>
+ <char name="at" width="600" llx="65" lly="-15" urx="642" ury="616"/>
+ <char name="atilde" width="600" llx="61" lly="-15" urx="643" ury="636"/>
+ <char name="b" width="600" llx="13" lly="-15" urx="636" ury="626"/>
+ <char name="backslash" width="600" llx="222" lly="-77" urx="496" ury="626"/>
+ <char name="bar" width="600" llx="201" lly="-250" urx="505" ury="750"/>
+ <char name="braceleft" width="600" llx="203" lly="-102" urx="595" ury="616"/>
+ <char name="braceright" width="600" llx="114" lly="-102" urx="506" ury="616"/>
+ <char name="bracketleft" width="600" llx="223" lly="-102" urx="606" ury="616"/>
+ <char name="bracketright" width="600" llx="103" lly="-102" urx="486" ury="616"/>
+ <char name="breve" width="600" llx="217" lly="468" urx="652" ury="631"/>
+ <char name="brokenbar" width="600" llx="217" lly="-175" urx="489" ury="675"/>
+ <char name="bullet" width="600" llx="196" lly="132" urx="523" ury="430"/>
+ <char name="c" width="600" llx="81" lly="-15" urx="631" ury="459"/>
+ <char name="caron" width="600" llx="238" lly="493" urx="633" ury="667"/>
+ <char name="ccedilla" width="600" llx="81" lly="-206" urx="631" ury="459"/>
+ <char name="cedilla" width="600" llx="168" lly="-206" urx="368" ury="0"/>
+ <char name="cent" width="600" llx="121" lly="-49" urx="605" ury="614"/>
+ <char name="circumflex" width="600" llx="212" lly="483" urx="607" ury="657"/>
+ <char name="colon" width="600" llx="205" lly="-15" urx="480" ury="425"/>
+ <char name="comma" width="600" llx="99" lly="-111" urx="430" ury="174"/>
+ <char name="copyright" width="600" llx="53" lly="-18" urx="667" ury="580"/>
+ <char name="currency" width="600" llx="77" lly="49" urx="644" ury="517"/>
+ <char name="d" width="600" llx="60" lly="-15" urx="645" ury="626"/>
+ <char name="dagger" width="600" llx="175" lly="-70" urx="586" ury="580"/>
+ <char name="daggerdbl" width="600" llx="121" lly="-70" urx="587" ury="580"/>
+ <char name="degree" width="600" llx="173" lly="243" urx="570" ury="616"/>
+ <char name="dieresis" width="600" llx="246" lly="498" urx="595" ury="638"/>
+ <char name="divide" width="600" llx="114" lly="16" urx="596" ury="500"/>
+ <char name="dollar" width="600" llx="87" lly="-126" urx="630" ury="666"/>
+ <char name="dotaccent" width="600" llx="348" lly="498" urx="493" ury="638"/>
+ <char name="dotlessi" width="600" llx="77" lly="0" urx="546" ury="439"/>
+ <char name="e" width="600" llx="81" lly="-15" urx="605" ury="454"/>
+ <char name="eacute" width="600" llx="81" lly="-15" urx="609" ury="661"/>
+ <char name="ecircumflex" width="600" llx="81" lly="-15" urx="607" ury="657"/>
+ <char name="edieresis" width="600" llx="81" lly="-15" urx="605" ury="638"/>
+ <char name="egrave" width="600" llx="81" lly="-15" urx="605" ury="661"/>
+ <char name="eight" width="600" llx="115" lly="-15" urx="604" ury="616"/>
+ <char name="ellipsis" width="600" llx="35" lly="-15" urx="587" ury="116"/>
+ <char name="emdash" width="600" llx="33" lly="203" urx="677" ury="313"/>
+ <char name="endash" width="600" llx="108" lly="203" urx="602" ury="313"/>
+ <char name="equal" width="600" llx="96" lly="118" urx="614" ury="398"/>
+ <char name="eth" width="600" llx="93" lly="-27" urx="661" ury="626"/>
+ <char name="exclam" width="600" llx="215" lly="-15" urx="495" ury="572"/>
+ <char name="exclamdown" width="600" llx="196" lly="-146" urx="477" ury="449"/>
+ <char name="f" width="600" llx="83" lly="0" urx="677" ury="626"/>
+ <char name="fi" width="600" llx="12" lly="0" urx="644" ury="626"/>
+ <char name="five" width="600" llx="77" lly="-15" urx="621" ury="601"/>
+ <char name="fl" width="600" llx="12" lly="0" urx="644" ury="626"/>
+ <char name="florin" width="600" llx="-57" lly="-131" urx="702" ury="616"/>
+ <char name="four" width="600" llx="81" lly="0" urx="559" ury="616"/>
+ <char name="fraction" width="600" llx="22" lly="-60" urx="708" ury="661"/>
+ <char name="g" width="600" llx="40" lly="-146" urx="674" ury="454"/>
+ <char name="germandbls" width="600" llx="22" lly="-15" urx="629" ury="626"/>
+ <char name="grave" width="600" llx="272" lly="508" urx="503" ury="661"/>
+ <char name="greater" width="600" llx="97" lly="15" urx="589" ury="501"/>
+ <char name="guillemotleft" width="600" llx="62" lly="70" urx="639" ury="446"/>
+ <char name="guillemotright" width="600" llx="71" lly="70" urx="647" ury="446"/>
+ <char name="guilsinglleft" width="600" llx="195" lly="70" urx="545" ury="446"/>
+ <char name="guilsinglright" width="600" llx="165" lly="70" urx="514" ury="446"/>
+ <char name="h" width="600" llx="18" lly="0" urx="615" ury="626"/>
+ <char name="hungarumlaut" width="600" llx="171" lly="488" urx="729" ury="661"/>
+ <char name="hyphen" width="600" llx="143" lly="203" urx="567" ury="313"/>
+ <char name="i" width="600" llx="77" lly="0" urx="546" ury="658"/>
+ <char name="iacute" width="600" llx="77" lly="0" urx="609" ury="661"/>
+ <char name="icircumflex" width="600" llx="77" lly="0" urx="577" ury="657"/>
+ <char name="idieresis" width="600" llx="77" lly="0" urx="561" ury="618"/>
+ <char name="igrave" width="600" llx="77" lly="0" urx="546" ury="661"/>
+ <char name="j" width="600" llx="36" lly="-146" urx="580" ury="658"/>
+ <char name="k" width="600" llx="33" lly="0" urx="643" ury="626"/>
+ <char name="l" width="600" llx="77" lly="0" urx="546" ury="626"/>
+ <char name="less" width="600" llx="120" lly="15" urx="613" ury="501"/>
+ <char name="logicalnot" width="600" llx="135" lly="103" urx="617" ury="413"/>
+ <char name="lslash" width="600" llx="77" lly="0" urx="587" ury="626"/>
+ <char name="m" width="600" llx="-22" lly="0" urx="649" ury="454"/>
+ <char name="macron" width="600" llx="195" lly="505" urx="637" ury="585"/>
+ <char name="minus" width="600" llx="114" lly="203" urx="596" ury="313"/>
+ <char name="mu" width="600" llx="49" lly="-142" urx="592" ury="439"/>
+ <char name="multiply" width="600" llx="104" lly="39" urx="606" ury="478"/>
+ <char name="n" width="600" llx="18" lly="0" urx="615" ury="454"/>
+ <char name="nine" width="600" llx="75" lly="-15" urx="592" ury="616"/>
+ <char name="ntilde" width="600" llx="18" lly="0" urx="643" ury="636"/>
+ <char name="numbersign" width="600" llx="88" lly="-45" urx="641" ury="651"/>
+ <char name="o" width="600" llx="71" lly="-15" urx="622" ury="454"/>
+ <char name="oacute" width="600" llx="71" lly="-15" urx="649" ury="661"/>
+ <char name="ocircumflex" width="600" llx="71" lly="-15" urx="622" ury="657"/>
+ <char name="odieresis" width="600" llx="71" lly="-15" urx="622" ury="638"/>
+ <char name="oe" width="600" llx="18" lly="-15" urx="662" ury="454"/>
+ <char name="ogonek" width="600" llx="143" lly="-199" urx="367" ury="0"/>
+ <char name="ograve" width="600" llx="71" lly="-15" urx="622" ury="661"/>
+ <char name="one" width="600" llx="93" lly="0" urx="562" ury="616"/>
+ <char name="onehalf" width="600" llx="22" lly="-60" urx="716" ury="661"/>
+ <char name="onequarter" width="600" llx="13" lly="-60" urx="707" ury="661"/>
+ <char name="onesuperior" width="600" llx="212" lly="230" urx="514" ury="616"/>
+ <char name="ordfeminine" width="600" llx="188" lly="196" urx="526" ury="580"/>
+ <char name="ordmasculine" width="600" llx="188" lly="196" urx="543" ury="580"/>
+ <char name="oslash" width="600" llx="54" lly="-24" urx="638" ury="463"/>
+ <char name="otilde" width="600" llx="71" lly="-15" urx="643" ury="636"/>
+ <char name="p" width="600" llx="-32" lly="-142" urx="622" ury="454"/>
+ <char name="paragraph" width="600" llx="61" lly="-70" urx="700" ury="580"/>
+ <char name="parenleft" width="600" llx="265" lly="-102" urx="592" ury="616"/>
+ <char name="parenright" width="600" llx="117" lly="-102" urx="444" ury="616"/>
+ <char name="percent" width="600" llx="101" lly="-15" urx="625" ury="616"/>
+ <char name="period" width="600" llx="206" lly="-15" urx="427" ury="171"/>
+ <char name="periodcentered" width="600" llx="248" lly="165" urx="461" ury="351"/>
+ <char name="perthousand" width="600" llx="-45" lly="-15" urx="743" ury="616"/>
+ <char name="plus" width="600" llx="114" lly="39" urx="596" ury="478"/>
+ <char name="plusminus" width="600" llx="76" lly="24" urx="614" ury="515"/>
+ <char name="q" width="600" llx="60" lly="-142" urx="685" ury="454"/>
+ <char name="question" width="600" llx="183" lly="-14" urx="592" ury="580"/>
+ <char name="questiondown" width="600" llx="100" lly="-146" urx="509" ury="449"/>
+ <char name="quotedbl" width="600" llx="211" lly="277" urx="585" ury="562"/>
+ <char name="quotedblbase" width="600" llx="34" lly="-142" urx="560" ury="143"/>
+ <char name="quotedblleft" width="600" llx="190" lly="277" urx="594" ury="562"/>
+ <char name="quotedblright" width="600" llx="119" lly="277" urx="645" ury="562"/>
+ <char name="quoteleft" width="600" llx="297" lly="277" urx="487" ury="562"/>
+ <char name="quoteright" width="600" llx="229" lly="277" urx="543" ury="562"/>
+ <char name="quotesinglbase" width="600" llx="144" lly="-142" urx="458" ury="143"/>
+ <char name="quotesingle" width="600" llx="303" lly="277" urx="493" ury="562"/>
+ <char name="r" width="600" llx="47" lly="0" urx="655" ury="454"/>
+ <char name="registered" width="600" llx="53" lly="-18" urx="667" ury="580"/>
+ <char name="ring" width="600" llx="319" lly="481" urx="528" ury="678"/>
+ <char name="s" width="600" llx="66" lly="-17" urx="608" ury="459"/>
+ <char name="scaron" width="600" llx="66" lly="-17" urx="633" ury="667"/>
+ <char name="scedilla" width="600" llx="66" lly="-206" urx="608" ury="459"/>
+ <char name="section" width="600" llx="74" lly="-70" urx="620" ury="580"/>
+ <char name="semicolon" width="600" llx="99" lly="-111" urx="481" ury="425"/>
+ <char name="seven" width="600" llx="147" lly="0" urx="622" ury="601"/>
+ <char name="six" width="600" llx="135" lly="-15" urx="652" ury="616"/>
+ <char name="slash" width="600" llx="90" lly="-77" urx="626" ury="626"/>
+ <char name="space" width="600" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="600"/>
- <char name="square" width="600"/>
- <char name="sterling" width="600"/>
- <char name="stop" width="600"/>
- <char name="t" width="600"/>
- <char name="tab" width="600"/>
- <char name="thorn" width="600"/>
- <char name="three" width="600"/>
- <char name="threequarters" width="600"/>
- <char name="threesuperior" width="600"/>
- <char name="tilde" width="600"/>
- <char name="trademark" width="600"/>
- <char name="two" width="600"/>
- <char name="twosuperior" width="600"/>
- <char name="u" width="600"/>
- <char name="uacute" width="600"/>
- <char name="ucircumflex" width="600"/>
- <char name="udieresis" width="600"/>
- <char name="ugrave" width="600"/>
- <char name="underscore" width="600"/>
- <char name="up" width="600"/>
- <char name="v" width="600"/>
- <char name="w" width="600"/>
- <char name="x" width="600"/>
- <char name="y" width="600"/>
- <char name="yacute" width="600"/>
- <char name="ydieresis" width="600"/>
- <char name="yen" width="600"/>
- <char name="z" width="600"/>
- <char name="zcaron" width="600"/>
- <char name="zero" width="600"/>
- </widths>
+ <char name="sterling" width="600" llx="106" lly="-28" urx="650" ury="611"/>
+ <char name="t" width="600" llx="118" lly="-15" urx="567" ury="562"/>
+ <char name="thorn" width="600" llx="-32" lly="-142" urx="622" ury="626"/>
+ <char name="three" width="600" llx="71" lly="-15" urx="571" ury="616"/>
+ <char name="threequarters" width="600" llx="8" lly="-60" urx="699" ury="661"/>
+ <char name="threesuperior" width="600" llx="193" lly="222" urx="526" ury="616"/>
+ <char name="tilde" width="600" llx="199" lly="493" urx="643" ury="636"/>
+ <char name="trademark" width="600" llx="86" lly="230" urx="869" ury="562"/>
+ <char name="two" width="600" llx="61" lly="0" urx="594" ury="616"/>
+ <char name="twosuperior" width="600" llx="191" lly="230" urx="542" ury="616"/>
+ <char name="u" width="600" llx="70" lly="-15" urx="592" ury="439"/>
+ <char name="uacute" width="600" llx="70" lly="-15" urx="599" ury="661"/>
+ <char name="ucircumflex" width="600" llx="70" lly="-15" urx="597" ury="657"/>
+ <char name="udieresis" width="600" llx="70" lly="-15" urx="595" ury="638"/>
+ <char name="ugrave" width="600" llx="70" lly="-15" urx="592" ury="661"/>
+ <char name="underscore" width="600" llx="-27" lly="-125" urx="585" ury="-75"/>
+ <char name="v" width="600" llx="70" lly="0" urx="695" ury="439"/>
+ <char name="w" width="600" llx="53" lly="0" urx="712" ury="439"/>
+ <char name="x" width="600" llx="6" lly="0" urx="671" ury="439"/>
+ <char name="y" width="600" llx="-21" lly="-142" urx="695" ury="439"/>
+ <char name="yacute" width="600" llx="-21" lly="-142" urx="695" ury="661"/>
+ <char name="ydieresis" width="600" llx="-21" lly="-142" urx="695" ury="638"/>
+ <char name="yen" width="600" llx="98" lly="0" urx="710" ury="562"/>
+ <char name="z" width="600" llx="81" lly="0" urx="614" ury="439"/>
+ <char name="zcaron" width="600" llx="81" lly="0" urx="643" ury="667"/>
+ <char name="zero" width="600" llx="135" lly="-15" urx="593" ury="616"/>
+ </char-metrics>
</font-metrics>
diff --git a/src/codegen/fonts/CourierOblique.xml b/src/codegen/fonts/CourierOblique.xml
index 2dc300645..e08fa9b9b 100644
--- a/src/codegen/fonts/CourierOblique.xml
+++ b/src/codegen/fonts/CourierOblique.xml
@@ -22,275 +22,246 @@
<family-name>Courier</family-name>
<class-name>CourierOblique</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>562</cap-height>
<x-height>426</x-height>
<ascender>629</ascender>
<descender>-157</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="600"/>
- <char name="AE" width="600"/>
- <char name="Aacute" width="600"/>
- <char name="Acircumflex" width="600"/>
- <char name="Adieresis" width="600"/>
- <char name="Agrave" width="600"/>
- <char name="Aring" width="600"/>
- <char name="Atilde" width="600"/>
- <char name="B" width="600"/>
- <char name="C" width="600"/>
- <char name="Ccedilla" width="600"/>
- <char name="D" width="600"/>
- <char name="E" width="600"/>
- <char name="Eacute" width="600"/>
- <char name="Ecircumflex" width="600"/>
- <char name="Edieresis" width="600"/>
- <char name="Egrave" width="600"/>
- <char name="Eth" width="600"/>
- <char name="Euro" width="600"/>
- <char name="F" width="600"/>
- <char name="G" width="600"/>
- <char name="Gcaron" width="600"/>
- <char name="H" width="600"/>
- <char name="I" width="600"/>
- <char name="IJ" width="600"/>
- <char name="Iacute" width="600"/>
- <char name="Icircumflex" width="600"/>
- <char name="Idieresis" width="600"/>
- <char name="Idot" width="600"/>
- <char name="Igrave" width="600"/>
- <char name="J" width="600"/>
- <char name="K" width="600"/>
- <char name="L" width="600"/>
- <char name="LL" width="600"/>
- <char name="Lslash" width="600"/>
- <char name="M" width="600"/>
- <char name="N" width="600"/>
- <char name="Ntilde" width="600"/>
- <char name="O" width="600"/>
- <char name="OE" width="600"/>
- <char name="Oacute" width="600"/>
- <char name="Ocircumflex" width="600"/>
- <char name="Odieresis" width="600"/>
- <char name="Ograve" width="600"/>
- <char name="Oslash" width="600"/>
- <char name="Otilde" width="600"/>
- <char name="P" width="600"/>
- <char name="Q" width="600"/>
- <char name="R" width="600"/>
- <char name="S" width="600"/>
- <char name="Scaron" width="600"/>
- <char name="Scedilla" width="600"/>
- <char name="T" width="600"/>
- <char name="Thorn" width="600"/>
- <char name="U" width="600"/>
- <char name="Uacute" width="600"/>
- <char name="Ucircumflex" width="600"/>
- <char name="Udieresis" width="600"/>
- <char name="Ugrave" width="600"/>
- <char name="V" width="600"/>
- <char name="W" width="600"/>
- <char name="X" width="600"/>
- <char name="Y" width="600"/>
- <char name="Yacute" width="600"/>
- <char name="Ydieresis" width="600"/>
- <char name="Z" width="600"/>
- <char name="Zcaron" width="600"/>
- <char name="a" width="600"/>
- <char name="aacute" width="600"/>
- <char name="acircumflex" width="600"/>
- <char name="acute" width="600"/>
- <char name="adieresis" width="600"/>
- <char name="ae" width="600"/>
- <char name="agrave" width="600"/>
- <char name="ampersand" width="600"/>
- <char name="aring" width="600"/>
- <char name="arrowboth" width="600"/>
- <char name="arrowdown" width="600"/>
- <char name="arrowleft" width="600"/>
- <char name="arrowright" width="600"/>
- <char name="arrowup" width="600"/>
- <char name="asciicircum" width="600"/>
- <char name="asciitilde" width="600"/>
- <char name="asterisk" width="600"/>
- <char name="at" width="600"/>
- <char name="atilde" width="600"/>
- <char name="b" width="600"/>
- <char name="backslash" width="600"/>
- <char name="bar" width="600"/>
- <char name="braceleft" width="600"/>
- <char name="braceright" width="600"/>
- <char name="bracketleft" width="600"/>
- <char name="bracketright" width="600"/>
- <char name="breve" width="600"/>
- <char name="brokenbar" width="600"/>
- <char name="bullet" width="600"/>
- <char name="c" width="600"/>
- <char name="caron" width="600"/>
- <char name="ccedilla" width="600"/>
- <char name="cedilla" width="600"/>
- <char name="cent" width="600"/>
- <char name="center" width="600"/>
- <char name="circumflex" width="600"/>
- <char name="colon" width="600"/>
- <char name="comma" width="600"/>
- <char name="copyright" width="600"/>
- <char name="currency" width="600"/>
- <char name="d" width="600"/>
- <char name="dagger" width="600"/>
- <char name="daggerdbl" width="600"/>
- <char name="dectab" width="600"/>
- <char name="degree" width="600"/>
- <char name="dieresis" width="600"/>
- <char name="divide" width="600"/>
- <char name="dollar" width="600"/>
- <char name="dotaccent" width="600"/>
- <char name="dotlessi" width="600"/>
- <char name="down" width="600"/>
- <char name="e" width="600"/>
- <char name="eacute" width="600"/>
- <char name="ecircumflex" width="600"/>
- <char name="edieresis" width="600"/>
- <char name="egrave" width="600"/>
- <char name="eight" width="600"/>
- <char name="ellipsis" width="600"/>
- <char name="emdash" width="600"/>
- <char name="endash" width="600"/>
- <char name="equal" width="600"/>
- <char name="eth" width="600"/>
- <char name="exclam" width="600"/>
- <char name="exclamdown" width="600"/>
- <char name="f" width="600"/>
- <char name="fi" width="600"/>
- <char name="five" width="600"/>
- <char name="fl" width="600"/>
- <char name="florin" width="600"/>
- <char name="format" width="600"/>
- <char name="four" width="600"/>
- <char name="fraction" width="600"/>
- <char name="g" width="600"/>
- <char name="gcaron" width="600"/>
- <char name="germandbls" width="600"/>
- <char name="grave" width="600"/>
- <char name="graybox" width="600"/>
- <char name="greater" width="600"/>
- <char name="guillemotleft" width="600"/>
- <char name="guillemotright" width="600"/>
- <char name="guilsinglleft" width="600"/>
- <char name="guilsinglright" width="600"/>
- <char name="h" width="600"/>
- <char name="hungarumlaut" width="600"/>
- <char name="hyphen" width="600"/>
- <char name="i" width="600"/>
- <char name="iacute" width="600"/>
- <char name="icircumflex" width="600"/>
- <char name="idieresis" width="600"/>
- <char name="igrave" width="600"/>
- <char name="ij" width="600"/>
- <char name="indent" width="600"/>
- <char name="j" width="600"/>
- <char name="k" width="600"/>
- <char name="l" width="600"/>
- <char name="largebullet" width="600"/>
- <char name="left" width="600"/>
- <char name="less" width="600"/>
- <char name="lira" width="600"/>
- <char name="ll" width="600"/>
- <char name="logicalnot" width="600"/>
- <char name="lslash" width="600"/>
- <char name="m" width="600"/>
- <char name="macron" width="600"/>
- <char name="merge" width="600"/>
- <char name="minus" width="600"/>
- <char name="mu" width="600"/>
- <char name="multiply" width="600"/>
- <char name="n" width="600"/>
- <char name="nine" width="600"/>
- <char name="notegraphic" width="600"/>
- <char name="ntilde" width="600"/>
- <char name="numbersign" width="600"/>
- <char name="o" width="600"/>
- <char name="oacute" width="600"/>
- <char name="ocircumflex" width="600"/>
- <char name="odieresis" width="600"/>
- <char name="oe" width="600"/>
- <char name="ogonek" width="600"/>
- <char name="ograve" width="600"/>
- <char name="one" width="600"/>
- <char name="onehalf" width="600"/>
- <char name="onequarter" width="600"/>
- <char name="onesuperior" width="600"/>
- <char name="ordfeminine" width="600"/>
- <char name="ordmasculine" width="600"/>
- <char name="oslash" width="600"/>
- <char name="otilde" width="600"/>
- <char name="overscore" width="600"/>
- <char name="p" width="600"/>
- <char name="paragraph" width="600"/>
- <char name="parenleft" width="600"/>
- <char name="parenright" width="600"/>
- <char name="percent" width="600"/>
- <char name="period" width="600"/>
- <char name="periodcentered" width="600"/>
- <char name="perthousand" width="600"/>
- <char name="plus" width="600"/>
- <char name="plusminus" width="600"/>
- <char name="prescription" width="600"/>
- <char name="q" width="600"/>
- <char name="question" width="600"/>
- <char name="questiondown" width="600"/>
- <char name="quotedbl" width="600"/>
- <char name="quotedblbase" width="600"/>
- <char name="quotedblleft" width="600"/>
- <char name="quotedblright" width="600"/>
- <char name="quoteleft" width="600"/>
- <char name="quoteright" width="600"/>
- <char name="quotesinglbase" width="600"/>
- <char name="quotesingle" width="600"/>
- <char name="r" width="600"/>
- <char name="registered" width="600"/>
- <char name="return" width="600"/>
- <char name="ring" width="600"/>
- <char name="s" width="600"/>
- <char name="scaron" width="600"/>
- <char name="scedilla" width="600"/>
- <char name="section" width="600"/>
- <char name="semicolon" width="600"/>
- <char name="seven" width="600"/>
- <char name="six" width="600"/>
- <char name="slash" width="600"/>
- <char name="space" width="600"/>
+ <char-metrics>
+ <char name="A" width="600" llx="3" lly="0" urx="607" ury="562"/>
+ <char name="AE" width="600" llx="3" lly="0" urx="655" ury="562"/>
+ <char name="Aacute" width="600" llx="3" lly="0" urx="660" ury="805"/>
+ <char name="Acircumflex" width="600" llx="3" lly="0" urx="607" ury="787"/>
+ <char name="Adieresis" width="600" llx="3" lly="0" urx="607" ury="753"/>
+ <char name="Agrave" width="600" llx="3" lly="0" urx="607" ury="805"/>
+ <char name="Aring" width="600" llx="3" lly="0" urx="607" ury="750"/>
+ <char name="Atilde" width="600" llx="3" lly="0" urx="655" ury="729"/>
+ <char name="B" width="600" llx="43" lly="0" urx="616" ury="562"/>
+ <char name="C" width="600" llx="93" lly="-18" urx="655" ury="580"/>
+ <char name="Ccedilla" width="600" llx="93" lly="-151" urx="658" ury="580"/>
+ <char name="D" width="600" llx="43" lly="0" urx="645" ury="562"/>
+ <char name="E" width="600" llx="53" lly="0" urx="660" ury="562"/>
+ <char name="Eacute" width="600" llx="53" lly="0" urx="670" ury="805"/>
+ <char name="Ecircumflex" width="600" llx="53" lly="0" urx="660" ury="787"/>
+ <char name="Edieresis" width="600" llx="53" lly="0" urx="660" ury="753"/>
+ <char name="Egrave" width="600" llx="53" lly="0" urx="660" ury="805"/>
+ <char name="Eth" width="600" llx="43" lly="0" urx="645" ury="562"/>
+ <char name="Euro" width="600" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="600" llx="53" lly="0" urx="660" ury="562"/>
+ <char name="G" width="600" llx="83" lly="-18" urx="645" ury="580"/>
+ <char name="H" width="600" llx="32" lly="0" urx="687" ury="562"/>
+ <char name="I" width="600" llx="96" lly="0" urx="623" ury="562"/>
+ <char name="Iacute" width="600" llx="96" lly="0" urx="640" ury="805"/>
+ <char name="Icircumflex" width="600" llx="96" lly="0" urx="623" ury="787"/>
+ <char name="Idieresis" width="600" llx="96" lly="0" urx="623" ury="753"/>
+ <char name="Igrave" width="600" llx="96" lly="0" urx="623" ury="805"/>
+ <char name="J" width="600" llx="52" lly="-18" urx="685" ury="562"/>
+ <char name="K" width="600" llx="38" lly="0" urx="671" ury="562"/>
+ <char name="L" width="600" llx="47" lly="0" urx="607" ury="562"/>
+ <char name="Lslash" width="600" llx="47" lly="0" urx="607" ury="562"/>
+ <char name="M" width="600" llx="4" lly="0" urx="715" ury="562"/>
+ <char name="N" width="600" llx="7" lly="-13" urx="712" ury="562"/>
+ <char name="Ntilde" width="600" llx="7" lly="-13" urx="712" ury="729"/>
+ <char name="O" width="600" llx="94" lly="-18" urx="625" ury="580"/>
+ <char name="OE" width="600" llx="59" lly="0" urx="672" ury="562"/>
+ <char name="Oacute" width="600" llx="94" lly="-18" urx="640" ury="805"/>
+ <char name="Ocircumflex" width="600" llx="94" lly="-18" urx="625" ury="787"/>
+ <char name="Odieresis" width="600" llx="94" lly="-18" urx="625" ury="753"/>
+ <char name="Ograve" width="600" llx="94" lly="-18" urx="625" ury="805"/>
+ <char name="Oslash" width="600" llx="94" lly="-80" urx="625" ury="629"/>
+ <char name="Otilde" width="600" llx="94" lly="-18" urx="655" ury="729"/>
+ <char name="P" width="600" llx="79" lly="0" urx="644" ury="562"/>
+ <char name="Q" width="600" llx="95" lly="-138" urx="625" ury="580"/>
+ <char name="R" width="600" llx="38" lly="0" urx="598" ury="562"/>
+ <char name="S" width="600" llx="76" lly="-20" urx="650" ury="580"/>
+ <char name="Scaron" width="600" llx="76" lly="-20" urx="672" ury="802"/>
+ <char name="Scedilla" width="600" llx="76" lly="-151" urx="650" ury="580"/>
+ <char name="T" width="600" llx="108" lly="0" urx="665" ury="562"/>
+ <char name="Thorn" width="600" llx="79" lly="0" urx="606" ury="562"/>
+ <char name="U" width="600" llx="125" lly="-18" urx="702" ury="562"/>
+ <char name="Uacute" width="600" llx="125" lly="-18" urx="702" ury="805"/>
+ <char name="Ucircumflex" width="600" llx="125" lly="-18" urx="702" ury="787"/>
+ <char name="Udieresis" width="600" llx="125" lly="-18" urx="702" ury="753"/>
+ <char name="Ugrave" width="600" llx="125" lly="-18" urx="702" ury="805"/>
+ <char name="V" width="600" llx="105" lly="-13" urx="723" ury="562"/>
+ <char name="W" width="600" llx="106" lly="-13" urx="722" ury="562"/>
+ <char name="X" width="600" llx="23" lly="0" urx="675" ury="562"/>
+ <char name="Y" width="600" llx="133" lly="0" urx="695" ury="562"/>
+ <char name="Yacute" width="600" llx="133" lly="0" urx="695" ury="805"/>
+ <char name="Ydieresis" width="600" llx="133" lly="0" urx="695" ury="753"/>
+ <char name="Z" width="600" llx="86" lly="0" urx="610" ury="562"/>
+ <char name="Zcaron" width="600" llx="86" lly="0" urx="642" ury="802"/>
+ <char name="a" width="600" llx="76" lly="-15" urx="569" ury="441"/>
+ <char name="aacute" width="600" llx="76" lly="-15" urx="612" ury="672"/>
+ <char name="acircumflex" width="600" llx="76" lly="-15" urx="581" ury="654"/>
+ <char name="acute" width="600" llx="348" lly="497" urx="612" ury="672"/>
+ <char name="adieresis" width="600" llx="76" lly="-15" urx="575" ury="620"/>
+ <char name="ae" width="600" llx="41" lly="-15" urx="626" ury="441"/>
+ <char name="agrave" width="600" llx="76" lly="-15" urx="569" ury="672"/>
+ <char name="ampersand" width="600" llx="87" lly="-15" urx="580" ury="543"/>
+ <char name="aring" width="600" llx="76" lly="-15" urx="569" ury="627"/>
+ <char name="asciicircum" width="600" llx="175" lly="354" urx="587" ury="622"/>
+ <char name="asciitilde" width="600" llx="116" lly="197" urx="600" ury="320"/>
+ <char name="asterisk" width="600" llx="212" lly="257" urx="580" ury="607"/>
+ <char name="at" width="600" llx="127" lly="-15" urx="582" ury="622"/>
+ <char name="atilde" width="600" llx="76" lly="-15" urx="629" ury="606"/>
+ <char name="b" width="600" llx="29" lly="-15" urx="625" ury="629"/>
+ <char name="backslash" width="600" llx="249" lly="-80" urx="468" ury="629"/>
+ <char name="bar" width="600" llx="222" lly="-250" urx="485" ury="750"/>
+ <char name="braceleft" width="600" llx="233" lly="-108" urx="569" ury="622"/>
+ <char name="braceright" width="600" llx="140" lly="-108" urx="477" ury="622"/>
+ <char name="bracketleft" width="600" llx="246" lly="-108" urx="574" ury="622"/>
+ <char name="bracketright" width="600" llx="135" lly="-108" urx="463" ury="622"/>
+ <char name="breve" width="600" llx="279" lly="501" urx="576" ury="609"/>
+ <char name="brokenbar" width="600" llx="238" lly="-175" urx="469" ury="675"/>
+ <char name="bullet" width="600" llx="224" lly="130" urx="485" ury="383"/>
+ <char name="c" width="600" llx="106" lly="-15" urx="608" ury="441"/>
+ <char name="caron" width="600" llx="262" lly="492" urx="614" ury="669"/>
+ <char name="ccedilla" width="600" llx="106" lly="-151" urx="614" ury="441"/>
+ <char name="cedilla" width="600" llx="197" lly="-151" urx="344" ury="10"/>
+ <char name="cent" width="600" llx="151" lly="-49" urx="588" ury="614"/>
+ <char name="circumflex" width="600" llx="229" lly="477" urx="581" ury="654"/>
+ <char name="colon" width="600" llx="238" lly="-15" urx="441" ury="385"/>
+ <char name="comma" width="600" llx="157" lly="-112" urx="370" ury="122"/>
+ <char name="copyright" width="600" llx="53" lly="-18" urx="667" ury="580"/>
+ <char name="currency" width="600" llx="94" lly="58" urx="628" ury="506"/>
+ <char name="d" width="600" llx="85" lly="-15" urx="640" ury="629"/>
+ <char name="dagger" width="600" llx="217" lly="-78" urx="546" ury="580"/>
+ <char name="daggerdbl" width="600" llx="163" lly="-78" urx="546" ury="580"/>
+ <char name="degree" width="600" llx="214" lly="269" urx="576" ury="622"/>
+ <char name="dieresis" width="600" llx="272" lly="537" urx="579" ury="640"/>
+ <char name="divide" width="600" llx="136" lly="48" urx="573" ury="467"/>
+ <char name="dollar" width="600" llx="108" lly="-126" urx="596" ury="662"/>
+ <char name="dotaccent" width="600" llx="373" lly="537" urx="478" ury="640"/>
+ <char name="dotlessi" width="600" llx="95" lly="0" urx="515" ury="426"/>
+ <char name="e" width="600" llx="106" lly="-15" urx="598" ury="441"/>
+ <char name="eacute" width="600" llx="106" lly="-15" urx="612" ury="672"/>
+ <char name="ecircumflex" width="600" llx="106" lly="-15" urx="598" ury="654"/>
+ <char name="edieresis" width="600" llx="106" lly="-15" urx="598" ury="620"/>
+ <char name="egrave" width="600" llx="106" lly="-15" urx="598" ury="672"/>
+ <char name="eight" width="600" llx="132" lly="-15" urx="588" ury="622"/>
+ <char name="ellipsis" width="600" llx="46" lly="-15" urx="575" ury="111"/>
+ <char name="emdash" width="600" llx="49" lly="231" urx="661" ury="285"/>
+ <char name="endash" width="600" llx="124" lly="231" urx="586" ury="285"/>
+ <char name="equal" width="600" llx="109" lly="138" urx="600" ury="376"/>
+ <char name="eth" width="600" llx="102" lly="-15" urx="639" ury="629"/>
+ <char name="exclam" width="600" llx="243" lly="-15" urx="464" ury="572"/>
+ <char name="exclamdown" width="600" llx="225" lly="-157" urx="445" ury="430"/>
+ <char name="f" width="600" llx="114" lly="0" urx="662" ury="629"/>
+ <char name="fi" width="600" llx="3" lly="0" urx="619" ury="629"/>
+ <char name="five" width="600" llx="99" lly="-15" urx="589" ury="607"/>
+ <char name="fl" width="600" llx="3" lly="0" urx="619" ury="629"/>
+ <char name="florin" width="600" llx="-26" lly="-143" urx="671" ury="622"/>
+ <char name="four" width="600" llx="108" lly="0" urx="541" ury="622"/>
+ <char name="fraction" width="600" llx="84" lly="-57" urx="646" ury="665"/>
+ <char name="g" width="600" llx="61" lly="-157" urx="657" ury="441"/>
+ <char name="germandbls" width="600" llx="48" lly="-15" urx="617" ury="629"/>
+ <char name="grave" width="600" llx="294" lly="497" urx="484" ury="672"/>
+ <char name="greater" width="600" llx="85" lly="42" urx="599" ury="472"/>
+ <char name="guillemotleft" width="600" llx="92" lly="70" urx="652" ury="446"/>
+ <char name="guillemotright" width="600" llx="58" lly="70" urx="618" ury="446"/>
+ <char name="guilsinglleft" width="600" llx="204" lly="70" urx="540" ury="446"/>
+ <char name="guilsinglright" width="600" llx="170" lly="70" urx="506" ury="446"/>
+ <char name="h" width="600" llx="33" lly="0" urx="592" ury="629"/>
+ <char name="hungarumlaut" width="600" llx="239" lly="497" urx="683" ury="672"/>
+ <char name="hyphen" width="600" llx="152" lly="231" urx="558" ury="285"/>
+ <char name="i" width="600" llx="95" lly="0" urx="515" ury="657"/>
+ <char name="iacute" width="600" llx="95" lly="0" urx="612" ury="672"/>
+ <char name="icircumflex" width="600" llx="95" lly="0" urx="551" ury="654"/>
+ <char name="idieresis" width="600" llx="95" lly="0" urx="545" ury="620"/>
+ <char name="igrave" width="600" llx="95" lly="0" urx="515" ury="672"/>
+ <char name="j" width="600" llx="52" lly="-157" urx="550" ury="657"/>
+ <char name="k" width="600" llx="58" lly="0" urx="633" ury="629"/>
+ <char name="l" width="600" llx="95" lly="0" urx="515" ury="629"/>
+ <char name="less" width="600" llx="96" lly="42" urx="610" ury="472"/>
+ <char name="logicalnot" width="600" llx="155" lly="108" urx="591" ury="369"/>
+ <char name="lslash" width="600" llx="95" lly="0" urx="587" ury="629"/>
+ <char name="m" width="600" llx="-5" lly="0" urx="615" ury="441"/>
+ <char name="macron" width="600" llx="232" lly="525" urx="600" ury="565"/>
+ <char name="minus" width="600" llx="129" lly="232" urx="580" ury="283"/>
+ <char name="mu" width="600" llx="72" lly="-157" urx="572" ury="426"/>
+ <char name="multiply" width="600" llx="103" lly="43" urx="607" ury="470"/>
+ <char name="n" width="600" llx="26" lly="0" urx="585" ury="441"/>
+ <char name="nine" width="600" llx="93" lly="-15" urx="574" ury="622"/>
+ <char name="ntilde" width="600" llx="26" lly="0" urx="629" ury="606"/>
+ <char name="numbersign" width="600" llx="133" lly="-32" urx="596" ury="639"/>
+ <char name="o" width="600" llx="102" lly="-15" urx="588" ury="441"/>
+ <char name="oacute" width="600" llx="102" lly="-15" urx="612" ury="672"/>
+ <char name="ocircumflex" width="600" llx="102" lly="-15" urx="588" ury="654"/>
+ <char name="odieresis" width="600" llx="102" lly="-15" urx="588" ury="620"/>
+ <char name="oe" width="600" llx="54" lly="-15" urx="615" ury="441"/>
+ <char name="ogonek" width="600" llx="189" lly="-172" urx="377" ury="4"/>
+ <char name="ograve" width="600" llx="102" lly="-15" urx="588" ury="672"/>
+ <char name="one" width="600" llx="98" lly="0" urx="515" ury="622"/>
+ <char name="onehalf" width="600" llx="65" lly="-57" urx="669" ury="665"/>
+ <char name="onequarter" width="600" llx="65" lly="-57" urx="674" ury="665"/>
+ <char name="onesuperior" width="600" llx="231" lly="249" urx="491" ury="622"/>
+ <char name="ordfeminine" width="600" llx="209" lly="249" urx="512" ury="580"/>
+ <char name="ordmasculine" width="600" llx="210" lly="249" urx="535" ury="580"/>
+ <char name="oslash" width="600" llx="102" lly="-80" urx="588" ury="506"/>
+ <char name="otilde" width="600" llx="102" lly="-15" urx="629" ury="606"/>
+ <char name="p" width="600" llx="-24" lly="-157" urx="605" ury="441"/>
+ <char name="paragraph" width="600" llx="100" lly="-78" urx="630" ury="562"/>
+ <char name="parenleft" width="600" llx="313" lly="-108" urx="572" ury="622"/>
+ <char name="parenright" width="600" llx="137" lly="-108" urx="396" ury="622"/>
+ <char name="percent" width="600" llx="134" lly="-15" urx="599" ury="622"/>
+ <char name="period" width="600" llx="238" lly="-15" urx="382" ury="109"/>
+ <char name="periodcentered" width="600" llx="275" lly="189" urx="434" ury="327"/>
+ <char name="perthousand" width="600" llx="59" lly="-15" urx="627" ury="622"/>
+ <char name="plus" width="600" llx="129" lly="44" urx="580" ury="470"/>
+ <char name="plusminus" width="600" llx="96" lly="44" urx="594" ury="558"/>
+ <char name="q" width="600" llx="85" lly="-157" urx="682" ury="441"/>
+ <char name="question" width="600" llx="222" lly="-15" urx="583" ury="572"/>
+ <char name="questiondown" width="600" llx="105" lly="-157" urx="466" ury="430"/>
+ <char name="quotedbl" width="600" llx="273" lly="328" urx="532" ury="562"/>
+ <char name="quotedblbase" width="600" llx="115" lly="-134" urx="478" ury="100"/>
+ <char name="quotedblleft" width="600" llx="262" lly="328" urx="541" ury="562"/>
+ <char name="quotedblright" width="600" llx="213" lly="328" urx="576" ury="562"/>
+ <char name="quoteleft" width="600" llx="343" lly="328" urx="457" ury="562"/>
+ <char name="quoteright" width="600" llx="283" lly="328" urx="495" ury="562"/>
+ <char name="quotesinglbase" width="600" llx="185" lly="-134" urx="397" ury="100"/>
+ <char name="quotesingle" width="600" llx="345" lly="328" urx="460" ury="562"/>
+ <char name="r" width="600" llx="60" lly="0" urx="636" ury="441"/>
+ <char name="registered" width="600" llx="53" lly="-18" urx="667" ury="580"/>
+ <char name="ring" width="600" llx="332" lly="463" urx="500" ury="627"/>
+ <char name="s" width="600" llx="78" lly="-15" urx="584" ury="441"/>
+ <char name="scaron" width="600" llx="78" lly="-15" urx="614" ury="669"/>
+ <char name="scedilla" width="600" llx="78" lly="-151" urx="584" ury="441"/>
+ <char name="section" width="600" llx="104" lly="-78" urx="590" ury="580"/>
+ <char name="semicolon" width="600" llx="157" lly="-112" urx="441" ury="385"/>
+ <char name="seven" width="600" llx="182" lly="0" urx="612" ury="607"/>
+ <char name="six" width="600" llx="155" lly="-15" urx="629" ury="622"/>
+ <char name="slash" width="600" llx="112" lly="-80" urx="604" ury="629"/>
+ <char name="space" width="600" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="600"/>
- <char name="square" width="600"/>
- <char name="sterling" width="600"/>
- <char name="stop" width="600"/>
- <char name="t" width="600"/>
- <char name="tab" width="600"/>
- <char name="thorn" width="600"/>
- <char name="three" width="600"/>
- <char name="threequarters" width="600"/>
- <char name="threesuperior" width="600"/>
- <char name="tilde" width="600"/>
- <char name="trademark" width="600"/>
- <char name="two" width="600"/>
- <char name="twosuperior" width="600"/>
- <char name="u" width="600"/>
- <char name="uacute" width="600"/>
- <char name="ucircumflex" width="600"/>
- <char name="udieresis" width="600"/>
- <char name="ugrave" width="600"/>
- <char name="underscore" width="600"/>
- <char name="up" width="600"/>
- <char name="v" width="600"/>
- <char name="w" width="600"/>
- <char name="x" width="600"/>
- <char name="y" width="600"/>
- <char name="yacute" width="600"/>
- <char name="ydieresis" width="600"/>
- <char name="yen" width="600"/>
- <char name="z" width="600"/>
- <char name="zcaron" width="600"/>
- <char name="zero" width="600"/>
- </widths>
-</font-metrics> \ No newline at end of file
+ <char name="sterling" width="600" llx="124" lly="-21" urx="621" ury="611"/>
+ <char name="t" width="600" llx="167" lly="-15" urx="561" ury="561"/>
+ <char name="thorn" width="600" llx="-24" lly="-157" urx="605" ury="629"/>
+ <char name="three" width="600" llx="82" lly="-15" urx="538" ury="622"/>
+ <char name="threequarters" width="600" llx="73" lly="-56" urx="659" ury="666"/>
+ <char name="threesuperior" width="600" llx="213" lly="240" urx="501" ury="622"/>
+ <char name="tilde" width="600" llx="212" lly="489" urx="629" ury="606"/>
+ <char name="trademark" width="600" llx="75" lly="263" urx="742" ury="562"/>
+ <char name="two" width="600" llx="70" lly="0" urx="568" ury="622"/>
+ <char name="twosuperior" width="600" llx="230" lly="249" urx="535" ury="622"/>
+ <char name="u" width="600" llx="101" lly="-15" urx="572" ury="426"/>
+ <char name="uacute" width="600" llx="101" lly="-15" urx="602" ury="672"/>
+ <char name="ucircumflex" width="600" llx="101" lly="-15" urx="572" ury="654"/>
+ <char name="udieresis" width="600" llx="101" lly="-15" urx="575" ury="620"/>
+ <char name="ugrave" width="600" llx="101" lly="-15" urx="572" ury="672"/>
+ <char name="underscore" width="600" llx="-27" lly="-125" urx="584" ury="-75"/>
+ <char name="v" width="600" llx="90" lly="-10" urx="681" ury="426"/>
+ <char name="w" width="600" llx="76" lly="-10" urx="695" ury="426"/>
+ <char name="x" width="600" llx="20" lly="0" urx="655" ury="426"/>
+ <char name="y" width="600" llx="-4" lly="-157" urx="683" ury="426"/>
+ <char name="yacute" width="600" llx="-4" lly="-157" urx="683" ury="672"/>
+ <char name="ydieresis" width="600" llx="-4" lly="-157" urx="683" ury="620"/>
+ <char name="yen" width="600" llx="120" lly="0" urx="693" ury="562"/>
+ <char name="z" width="600" llx="99" lly="0" urx="593" ury="426"/>
+ <char name="zcaron" width="600" llx="99" lly="0" urx="624" ury="669"/>
+ <char name="zero" width="600" llx="154" lly="-15" urx="575" ury="622"/>
+ </char-metrics>
+</font-metrics>
diff --git a/src/codegen/fonts/Helvetica.xml b/src/codegen/fonts/Helvetica.xml
index 38ae23055..b6e11a72b 100644
--- a/src/codegen/fonts/Helvetica.xml
+++ b/src/codegen/fonts/Helvetica.xml
@@ -22,245 +22,246 @@
<family-name>Helvetica</family-name>
<class-name>Helvetica</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>718</cap-height>
<x-height>523</x-height>
<ascender>718</ascender>
<descender>-207</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="667"/>
- <char name="AE" width="1000"/>
- <char name="Aacute" width="667"/>
- <char name="Acircumflex" width="667"/>
- <char name="Adieresis" width="667"/>
- <char name="Agrave" width="667"/>
- <char name="Aring" width="667"/>
- <char name="Atilde" width="667"/>
- <char name="B" width="667"/>
- <char name="C" width="722"/>
- <char name="Ccedilla" width="722"/>
- <char name="D" width="722"/>
- <char name="E" width="667"/>
- <char name="Eacute" width="667"/>
- <char name="Ecircumflex" width="667"/>
- <char name="Edieresis" width="667"/>
- <char name="Egrave" width="667"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="556"/>
- <char name="F" width="611"/>
- <char name="G" width="778"/>
- <char name="H" width="722"/>
- <char name="I" width="278"/>
- <char name="Iacute" width="278"/>
- <char name="Icircumflex" width="278"/>
- <char name="Idieresis" width="278"/>
- <char name="Igrave" width="278"/>
- <char name="J" width="500"/>
- <char name="K" width="667"/>
- <char name="L" width="556"/>
- <char name="Lslash" width="556"/>
- <char name="M" width="833"/>
- <char name="N" width="722"/>
- <char name="Ntilde" width="722"/>
- <char name="O" width="778"/>
- <char name="OE" width="1000"/>
- <char name="Oacute" width="778"/>
- <char name="Ocircumflex" width="778"/>
- <char name="Odieresis" width="778"/>
- <char name="Ograve" width="778"/>
- <char name="Oslash" width="778"/>
- <char name="Otilde" width="778"/>
- <char name="P" width="667"/>
- <char name="Q" width="778"/>
- <char name="R" width="722"/>
- <char name="S" width="667"/>
- <char name="Scaron" width="667"/>
- <char name="T" width="611"/>
- <char name="Thorn" width="667"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="667"/>
- <char name="W" width="944"/>
- <char name="X" width="667"/>
- <char name="Y" width="667"/>
- <char name="Yacute" width="667"/>
- <char name="Ydieresis" width="667"/>
- <char name="Z" width="611"/>
- <char name="Zcaron" width="611"/>
- <char name="a" width="556"/>
- <char name="aacute" width="556"/>
- <char name="acircumflex" width="556"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="556"/>
- <char name="ae" width="889"/>
- <char name="agrave" width="556"/>
- <char name="ampersand" width="667"/>
- <char name="aring" width="556"/>
- <char name="asciicircum" width="469"/>
- <char name="asciitilde" width="584"/>
- <char name="asterisk" width="389"/>
- <char name="at" width="1015"/>
- <char name="atilde" width="556"/>
- <char name="b" width="556"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="260"/>
- <char name="braceleft" width="334"/>
- <char name="braceright" width="334"/>
- <char name="bracketleft" width="278"/>
- <char name="bracketright" width="278"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="260"/>
- <char name="bullet" width="350"/>
- <char name="c" width="500"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="500"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="556"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="278"/>
- <char name="comma" width="278"/>
- <char name="copyright" width="737"/>
- <char name="currency" width="556"/>
- <char name="d" width="556"/>
- <char name="dagger" width="556"/>
- <char name="daggerdbl" width="556"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="584"/>
- <char name="dollar" width="556"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="556"/>
- <char name="eacute" width="556"/>
- <char name="ecircumflex" width="556"/>
- <char name="edieresis" width="556"/>
- <char name="egrave" width="556"/>
- <char name="eight" width="556"/>
- <char name="ellipsis" width="1000"/>
- <char name="emdash" width="1000"/>
- <char name="endash" width="556"/>
- <char name="equal" width="584"/>
- <char name="eth" width="556"/>
- <char name="exclam" width="278"/>
- <char name="exclamdown" width="333"/>
- <char name="f" width="278"/>
- <char name="fi" width="500"/>
- <char name="five" width="556"/>
- <char name="fl" width="500"/>
- <char name="florin" width="556"/>
- <char name="four" width="556"/>
- <char name="fraction" width="167"/>
- <char name="g" width="556"/>
- <char name="germandbls" width="611"/>
- <char name="grave" width="333"/>
- <char name="greater" width="584"/>
- <char name="guillemotleft" width="556"/>
- <char name="guillemotright" width="556"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="556"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="222"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="222"/>
- <char name="k" width="500"/>
- <char name="l" width="222"/>
- <char name="less" width="584"/>
- <char name="logicalnot" width="584"/>
- <char name="lslash" width="222"/>
- <char name="m" width="833"/>
- <char name="macron" width="333"/>
- <char name="minus" width="324"/>
- <char name="mu" width="556"/>
- <char name="multiply" width="584"/>
- <char name="n" width="556"/>
- <char name="nine" width="556"/>
- <char name="ntilde" width="556"/>
- <char name="numbersign" width="556"/>
- <char name="o" width="556"/>
- <char name="oacute" width="556"/>
- <char name="ocircumflex" width="556"/>
- <char name="odieresis" width="556"/>
- <char name="oe" width="944"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="556"/>
- <char name="one" width="556"/>
- <char name="onehalf" width="834"/>
- <char name="onequarter" width="834"/>
- <char name="onesuperior" width="333"/>
- <char name="ordfeminine" width="370"/>
- <char name="ordmasculine" width="365"/>
- <char name="oslash" width="611"/>
- <char name="otilde" width="556"/>
- <char name="p" width="556"/>
- <char name="paragraph" width="537"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="889"/>
- <char name="period" width="278"/>
- <char name="periodcentered" width="278"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="584"/>
- <char name="plusminus" width="584"/>
- <char name="q" width="556"/>
- <char name="question" width="556"/>
- <char name="questiondown" width="611"/>
- <char name="quotedbl" width="355"/>
- <char name="quotedblbase" width="333"/>
- <char name="quotedblleft" width="333"/>
- <char name="quotedblright" width="333"/>
- <char name="quoteleft" width="222"/>
- <char name="quoteright" width="222"/>
- <char name="quotesinglbase" width="222"/>
- <char name="quotesingle" width="191"/>
- <char name="r" width="333"/>
- <char name="registered" width="737"/>
- <char name="ring" width="333"/>
- <char name="s" width="500"/>
- <char name="scaron" width="500"/>
- <char name="section" width="556"/>
- <char name="semicolon" width="278"/>
- <char name="seven" width="556"/>
- <char name="six" width="556"/>
- <char name="slash" width="278"/>
- <char name="space" width="278"/>
+ <char-metrics>
+ <char name="A" width="667" llx="14" lly="0" urx="654" ury="718"/>
+ <char name="AE" width="1000" llx="8" lly="0" urx="951" ury="718"/>
+ <char name="Aacute" width="667" llx="14" lly="0" urx="654" ury="929"/>
+ <char name="Acircumflex" width="667" llx="14" lly="0" urx="654" ury="929"/>
+ <char name="Adieresis" width="667" llx="14" lly="0" urx="654" ury="901"/>
+ <char name="Agrave" width="667" llx="14" lly="0" urx="654" ury="929"/>
+ <char name="Aring" width="667" llx="14" lly="0" urx="654" ury="931"/>
+ <char name="Atilde" width="667" llx="14" lly="0" urx="654" ury="917"/>
+ <char name="B" width="667" llx="74" lly="0" urx="627" ury="718"/>
+ <char name="C" width="722" llx="44" lly="-19" urx="681" ury="737"/>
+ <char name="Ccedilla" width="722" llx="44" lly="-225" urx="681" ury="737"/>
+ <char name="D" width="722" llx="81" lly="0" urx="674" ury="718"/>
+ <char name="E" width="667" llx="86" lly="0" urx="616" ury="718"/>
+ <char name="Eacute" width="667" llx="86" lly="0" urx="616" ury="929"/>
+ <char name="Ecircumflex" width="667" llx="86" lly="0" urx="616" ury="929"/>
+ <char name="Edieresis" width="667" llx="86" lly="0" urx="616" ury="901"/>
+ <char name="Egrave" width="667" llx="86" lly="0" urx="616" ury="929"/>
+ <char name="Eth" width="722" llx="0" lly="0" urx="674" ury="718"/>
+ <char name="Euro" width="556" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="611" llx="86" lly="0" urx="583" ury="718"/>
+ <char name="G" width="778" llx="48" lly="-19" urx="704" ury="737"/>
+ <char name="H" width="722" llx="77" lly="0" urx="646" ury="718"/>
+ <char name="I" width="278" llx="91" lly="0" urx="188" ury="718"/>
+ <char name="Iacute" width="278" llx="91" lly="0" urx="292" ury="929"/>
+ <char name="Icircumflex" width="278" llx="-6" lly="0" urx="285" ury="929"/>
+ <char name="Idieresis" width="278" llx="13" lly="0" urx="266" ury="901"/>
+ <char name="Igrave" width="278" llx="-13" lly="0" urx="188" ury="929"/>
+ <char name="J" width="500" llx="17" lly="-19" urx="428" ury="718"/>
+ <char name="K" width="667" llx="76" lly="0" urx="663" ury="718"/>
+ <char name="L" width="556" llx="76" lly="0" urx="537" ury="718"/>
+ <char name="Lslash" width="556" llx="-20" lly="0" urx="537" ury="718"/>
+ <char name="M" width="833" llx="73" lly="0" urx="761" ury="718"/>
+ <char name="N" width="722" llx="76" lly="0" urx="646" ury="718"/>
+ <char name="Ntilde" width="722" llx="76" lly="0" urx="646" ury="917"/>
+ <char name="O" width="778" llx="39" lly="-19" urx="739" ury="737"/>
+ <char name="OE" width="1000" llx="36" lly="-19" urx="965" ury="737"/>
+ <char name="Oacute" width="778" llx="39" lly="-19" urx="739" ury="929"/>
+ <char name="Ocircumflex" width="778" llx="39" lly="-19" urx="739" ury="929"/>
+ <char name="Odieresis" width="778" llx="39" lly="-19" urx="739" ury="901"/>
+ <char name="Ograve" width="778" llx="39" lly="-19" urx="739" ury="929"/>
+ <char name="Oslash" width="778" llx="39" lly="-19" urx="740" ury="737"/>
+ <char name="Otilde" width="778" llx="39" lly="-19" urx="739" ury="917"/>
+ <char name="P" width="667" llx="86" lly="0" urx="622" ury="718"/>
+ <char name="Q" width="778" llx="39" lly="-56" urx="739" ury="737"/>
+ <char name="R" width="722" llx="88" lly="0" urx="684" ury="718"/>
+ <char name="S" width="667" llx="49" lly="-19" urx="620" ury="737"/>
+ <char name="Scaron" width="667" llx="49" lly="-19" urx="620" ury="929"/>
+ <char name="T" width="611" llx="14" lly="0" urx="597" ury="718"/>
+ <char name="Thorn" width="667" llx="86" lly="0" urx="622" ury="718"/>
+ <char name="U" width="722" llx="79" lly="-19" urx="644" ury="718"/>
+ <char name="Uacute" width="722" llx="79" lly="-19" urx="644" ury="929"/>
+ <char name="Ucircumflex" width="722" llx="79" lly="-19" urx="644" ury="929"/>
+ <char name="Udieresis" width="722" llx="79" lly="-19" urx="644" ury="901"/>
+ <char name="Ugrave" width="722" llx="79" lly="-19" urx="644" ury="929"/>
+ <char name="V" width="667" llx="20" lly="0" urx="647" ury="718"/>
+ <char name="W" width="944" llx="16" lly="0" urx="928" ury="718"/>
+ <char name="X" width="667" llx="19" lly="0" urx="648" ury="718"/>
+ <char name="Y" width="667" llx="14" lly="0" urx="653" ury="718"/>
+ <char name="Yacute" width="667" llx="14" lly="0" urx="653" ury="929"/>
+ <char name="Ydieresis" width="667" llx="14" lly="0" urx="653" ury="901"/>
+ <char name="Z" width="611" llx="23" lly="0" urx="588" ury="718"/>
+ <char name="Zcaron" width="611" llx="23" lly="0" urx="588" ury="929"/>
+ <char name="a" width="556" llx="36" lly="-15" urx="530" ury="538"/>
+ <char name="aacute" width="556" llx="36" lly="-15" urx="530" ury="734"/>
+ <char name="acircumflex" width="556" llx="36" lly="-15" urx="530" ury="734"/>
+ <char name="acute" width="333" llx="122" lly="593" urx="319" ury="734"/>
+ <char name="adieresis" width="556" llx="36" lly="-15" urx="530" ury="706"/>
+ <char name="ae" width="889" llx="36" lly="-15" urx="847" ury="538"/>
+ <char name="agrave" width="556" llx="36" lly="-15" urx="530" ury="734"/>
+ <char name="ampersand" width="667" llx="44" lly="-15" urx="645" ury="718"/>
+ <char name="aring" width="556" llx="36" lly="-15" urx="530" ury="756"/>
+ <char name="asciicircum" width="469" llx="-14" lly="264" urx="483" ury="688"/>
+ <char name="asciitilde" width="584" llx="61" lly="180" urx="523" ury="326"/>
+ <char name="asterisk" width="389" llx="39" lly="431" urx="349" ury="718"/>
+ <char name="at" width="1015" llx="147" lly="-19" urx="868" ury="737"/>
+ <char name="atilde" width="556" llx="36" lly="-15" urx="530" ury="722"/>
+ <char name="b" width="556" llx="58" lly="-15" urx="517" ury="718"/>
+ <char name="backslash" width="278" llx="-17" lly="-19" urx="295" ury="737"/>
+ <char name="bar" width="260" llx="94" lly="-225" urx="167" ury="775"/>
+ <char name="braceleft" width="334" llx="42" lly="-196" urx="292" ury="722"/>
+ <char name="braceright" width="334" llx="42" lly="-196" urx="292" ury="722"/>
+ <char name="bracketleft" width="278" llx="63" lly="-196" urx="250" ury="722"/>
+ <char name="bracketright" width="278" llx="28" lly="-196" urx="215" ury="722"/>
+ <char name="breve" width="333" llx="13" lly="595" urx="321" ury="731"/>
+ <char name="brokenbar" width="260" llx="94" lly="-150" urx="167" ury="700"/>
+ <char name="bullet" width="350" llx="18" lly="202" urx="333" ury="517"/>
+ <char name="c" width="500" llx="30" lly="-15" urx="477" ury="538"/>
+ <char name="caron" width="333" llx="21" lly="593" urx="312" ury="734"/>
+ <char name="ccedilla" width="500" llx="30" lly="-225" urx="477" ury="538"/>
+ <char name="cedilla" width="333" llx="45" lly="-225" urx="259" ury="0"/>
+ <char name="cent" width="556" llx="51" lly="-115" urx="513" ury="623"/>
+ <char name="circumflex" width="333" llx="21" lly="593" urx="312" ury="734"/>
+ <char name="colon" width="278" llx="87" lly="0" urx="191" ury="516"/>
+ <char name="comma" width="278" llx="87" lly="-147" urx="191" ury="106"/>
+ <char name="copyright" width="737" llx="-14" lly="-19" urx="752" ury="737"/>
+ <char name="currency" width="556" llx="28" lly="99" urx="528" ury="603"/>
+ <char name="d" width="556" llx="35" lly="-15" urx="499" ury="718"/>
+ <char name="dagger" width="556" llx="43" lly="-159" urx="514" ury="718"/>
+ <char name="daggerdbl" width="556" llx="43" lly="-159" urx="514" ury="718"/>
+ <char name="degree" width="400" llx="54" lly="411" urx="346" ury="703"/>
+ <char name="dieresis" width="333" llx="40" lly="604" urx="293" ury="706"/>
+ <char name="divide" width="584" llx="39" lly="-19" urx="545" ury="524"/>
+ <char name="dollar" width="556" llx="32" lly="-115" urx="520" ury="775"/>
+ <char name="dotaccent" width="333" llx="121" lly="604" urx="212" ury="706"/>
+ <char name="dotlessi" width="278" llx="95" lly="0" urx="183" ury="523"/>
+ <char name="e" width="556" llx="40" lly="-15" urx="516" ury="538"/>
+ <char name="eacute" width="556" llx="40" lly="-15" urx="516" ury="734"/>
+ <char name="ecircumflex" width="556" llx="40" lly="-15" urx="516" ury="734"/>
+ <char name="edieresis" width="556" llx="40" lly="-15" urx="516" ury="706"/>
+ <char name="egrave" width="556" llx="40" lly="-15" urx="516" ury="734"/>
+ <char name="eight" width="556" llx="38" lly="-19" urx="517" ury="703"/>
+ <char name="ellipsis" width="1000" llx="115" lly="0" urx="885" ury="106"/>
+ <char name="emdash" width="1000" llx="0" lly="240" urx="1000" ury="313"/>
+ <char name="endash" width="556" llx="0" lly="240" urx="556" ury="313"/>
+ <char name="equal" width="584" llx="39" lly="115" urx="545" ury="390"/>
+ <char name="eth" width="556" llx="35" lly="-15" urx="522" ury="737"/>
+ <char name="exclam" width="278" llx="90" lly="0" urx="187" ury="718"/>
+ <char name="exclamdown" width="333" llx="118" lly="-195" urx="215" ury="523"/>
+ <char name="f" width="278" llx="14" lly="0" urx="262" ury="728"/>
+ <char name="fi" width="500" llx="14" lly="0" urx="434" ury="728"/>
+ <char name="five" width="556" llx="32" lly="-19" urx="514" ury="688"/>
+ <char name="fl" width="500" llx="14" lly="0" urx="432" ury="728"/>
+ <char name="florin" width="556" llx="-11" lly="-207" urx="501" ury="737"/>
+ <char name="four" width="556" llx="25" lly="0" urx="523" ury="703"/>
+ <char name="fraction" width="167" llx="-166" lly="-19" urx="333" ury="703"/>
+ <char name="g" width="556" llx="40" lly="-220" urx="499" ury="538"/>
+ <char name="germandbls" width="611" llx="67" lly="-15" urx="571" ury="728"/>
+ <char name="grave" width="333" llx="14" lly="593" urx="211" ury="734"/>
+ <char name="greater" width="584" llx="48" lly="11" urx="536" ury="495"/>
+ <char name="guillemotleft" width="556" llx="97" lly="108" urx="459" ury="446"/>
+ <char name="guillemotright" width="556" llx="97" lly="108" urx="459" ury="446"/>
+ <char name="guilsinglleft" width="333" llx="88" lly="108" urx="245" ury="446"/>
+ <char name="guilsinglright" width="333" llx="88" lly="108" urx="245" ury="446"/>
+ <char name="h" width="556" llx="65" lly="0" urx="491" ury="718"/>
+ <char name="hungarumlaut" width="333" llx="31" lly="593" urx="409" ury="734"/>
+ <char name="hyphen" width="333" llx="44" lly="232" urx="289" ury="322"/>
+ <char name="i" width="222" llx="67" lly="0" urx="155" ury="718"/>
+ <char name="iacute" width="278" llx="95" lly="0" urx="292" ury="734"/>
+ <char name="icircumflex" width="278" llx="-6" lly="0" urx="285" ury="734"/>
+ <char name="idieresis" width="278" llx="13" lly="0" urx="266" ury="706"/>
+ <char name="igrave" width="278" llx="-13" lly="0" urx="184" ury="734"/>
+ <char name="j" width="222" llx="-16" lly="-210" urx="155" ury="718"/>
+ <char name="k" width="500" llx="67" lly="0" urx="501" ury="718"/>
+ <char name="l" width="222" llx="67" lly="0" urx="155" ury="718"/>
+ <char name="less" width="584" llx="48" lly="11" urx="536" ury="495"/>
+ <char name="logicalnot" width="584" llx="39" lly="108" urx="545" ury="390"/>
+ <char name="lslash" width="222" llx="-20" lly="0" urx="242" ury="718"/>
+ <char name="m" width="833" llx="65" lly="0" urx="769" ury="538"/>
+ <char name="macron" width="333" llx="10" lly="627" urx="323" ury="684"/>
+ <char name="minus" width="324" llx="39" lly="216" urx="545" ury="289"/>
+ <char name="mu" width="556" llx="68" lly="-207" urx="489" ury="523"/>
+ <char name="multiply" width="584" llx="39" lly="0" urx="545" ury="506"/>
+ <char name="n" width="556" llx="65" lly="0" urx="491" ury="538"/>
+ <char name="nine" width="556" llx="42" lly="-19" urx="514" ury="703"/>
+ <char name="ntilde" width="556" llx="65" lly="0" urx="491" ury="722"/>
+ <char name="numbersign" width="556" llx="28" lly="0" urx="529" ury="688"/>
+ <char name="o" width="556" llx="35" lly="-14" urx="521" ury="538"/>
+ <char name="oacute" width="556" llx="35" lly="-14" urx="521" ury="734"/>
+ <char name="ocircumflex" width="556" llx="35" lly="-14" urx="521" ury="734"/>
+ <char name="odieresis" width="556" llx="35" lly="-14" urx="521" ury="706"/>
+ <char name="oe" width="944" llx="35" lly="-15" urx="902" ury="538"/>
+ <char name="ogonek" width="333" llx="73" lly="-225" urx="287" ury="0"/>
+ <char name="ograve" width="556" llx="35" lly="-14" urx="521" ury="734"/>
+ <char name="one" width="556" llx="101" lly="0" urx="359" ury="703"/>
+ <char name="onehalf" width="834" llx="43" lly="-19" urx="773" ury="703"/>
+ <char name="onequarter" width="834" llx="73" lly="-19" urx="756" ury="703"/>
+ <char name="onesuperior" width="333" llx="43" lly="281" urx="222" ury="703"/>
+ <char name="ordfeminine" width="370" llx="24" lly="405" urx="346" ury="737"/>
+ <char name="ordmasculine" width="365" llx="25" lly="405" urx="341" ury="737"/>
+ <char name="oslash" width="611" llx="28" lly="-22" urx="537" ury="545"/>
+ <char name="otilde" width="556" llx="35" lly="-14" urx="521" ury="722"/>
+ <char name="p" width="556" llx="58" lly="-207" urx="517" ury="538"/>
+ <char name="paragraph" width="537" llx="18" lly="-173" urx="497" ury="718"/>
+ <char name="parenleft" width="333" llx="68" lly="-207" urx="299" ury="733"/>
+ <char name="parenright" width="333" llx="34" lly="-207" urx="265" ury="733"/>
+ <char name="percent" width="889" llx="39" lly="-19" urx="850" ury="703"/>
+ <char name="period" width="278" llx="87" lly="0" urx="191" ury="106"/>
+ <char name="periodcentered" width="278" llx="77" lly="190" urx="202" ury="315"/>
+ <char name="perthousand" width="1000" llx="7" lly="-19" urx="994" ury="703"/>
+ <char name="plus" width="584" llx="39" lly="0" urx="545" ury="505"/>
+ <char name="plusminus" width="584" llx="39" lly="0" urx="545" ury="506"/>
+ <char name="q" width="556" llx="35" lly="-207" urx="494" ury="538"/>
+ <char name="question" width="556" llx="56" lly="0" urx="492" ury="727"/>
+ <char name="questiondown" width="611" llx="91" lly="-201" urx="527" ury="525"/>
+ <char name="quotedbl" width="355" llx="70" lly="463" urx="285" ury="718"/>
+ <char name="quotedblbase" width="333" llx="26" lly="-149" urx="295" ury="106"/>
+ <char name="quotedblleft" width="333" llx="38" lly="470" urx="307" ury="725"/>
+ <char name="quotedblright" width="333" llx="26" lly="463" urx="295" ury="718"/>
+ <char name="quoteleft" width="222" llx="65" lly="470" urx="169" ury="725"/>
+ <char name="quoteright" width="222" llx="53" lly="463" urx="157" ury="718"/>
+ <char name="quotesinglbase" width="222" llx="53" lly="-149" urx="157" ury="106"/>
+ <char name="quotesingle" width="191" llx="59" lly="463" urx="132" ury="718"/>
+ <char name="r" width="333" llx="77" lly="0" urx="332" ury="538"/>
+ <char name="registered" width="737" llx="-14" lly="-19" urx="752" ury="737"/>
+ <char name="ring" width="333" llx="75" lly="572" urx="259" ury="756"/>
+ <char name="s" width="500" llx="32" lly="-15" urx="464" ury="538"/>
+ <char name="scaron" width="500" llx="32" lly="-15" urx="464" ury="734"/>
+ <char name="section" width="556" llx="43" lly="-191" urx="512" ury="737"/>
+ <char name="semicolon" width="278" llx="87" lly="-147" urx="191" ury="516"/>
+ <char name="seven" width="556" llx="37" lly="0" urx="523" ury="688"/>
+ <char name="six" width="556" llx="38" lly="-19" urx="518" ury="703"/>
+ <char name="slash" width="278" llx="-17" lly="-19" urx="295" ury="737"/>
+ <char name="space" width="278" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="278"/>
- <char name="sterling" width="556"/>
- <char name="t" width="278"/>
- <char name="thorn" width="556"/>
- <char name="three" width="556"/>
- <char name="threequarters" width="834"/>
- <char name="threesuperior" width="333"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="1000"/>
- <char name="two" width="556"/>
- <char name="twosuperior" width="333"/>
- <char name="u" width="556"/>
- <char name="uacute" width="556"/>
- <char name="ucircumflex" width="556"/>
- <char name="udieresis" width="556"/>
- <char name="ugrave" width="556"/>
- <char name="underscore" width="556"/>
- <char name="v" width="500"/>
- <char name="w" width="722"/>
- <char name="x" width="500"/>
- <char name="y" width="500"/>
- <char name="yacute" width="500"/>
- <char name="ydieresis" width="500"/>
- <char name="yen" width="556"/>
- <char name="z" width="500"/>
- <char name="zcaron" width="500"/>
- <char name="zero" width="556"/>
- </widths>
+ <char name="sterling" width="556" llx="33" lly="-16" urx="539" ury="718"/>
+ <char name="t" width="278" llx="14" lly="-7" urx="257" ury="669"/>
+ <char name="thorn" width="556" llx="58" lly="-207" urx="517" ury="718"/>
+ <char name="three" width="556" llx="34" lly="-19" urx="522" ury="703"/>
+ <char name="threequarters" width="834" llx="45" lly="-19" urx="810" ury="703"/>
+ <char name="threesuperior" width="333" llx="5" lly="270" urx="325" ury="703"/>
+ <char name="tilde" width="333" llx="-4" lly="606" urx="337" ury="722"/>
+ <char name="trademark" width="1000" llx="46" lly="306" urx="903" ury="718"/>
+ <char name="two" width="556" llx="26" lly="0" urx="507" ury="703"/>
+ <char name="twosuperior" width="333" llx="4" lly="281" urx="323" ury="703"/>
+ <char name="u" width="556" llx="68" lly="-15" urx="489" ury="523"/>
+ <char name="uacute" width="556" llx="68" lly="-15" urx="489" ury="734"/>
+ <char name="ucircumflex" width="556" llx="68" lly="-15" urx="489" ury="734"/>
+ <char name="udieresis" width="556" llx="68" lly="-15" urx="489" ury="706"/>
+ <char name="ugrave" width="556" llx="68" lly="-15" urx="489" ury="734"/>
+ <char name="underscore" width="556" llx="0" lly="-125" urx="556" ury="-75"/>
+ <char name="v" width="500" llx="8" lly="0" urx="492" ury="523"/>
+ <char name="w" width="722" llx="14" lly="0" urx="709" ury="523"/>
+ <char name="x" width="500" llx="11" lly="0" urx="490" ury="523"/>
+ <char name="y" width="500" llx="11" lly="-214" urx="489" ury="523"/>
+ <char name="yacute" width="500" llx="11" lly="-214" urx="489" ury="734"/>
+ <char name="ydieresis" width="500" llx="11" lly="-214" urx="489" ury="706"/>
+ <char name="yen" width="556" llx="3" lly="0" urx="553" ury="688"/>
+ <char name="z" width="500" llx="31" lly="0" urx="469" ury="523"/>
+ <char name="zcaron" width="500" llx="31" lly="0" urx="469" ury="734"/>
+ <char name="zero" width="556" llx="37" lly="-19" urx="519" ury="703"/>
+ </char-metrics>
<kerning kpx1="107">
<pair kern="-20" kpx2="111"/>
<pair kern="-20" kpx2="101"/>
diff --git a/src/codegen/fonts/HelveticaBold.xml b/src/codegen/fonts/HelveticaBold.xml
index 2620a128e..4efd01f22 100644
--- a/src/codegen/fonts/HelveticaBold.xml
+++ b/src/codegen/fonts/HelveticaBold.xml
@@ -22,245 +22,246 @@
<family-name>Helvetica</family-name>
<class-name>HelveticaBold</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>718</cap-height>
<x-height>532</x-height>
<ascender>718</ascender>
<descender>-207</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="722"/>
- <char name="AE" width="1000"/>
- <char name="Aacute" width="722"/>
- <char name="Acircumflex" width="722"/>
- <char name="Adieresis" width="722"/>
- <char name="Agrave" width="722"/>
- <char name="Aring" width="722"/>
- <char name="Atilde" width="722"/>
- <char name="B" width="722"/>
- <char name="C" width="722"/>
- <char name="Ccedilla" width="722"/>
- <char name="D" width="722"/>
- <char name="E" width="667"/>
- <char name="Eacute" width="667"/>
- <char name="Ecircumflex" width="667"/>
- <char name="Edieresis" width="667"/>
- <char name="Egrave" width="667"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="556"/>
- <char name="F" width="611"/>
- <char name="G" width="778"/>
- <char name="H" width="722"/>
- <char name="I" width="278"/>
- <char name="Iacute" width="278"/>
- <char name="Icircumflex" width="278"/>
- <char name="Idieresis" width="278"/>
- <char name="Igrave" width="278"/>
- <char name="J" width="556"/>
- <char name="K" width="722"/>
- <char name="L" width="611"/>
- <char name="Lslash" width="611"/>
- <char name="M" width="833"/>
- <char name="N" width="722"/>
- <char name="Ntilde" width="722"/>
- <char name="O" width="778"/>
- <char name="OE" width="1000"/>
- <char name="Oacute" width="778"/>
- <char name="Ocircumflex" width="778"/>
- <char name="Odieresis" width="778"/>
- <char name="Ograve" width="778"/>
- <char name="Oslash" width="778"/>
- <char name="Otilde" width="778"/>
- <char name="P" width="667"/>
- <char name="Q" width="778"/>
- <char name="R" width="722"/>
- <char name="S" width="667"/>
- <char name="Scaron" width="667"/>
- <char name="T" width="611"/>
- <char name="Thorn" width="667"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="667"/>
- <char name="W" width="944"/>
- <char name="X" width="667"/>
- <char name="Y" width="667"/>
- <char name="Yacute" width="667"/>
- <char name="Ydieresis" width="667"/>
- <char name="Z" width="611"/>
- <char name="Zcaron" width="611"/>
- <char name="a" width="556"/>
- <char name="aacute" width="556"/>
- <char name="acircumflex" width="556"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="556"/>
- <char name="ae" width="889"/>
- <char name="agrave" width="556"/>
- <char name="ampersand" width="722"/>
- <char name="aring" width="556"/>
- <char name="asciicircum" width="584"/>
- <char name="asciitilde" width="584"/>
- <char name="asterisk" width="389"/>
- <char name="at" width="975"/>
- <char name="atilde" width="556"/>
- <char name="b" width="611"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="280"/>
- <char name="braceleft" width="389"/>
- <char name="braceright" width="389"/>
- <char name="bracketleft" width="333"/>
- <char name="bracketright" width="333"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="280"/>
- <char name="bullet" width="350"/>
- <char name="c" width="556"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="556"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="556"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="333"/>
- <char name="comma" width="278"/>
- <char name="copyright" width="737"/>
- <char name="currency" width="556"/>
- <char name="d" width="611"/>
- <char name="dagger" width="556"/>
- <char name="daggerdbl" width="556"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="584"/>
- <char name="dollar" width="556"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="556"/>
- <char name="eacute" width="556"/>
- <char name="ecircumflex" width="556"/>
- <char name="edieresis" width="556"/>
- <char name="egrave" width="556"/>
- <char name="eight" width="556"/>
- <char name="ellipsis" width="1000"/>
- <char name="emdash" width="1000"/>
- <char name="endash" width="556"/>
- <char name="equal" width="584"/>
- <char name="eth" width="611"/>
- <char name="exclam" width="333"/>
- <char name="exclamdown" width="333"/>
- <char name="f" width="333"/>
- <char name="fi" width="611"/>
- <char name="five" width="556"/>
- <char name="fl" width="611"/>
- <char name="florin" width="556"/>
- <char name="four" width="556"/>
- <char name="fraction" width="167"/>
- <char name="g" width="611"/>
- <char name="germandbls" width="611"/>
- <char name="grave" width="333"/>
- <char name="greater" width="584"/>
- <char name="guillemotleft" width="556"/>
- <char name="guillemotright" width="556"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="611"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="278"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="278"/>
- <char name="k" width="556"/>
- <char name="l" width="278"/>
- <char name="less" width="584"/>
- <char name="logicalnot" width="584"/>
- <char name="lslash" width="278"/>
- <char name="m" width="889"/>
- <char name="macron" width="333"/>
- <char name="minus" width="324"/>
- <char name="mu" width="611"/>
- <char name="multiply" width="584"/>
- <char name="n" width="611"/>
- <char name="nine" width="556"/>
- <char name="ntilde" width="611"/>
- <char name="numbersign" width="556"/>
- <char name="o" width="611"/>
- <char name="oacute" width="611"/>
- <char name="ocircumflex" width="611"/>
- <char name="odieresis" width="611"/>
- <char name="oe" width="944"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="611"/>
- <char name="one" width="556"/>
- <char name="onehalf" width="834"/>
- <char name="onequarter" width="834"/>
- <char name="onesuperior" width="333"/>
- <char name="ordfeminine" width="370"/>
- <char name="ordmasculine" width="365"/>
- <char name="oslash" width="611"/>
- <char name="otilde" width="611"/>
- <char name="p" width="611"/>
- <char name="paragraph" width="556"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="889"/>
- <char name="period" width="278"/>
- <char name="periodcentered" width="278"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="584"/>
- <char name="plusminus" width="584"/>
- <char name="q" width="611"/>
- <char name="question" width="611"/>
- <char name="questiondown" width="611"/>
- <char name="quotedbl" width="474"/>
- <char name="quotedblbase" width="500"/>
- <char name="quotedblleft" width="500"/>
- <char name="quotedblright" width="500"/>
- <char name="quoteleft" width="278"/>
- <char name="quoteright" width="278"/>
- <char name="quotesinglbase" width="278"/>
- <char name="quotesingle" width="238"/>
- <char name="r" width="389"/>
- <char name="registered" width="737"/>
- <char name="ring" width="333"/>
- <char name="s" width="556"/>
- <char name="scaron" width="556"/>
- <char name="section" width="556"/>
- <char name="semicolon" width="333"/>
- <char name="seven" width="556"/>
- <char name="six" width="556"/>
- <char name="slash" width="278"/>
- <char name="space" width="278"/>
+ <char-metrics>
+ <char name="A" width="722" llx="20" lly="0" urx="702" ury="718"/>
+ <char name="AE" width="1000" llx="5" lly="0" urx="954" ury="718"/>
+ <char name="Aacute" width="722" llx="20" lly="0" urx="702" ury="936"/>
+ <char name="Acircumflex" width="722" llx="20" lly="0" urx="702" ury="936"/>
+ <char name="Adieresis" width="722" llx="20" lly="0" urx="702" ury="915"/>
+ <char name="Agrave" width="722" llx="20" lly="0" urx="702" ury="936"/>
+ <char name="Aring" width="722" llx="20" lly="0" urx="702" ury="962"/>
+ <char name="Atilde" width="722" llx="20" lly="0" urx="702" ury="923"/>
+ <char name="B" width="722" llx="76" lly="0" urx="669" ury="718"/>
+ <char name="C" width="722" llx="44" lly="-19" urx="684" ury="737"/>
+ <char name="Ccedilla" width="722" llx="44" lly="-228" urx="684" ury="737"/>
+ <char name="D" width="722" llx="76" lly="0" urx="685" ury="718"/>
+ <char name="E" width="667" llx="76" lly="0" urx="621" ury="718"/>
+ <char name="Eacute" width="667" llx="76" lly="0" urx="621" ury="936"/>
+ <char name="Ecircumflex" width="667" llx="76" lly="0" urx="621" ury="936"/>
+ <char name="Edieresis" width="667" llx="76" lly="0" urx="621" ury="915"/>
+ <char name="Egrave" width="667" llx="76" lly="0" urx="621" ury="936"/>
+ <char name="Eth" width="722" llx="-5" lly="0" urx="685" ury="718"/>
+ <char name="Euro" width="556" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="611" llx="76" lly="0" urx="587" ury="718"/>
+ <char name="G" width="778" llx="44" lly="-19" urx="713" ury="737"/>
+ <char name="H" width="722" llx="71" lly="0" urx="651" ury="718"/>
+ <char name="I" width="278" llx="64" lly="0" urx="214" ury="718"/>
+ <char name="Iacute" width="278" llx="64" lly="0" urx="329" ury="936"/>
+ <char name="Icircumflex" width="278" llx="-37" lly="0" urx="316" ury="936"/>
+ <char name="Idieresis" width="278" llx="-21" lly="0" urx="300" ury="915"/>
+ <char name="Igrave" width="278" llx="-50" lly="0" urx="214" ury="936"/>
+ <char name="J" width="556" llx="22" lly="-18" urx="484" ury="718"/>
+ <char name="K" width="722" llx="87" lly="0" urx="722" ury="718"/>
+ <char name="L" width="611" llx="76" lly="0" urx="583" ury="718"/>
+ <char name="Lslash" width="611" llx="-20" lly="0" urx="583" ury="718"/>
+ <char name="M" width="833" llx="69" lly="0" urx="765" ury="718"/>
+ <char name="N" width="722" llx="69" lly="0" urx="654" ury="718"/>
+ <char name="Ntilde" width="722" llx="69" lly="0" urx="654" ury="923"/>
+ <char name="O" width="778" llx="44" lly="-19" urx="734" ury="737"/>
+ <char name="OE" width="1000" llx="37" lly="-19" urx="961" ury="737"/>
+ <char name="Oacute" width="778" llx="44" lly="-19" urx="734" ury="936"/>
+ <char name="Ocircumflex" width="778" llx="44" lly="-19" urx="734" ury="936"/>
+ <char name="Odieresis" width="778" llx="44" lly="-19" urx="734" ury="915"/>
+ <char name="Ograve" width="778" llx="44" lly="-19" urx="734" ury="936"/>
+ <char name="Oslash" width="778" llx="33" lly="-27" urx="744" ury="745"/>
+ <char name="Otilde" width="778" llx="44" lly="-19" urx="734" ury="923"/>
+ <char name="P" width="667" llx="76" lly="0" urx="627" ury="718"/>
+ <char name="Q" width="778" llx="44" lly="-52" urx="737" ury="737"/>
+ <char name="R" width="722" llx="76" lly="0" urx="677" ury="718"/>
+ <char name="S" width="667" llx="39" lly="-19" urx="629" ury="737"/>
+ <char name="Scaron" width="667" llx="39" lly="-19" urx="629" ury="936"/>
+ <char name="T" width="611" llx="14" lly="0" urx="598" ury="718"/>
+ <char name="Thorn" width="667" llx="76" lly="0" urx="627" ury="718"/>
+ <char name="U" width="722" llx="72" lly="-19" urx="651" ury="718"/>
+ <char name="Uacute" width="722" llx="72" lly="-19" urx="651" ury="936"/>
+ <char name="Ucircumflex" width="722" llx="72" lly="-19" urx="651" ury="936"/>
+ <char name="Udieresis" width="722" llx="72" lly="-19" urx="651" ury="915"/>
+ <char name="Ugrave" width="722" llx="72" lly="-19" urx="651" ury="936"/>
+ <char name="V" width="667" llx="19" lly="0" urx="648" ury="718"/>
+ <char name="W" width="944" llx="16" lly="0" urx="929" ury="718"/>
+ <char name="X" width="667" llx="14" lly="0" urx="653" ury="718"/>
+ <char name="Y" width="667" llx="15" lly="0" urx="653" ury="718"/>
+ <char name="Yacute" width="667" llx="15" lly="0" urx="653" ury="936"/>
+ <char name="Ydieresis" width="667" llx="15" lly="0" urx="653" ury="915"/>
+ <char name="Z" width="611" llx="25" lly="0" urx="586" ury="718"/>
+ <char name="Zcaron" width="611" llx="25" lly="0" urx="586" ury="936"/>
+ <char name="a" width="556" llx="29" lly="-14" urx="527" ury="546"/>
+ <char name="aacute" width="556" llx="29" lly="-14" urx="527" ury="750"/>
+ <char name="acircumflex" width="556" llx="29" lly="-14" urx="527" ury="750"/>
+ <char name="acute" width="333" llx="108" lly="604" urx="356" ury="750"/>
+ <char name="adieresis" width="556" llx="29" lly="-14" urx="527" ury="729"/>
+ <char name="ae" width="889" llx="29" lly="-14" urx="858" ury="546"/>
+ <char name="agrave" width="556" llx="29" lly="-14" urx="527" ury="750"/>
+ <char name="ampersand" width="722" llx="54" lly="-19" urx="701" ury="718"/>
+ <char name="aring" width="556" llx="29" lly="-14" urx="527" ury="776"/>
+ <char name="asciicircum" width="584" llx="62" lly="323" urx="522" ury="698"/>
+ <char name="asciitilde" width="584" llx="61" lly="163" urx="523" ury="343"/>
+ <char name="asterisk" width="389" llx="27" lly="387" urx="362" ury="718"/>
+ <char name="at" width="975" llx="118" lly="-19" urx="856" ury="737"/>
+ <char name="atilde" width="556" llx="29" lly="-14" urx="527" ury="737"/>
+ <char name="b" width="611" llx="61" lly="-14" urx="578" ury="718"/>
+ <char name="backslash" width="278" llx="-33" lly="-19" urx="311" ury="737"/>
+ <char name="bar" width="280" llx="84" lly="-225" urx="196" ury="775"/>
+ <char name="braceleft" width="389" llx="48" lly="-196" urx="365" ury="722"/>
+ <char name="braceright" width="389" llx="24" lly="-196" urx="341" ury="722"/>
+ <char name="bracketleft" width="333" llx="63" lly="-196" urx="309" ury="722"/>
+ <char name="bracketright" width="333" llx="24" lly="-196" urx="270" ury="722"/>
+ <char name="breve" width="333" llx="-2" lly="604" urx="335" ury="750"/>
+ <char name="brokenbar" width="280" llx="84" lly="-150" urx="196" ury="700"/>
+ <char name="bullet" width="350" llx="10" lly="194" urx="340" ury="524"/>
+ <char name="c" width="556" llx="34" lly="-14" urx="524" ury="546"/>
+ <char name="caron" width="333" llx="-10" lly="604" urx="343" ury="750"/>
+ <char name="ccedilla" width="556" llx="34" lly="-228" urx="524" ury="546"/>
+ <char name="cedilla" width="333" llx="6" lly="-228" urx="245" ury="0"/>
+ <char name="cent" width="556" llx="34" lly="-118" urx="524" ury="628"/>
+ <char name="circumflex" width="333" llx="-10" lly="604" urx="343" ury="750"/>
+ <char name="colon" width="333" llx="92" lly="0" urx="242" ury="512"/>
+ <char name="comma" width="278" llx="64" lly="-168" urx="214" ury="146"/>
+ <char name="copyright" width="737" llx="-11" lly="-19" urx="749" ury="737"/>
+ <char name="currency" width="556" llx="-3" lly="76" urx="559" ury="636"/>
+ <char name="d" width="611" llx="34" lly="-14" urx="551" ury="718"/>
+ <char name="dagger" width="556" llx="36" lly="-171" urx="520" ury="718"/>
+ <char name="daggerdbl" width="556" llx="36" lly="-171" urx="520" ury="718"/>
+ <char name="degree" width="400" llx="57" lly="426" urx="343" ury="712"/>
+ <char name="dieresis" width="333" llx="6" lly="614" urx="327" ury="729"/>
+ <char name="divide" width="584" llx="40" lly="-42" urx="544" ury="548"/>
+ <char name="dollar" width="556" llx="30" lly="-115" urx="523" ury="775"/>
+ <char name="dotaccent" width="333" llx="104" lly="614" urx="230" ury="729"/>
+ <char name="dotlessi" width="278" llx="69" lly="0" urx="209" ury="532"/>
+ <char name="e" width="556" llx="23" lly="-14" urx="528" ury="546"/>
+ <char name="eacute" width="556" llx="23" lly="-14" urx="528" ury="750"/>
+ <char name="ecircumflex" width="556" llx="23" lly="-14" urx="528" ury="750"/>
+ <char name="edieresis" width="556" llx="23" lly="-14" urx="528" ury="729"/>
+ <char name="egrave" width="556" llx="23" lly="-14" urx="528" ury="750"/>
+ <char name="eight" width="556" llx="32" lly="-19" urx="524" ury="710"/>
+ <char name="ellipsis" width="1000" llx="92" lly="0" urx="908" ury="146"/>
+ <char name="emdash" width="1000" llx="0" lly="227" urx="1000" ury="333"/>
+ <char name="endash" width="556" llx="0" lly="227" urx="556" ury="333"/>
+ <char name="equal" width="584" llx="40" lly="87" urx="544" ury="419"/>
+ <char name="eth" width="611" llx="34" lly="-14" urx="578" ury="737"/>
+ <char name="exclam" width="333" llx="90" lly="0" urx="244" ury="718"/>
+ <char name="exclamdown" width="333" llx="90" lly="-186" urx="244" ury="532"/>
+ <char name="f" width="333" llx="10" lly="0" urx="318" ury="727"/>
+ <char name="fi" width="611" llx="10" lly="0" urx="542" ury="727"/>
+ <char name="five" width="556" llx="27" lly="-19" urx="516" ury="698"/>
+ <char name="fl" width="611" llx="10" lly="0" urx="542" ury="727"/>
+ <char name="florin" width="556" llx="-10" lly="-210" urx="516" ury="737"/>
+ <char name="four" width="556" llx="27" lly="0" urx="526" ury="710"/>
+ <char name="fraction" width="167" llx="-170" lly="-19" urx="336" ury="710"/>
+ <char name="g" width="611" llx="40" lly="-217" urx="553" ury="546"/>
+ <char name="germandbls" width="611" llx="69" lly="-14" urx="579" ury="731"/>
+ <char name="grave" width="333" llx="-23" lly="604" urx="225" ury="750"/>
+ <char name="greater" width="584" llx="38" lly="-8" urx="546" ury="514"/>
+ <char name="guillemotleft" width="556" llx="88" lly="76" urx="468" ury="484"/>
+ <char name="guillemotright" width="556" llx="88" lly="76" urx="468" ury="484"/>
+ <char name="guilsinglleft" width="333" llx="83" lly="76" urx="250" ury="484"/>
+ <char name="guilsinglright" width="333" llx="83" lly="76" urx="250" ury="484"/>
+ <char name="h" width="611" llx="65" lly="0" urx="546" ury="718"/>
+ <char name="hungarumlaut" width="333" llx="9" lly="604" urx="486" ury="750"/>
+ <char name="hyphen" width="333" llx="27" lly="215" urx="306" ury="345"/>
+ <char name="i" width="278" llx="69" lly="0" urx="209" ury="725"/>
+ <char name="iacute" width="278" llx="69" lly="0" urx="329" ury="750"/>
+ <char name="icircumflex" width="278" llx="-37" lly="0" urx="316" ury="750"/>
+ <char name="idieresis" width="278" llx="-21" lly="0" urx="300" ury="729"/>
+ <char name="igrave" width="278" llx="-50" lly="0" urx="209" ury="750"/>
+ <char name="j" width="278" llx="3" lly="-214" urx="209" ury="725"/>
+ <char name="k" width="556" llx="69" lly="0" urx="562" ury="718"/>
+ <char name="l" width="278" llx="69" lly="0" urx="209" ury="718"/>
+ <char name="less" width="584" llx="38" lly="-8" urx="546" ury="514"/>
+ <char name="logicalnot" width="584" llx="40" lly="108" urx="544" ury="419"/>
+ <char name="lslash" width="278" llx="-18" lly="0" urx="296" ury="718"/>
+ <char name="m" width="889" llx="64" lly="0" urx="826" ury="546"/>
+ <char name="macron" width="333" llx="-6" lly="604" urx="339" ury="678"/>
+ <char name="minus" width="324" llx="40" lly="197" urx="544" ury="309"/>
+ <char name="mu" width="611" llx="66" lly="-207" urx="545" ury="532"/>
+ <char name="multiply" width="584" llx="40" lly="1" urx="545" ury="505"/>
+ <char name="n" width="611" llx="65" lly="0" urx="546" ury="546"/>
+ <char name="nine" width="556" llx="30" lly="-19" urx="522" ury="710"/>
+ <char name="ntilde" width="611" llx="65" lly="0" urx="546" ury="737"/>
+ <char name="numbersign" width="556" llx="18" lly="0" urx="538" ury="698"/>
+ <char name="o" width="611" llx="34" lly="-14" urx="578" ury="546"/>
+ <char name="oacute" width="611" llx="34" lly="-14" urx="578" ury="750"/>
+ <char name="ocircumflex" width="611" llx="34" lly="-14" urx="578" ury="750"/>
+ <char name="odieresis" width="611" llx="34" lly="-14" urx="578" ury="729"/>
+ <char name="oe" width="944" llx="34" lly="-14" urx="912" ury="546"/>
+ <char name="ogonek" width="333" llx="71" lly="-228" urx="304" ury="0"/>
+ <char name="ograve" width="611" llx="34" lly="-14" urx="578" ury="750"/>
+ <char name="one" width="556" llx="69" lly="0" urx="378" ury="710"/>
+ <char name="onehalf" width="834" llx="26" lly="-19" urx="794" ury="710"/>
+ <char name="onequarter" width="834" llx="26" lly="-19" urx="766" ury="710"/>
+ <char name="onesuperior" width="333" llx="26" lly="283" urx="237" ury="710"/>
+ <char name="ordfeminine" width="370" llx="22" lly="401" urx="347" ury="737"/>
+ <char name="ordmasculine" width="365" llx="6" lly="401" urx="360" ury="737"/>
+ <char name="oslash" width="611" llx="22" lly="-29" urx="589" ury="560"/>
+ <char name="otilde" width="611" llx="34" lly="-14" urx="578" ury="737"/>
+ <char name="p" width="611" llx="62" lly="-207" urx="578" ury="546"/>
+ <char name="paragraph" width="556" llx="-8" lly="-191" urx="539" ury="700"/>
+ <char name="parenleft" width="333" llx="35" lly="-208" urx="314" ury="734"/>
+ <char name="parenright" width="333" llx="19" lly="-208" urx="298" ury="734"/>
+ <char name="percent" width="889" llx="28" lly="-19" urx="861" ury="710"/>
+ <char name="period" width="278" llx="64" lly="0" urx="214" ury="146"/>
+ <char name="periodcentered" width="278" llx="58" lly="172" urx="220" ury="334"/>
+ <char name="perthousand" width="1000" llx="-3" lly="-19" urx="1003" ury="710"/>
+ <char name="plus" width="584" llx="40" lly="0" urx="544" ury="506"/>
+ <char name="plusminus" width="584" llx="40" lly="0" urx="544" ury="506"/>
+ <char name="q" width="611" llx="34" lly="-207" urx="552" ury="546"/>
+ <char name="question" width="611" llx="60" lly="0" urx="556" ury="727"/>
+ <char name="questiondown" width="611" llx="55" lly="-195" urx="551" ury="532"/>
+ <char name="quotedbl" width="474" llx="98" lly="447" urx="376" ury="718"/>
+ <char name="quotedblbase" width="500" llx="64" lly="-146" urx="436" ury="127"/>
+ <char name="quotedblleft" width="500" llx="64" lly="454" urx="436" ury="727"/>
+ <char name="quotedblright" width="500" llx="64" lly="445" urx="436" ury="718"/>
+ <char name="quoteleft" width="278" llx="69" lly="454" urx="209" ury="727"/>
+ <char name="quoteright" width="278" llx="69" lly="445" urx="209" ury="718"/>
+ <char name="quotesinglbase" width="278" llx="69" lly="-146" urx="209" ury="127"/>
+ <char name="quotesingle" width="238" llx="70" lly="447" urx="168" ury="718"/>
+ <char name="r" width="389" llx="64" lly="0" urx="373" ury="546"/>
+ <char name="registered" width="737" llx="-11" lly="-19" urx="748" ury="737"/>
+ <char name="ring" width="333" llx="59" lly="568" urx="275" ury="776"/>
+ <char name="s" width="556" llx="30" lly="-14" urx="519" ury="546"/>
+ <char name="scaron" width="556" llx="30" lly="-14" urx="519" ury="750"/>
+ <char name="section" width="556" llx="34" lly="-184" urx="522" ury="727"/>
+ <char name="semicolon" width="333" llx="92" lly="-168" urx="242" ury="512"/>
+ <char name="seven" width="556" llx="25" lly="0" urx="528" ury="698"/>
+ <char name="six" width="556" llx="31" lly="-19" urx="520" ury="710"/>
+ <char name="slash" width="278" llx="-33" lly="-19" urx="311" ury="737"/>
+ <char name="space" width="278" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="278"/>
- <char name="sterling" width="556"/>
- <char name="t" width="333"/>
- <char name="thorn" width="611"/>
- <char name="three" width="556"/>
- <char name="threequarters" width="834"/>
- <char name="threesuperior" width="333"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="1000"/>
- <char name="two" width="556"/>
- <char name="twosuperior" width="333"/>
- <char name="u" width="611"/>
- <char name="uacute" width="611"/>
- <char name="ucircumflex" width="611"/>
- <char name="udieresis" width="611"/>
- <char name="ugrave" width="611"/>
- <char name="underscore" width="556"/>
- <char name="v" width="556"/>
- <char name="w" width="778"/>
- <char name="x" width="556"/>
- <char name="y" width="556"/>
- <char name="yacute" width="556"/>
- <char name="ydieresis" width="556"/>
- <char name="yen" width="556"/>
- <char name="z" width="500"/>
- <char name="zcaron" width="500"/>
- <char name="zero" width="556"/>
- </widths>
+ <char name="sterling" width="556" llx="28" lly="-16" urx="541" ury="718"/>
+ <char name="t" width="333" llx="10" lly="-6" urx="309" ury="676"/>
+ <char name="thorn" width="611" llx="62" lly="-208" urx="578" ury="718"/>
+ <char name="three" width="556" llx="27" lly="-19" urx="516" ury="710"/>
+ <char name="threequarters" width="834" llx="16" lly="-19" urx="799" ury="710"/>
+ <char name="threesuperior" width="333" llx="8" lly="271" urx="326" ury="710"/>
+ <char name="tilde" width="333" llx="-17" lly="610" urx="350" ury="737"/>
+ <char name="trademark" width="1000" llx="44" lly="306" urx="956" ury="718"/>
+ <char name="two" width="556" llx="26" lly="0" urx="511" ury="710"/>
+ <char name="twosuperior" width="333" llx="9" lly="283" urx="324" ury="710"/>
+ <char name="u" width="611" llx="66" lly="-14" urx="545" ury="532"/>
+ <char name="uacute" width="611" llx="66" lly="-14" urx="545" ury="750"/>
+ <char name="ucircumflex" width="611" llx="66" lly="-14" urx="545" ury="750"/>
+ <char name="udieresis" width="611" llx="66" lly="-14" urx="545" ury="729"/>
+ <char name="ugrave" width="611" llx="66" lly="-14" urx="545" ury="750"/>
+ <char name="underscore" width="556" llx="0" lly="-125" urx="556" ury="-75"/>
+ <char name="v" width="556" llx="13" lly="0" urx="543" ury="532"/>
+ <char name="w" width="778" llx="10" lly="0" urx="769" ury="532"/>
+ <char name="x" width="556" llx="15" lly="0" urx="541" ury="532"/>
+ <char name="y" width="556" llx="10" lly="-214" urx="539" ury="532"/>
+ <char name="yacute" width="556" llx="10" lly="-214" urx="539" ury="750"/>
+ <char name="ydieresis" width="556" llx="10" lly="-214" urx="539" ury="729"/>
+ <char name="yen" width="556" llx="-9" lly="0" urx="565" ury="698"/>
+ <char name="z" width="500" llx="20" lly="0" urx="480" ury="532"/>
+ <char name="zcaron" width="500" llx="20" lly="0" urx="480" ury="750"/>
+ <char name="zero" width="556" llx="32" lly="-19" urx="524" ury="710"/>
+ </char-metrics>
<kerning kpx1="107">
<pair kern="-15" kpx2="111"/>
</kerning>
diff --git a/src/codegen/fonts/HelveticaBoldOblique.xml b/src/codegen/fonts/HelveticaBoldOblique.xml
index c11e0ba18..e1d0fd294 100644
--- a/src/codegen/fonts/HelveticaBoldOblique.xml
+++ b/src/codegen/fonts/HelveticaBoldOblique.xml
@@ -22,245 +22,246 @@
<family-name>Helvetica</family-name>
<class-name>HelveticaBoldOblique</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>718</cap-height>
<x-height>532</x-height>
<ascender>718</ascender>
<descender>-207</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="722"/>
- <char name="AE" width="1000"/>
- <char name="Aacute" width="722"/>
- <char name="Acircumflex" width="722"/>
- <char name="Adieresis" width="722"/>
- <char name="Agrave" width="722"/>
- <char name="Aring" width="722"/>
- <char name="Atilde" width="722"/>
- <char name="B" width="722"/>
- <char name="C" width="722"/>
- <char name="Ccedilla" width="722"/>
- <char name="D" width="722"/>
- <char name="E" width="667"/>
- <char name="Eacute" width="667"/>
- <char name="Ecircumflex" width="667"/>
- <char name="Edieresis" width="667"/>
- <char name="Egrave" width="667"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="556"/>
- <char name="F" width="611"/>
- <char name="G" width="778"/>
- <char name="H" width="722"/>
- <char name="I" width="278"/>
- <char name="Iacute" width="278"/>
- <char name="Icircumflex" width="278"/>
- <char name="Idieresis" width="278"/>
- <char name="Igrave" width="278"/>
- <char name="J" width="556"/>
- <char name="K" width="722"/>
- <char name="L" width="611"/>
- <char name="Lslash" width="611"/>
- <char name="M" width="833"/>
- <char name="N" width="722"/>
- <char name="Ntilde" width="722"/>
- <char name="O" width="778"/>
- <char name="OE" width="1000"/>
- <char name="Oacute" width="778"/>
- <char name="Ocircumflex" width="778"/>
- <char name="Odieresis" width="778"/>
- <char name="Ograve" width="778"/>
- <char name="Oslash" width="778"/>
- <char name="Otilde" width="778"/>
- <char name="P" width="667"/>
- <char name="Q" width="778"/>
- <char name="R" width="722"/>
- <char name="S" width="667"/>
- <char name="Scaron" width="667"/>
- <char name="T" width="611"/>
- <char name="Thorn" width="667"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="667"/>
- <char name="W" width="944"/>
- <char name="X" width="667"/>
- <char name="Y" width="667"/>
- <char name="Yacute" width="667"/>
- <char name="Ydieresis" width="667"/>
- <char name="Z" width="611"/>
- <char name="Zcaron" width="611"/>
- <char name="a" width="556"/>
- <char name="aacute" width="556"/>
- <char name="acircumflex" width="556"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="556"/>
- <char name="ae" width="889"/>
- <char name="agrave" width="556"/>
- <char name="ampersand" width="722"/>
- <char name="aring" width="556"/>
- <char name="asciicircum" width="584"/>
- <char name="asciitilde" width="584"/>
- <char name="asterisk" width="389"/>
- <char name="at" width="975"/>
- <char name="atilde" width="556"/>
- <char name="b" width="611"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="280"/>
- <char name="braceleft" width="389"/>
- <char name="braceright" width="389"/>
- <char name="bracketleft" width="333"/>
- <char name="bracketright" width="333"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="280"/>
- <char name="bullet" width="350"/>
- <char name="c" width="556"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="556"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="556"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="333"/>
- <char name="comma" width="278"/>
- <char name="copyright" width="737"/>
- <char name="currency" width="556"/>
- <char name="d" width="611"/>
- <char name="dagger" width="556"/>
- <char name="daggerdbl" width="556"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="584"/>
- <char name="dollar" width="556"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="556"/>
- <char name="eacute" width="556"/>
- <char name="ecircumflex" width="556"/>
- <char name="edieresis" width="556"/>
- <char name="egrave" width="556"/>
- <char name="eight" width="556"/>
- <char name="ellipsis" width="1000"/>
- <char name="emdash" width="1000"/>
- <char name="endash" width="556"/>
- <char name="equal" width="584"/>
- <char name="eth" width="611"/>
- <char name="exclam" width="333"/>
- <char name="exclamdown" width="333"/>
- <char name="f" width="333"/>
- <char name="fi" width="611"/>
- <char name="five" width="556"/>
- <char name="fl" width="611"/>
- <char name="florin" width="556"/>
- <char name="four" width="556"/>
- <char name="fraction" width="167"/>
- <char name="g" width="611"/>
- <char name="germandbls" width="611"/>
- <char name="grave" width="333"/>
- <char name="greater" width="584"/>
- <char name="guillemotleft" width="556"/>
- <char name="guillemotright" width="556"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="611"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="278"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="278"/>
- <char name="k" width="556"/>
- <char name="l" width="278"/>
- <char name="less" width="584"/>
- <char name="logicalnot" width="584"/>
- <char name="lslash" width="278"/>
- <char name="m" width="889"/>
- <char name="macron" width="333"/>
- <char name="minus" width="324"/>
- <char name="mu" width="611"/>
- <char name="multiply" width="584"/>
- <char name="n" width="611"/>
- <char name="nine" width="556"/>
- <char name="ntilde" width="611"/>
- <char name="numbersign" width="556"/>
- <char name="o" width="611"/>
- <char name="oacute" width="611"/>
- <char name="ocircumflex" width="611"/>
- <char name="odieresis" width="611"/>
- <char name="oe" width="944"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="611"/>
- <char name="one" width="556"/>
- <char name="onehalf" width="834"/>
- <char name="onequarter" width="834"/>
- <char name="onesuperior" width="333"/>
- <char name="ordfeminine" width="370"/>
- <char name="ordmasculine" width="365"/>
- <char name="oslash" width="611"/>
- <char name="otilde" width="611"/>
- <char name="p" width="611"/>
- <char name="paragraph" width="556"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="889"/>
- <char name="period" width="278"/>
- <char name="periodcentered" width="278"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="584"/>
- <char name="plusminus" width="584"/>
- <char name="q" width="611"/>
- <char name="question" width="611"/>
- <char name="questiondown" width="611"/>
- <char name="quotedbl" width="474"/>
- <char name="quotedblbase" width="500"/>
- <char name="quotedblleft" width="500"/>
- <char name="quotedblright" width="500"/>
- <char name="quoteleft" width="278"/>
- <char name="quoteright" width="278"/>
- <char name="quotesinglbase" width="278"/>
- <char name="quotesingle" width="238"/>
- <char name="r" width="389"/>
- <char name="registered" width="737"/>
- <char name="ring" width="333"/>
- <char name="s" width="556"/>
- <char name="scaron" width="556"/>
- <char name="section" width="556"/>
- <char name="semicolon" width="333"/>
- <char name="seven" width="556"/>
- <char name="six" width="556"/>
- <char name="slash" width="278"/>
- <char name="space" width="278"/>
+ <char-metrics>
+ <char name="A" width="722" llx="20" lly="0" urx="702" ury="718"/>
+ <char name="AE" width="1000" llx="5" lly="0" urx="1100" ury="718"/>
+ <char name="Aacute" width="722" llx="20" lly="0" urx="750" ury="936"/>
+ <char name="Acircumflex" width="722" llx="20" lly="0" urx="706" ury="936"/>
+ <char name="Adieresis" width="722" llx="20" lly="0" urx="716" ury="915"/>
+ <char name="Agrave" width="722" llx="20" lly="0" urx="702" ury="936"/>
+ <char name="Aring" width="722" llx="20" lly="0" urx="702" ury="962"/>
+ <char name="Atilde" width="722" llx="20" lly="0" urx="741" ury="923"/>
+ <char name="B" width="722" llx="76" lly="0" urx="764" ury="718"/>
+ <char name="C" width="722" llx="107" lly="-19" urx="789" ury="737"/>
+ <char name="Ccedilla" width="722" llx="107" lly="-228" urx="789" ury="737"/>
+ <char name="D" width="722" llx="76" lly="0" urx="777" ury="718"/>
+ <char name="E" width="667" llx="76" lly="0" urx="757" ury="718"/>
+ <char name="Eacute" width="667" llx="76" lly="0" urx="757" ury="936"/>
+ <char name="Ecircumflex" width="667" llx="76" lly="0" urx="757" ury="936"/>
+ <char name="Edieresis" width="667" llx="76" lly="0" urx="757" ury="915"/>
+ <char name="Egrave" width="667" llx="76" lly="0" urx="757" ury="936"/>
+ <char name="Eth" width="722" llx="62" lly="0" urx="777" ury="718"/>
+ <char name="Euro" width="556" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="611" llx="76" lly="0" urx="740" ury="718"/>
+ <char name="G" width="778" llx="108" lly="-19" urx="817" ury="737"/>
+ <char name="H" width="722" llx="71" lly="0" urx="804" ury="718"/>
+ <char name="I" width="278" llx="64" lly="0" urx="367" ury="718"/>
+ <char name="Iacute" width="278" llx="64" lly="0" urx="528" ury="936"/>
+ <char name="Icircumflex" width="278" llx="64" lly="0" urx="484" ury="936"/>
+ <char name="Idieresis" width="278" llx="64" lly="0" urx="494" ury="915"/>
+ <char name="Igrave" width="278" llx="64" lly="0" urx="367" ury="936"/>
+ <char name="J" width="556" llx="60" lly="-18" urx="637" ury="718"/>
+ <char name="K" width="722" llx="87" lly="0" urx="858" ury="718"/>
+ <char name="L" width="611" llx="76" lly="0" urx="611" ury="718"/>
+ <char name="Lslash" width="611" llx="34" lly="0" urx="611" ury="718"/>
+ <char name="M" width="833" llx="69" lly="0" urx="918" ury="718"/>
+ <char name="N" width="722" llx="69" lly="0" urx="807" ury="718"/>
+ <char name="Ntilde" width="722" llx="69" lly="0" urx="807" ury="923"/>
+ <char name="O" width="778" llx="107" lly="-19" urx="823" ury="737"/>
+ <char name="OE" width="1000" llx="99" lly="-19" urx="1114" ury="737"/>
+ <char name="Oacute" width="778" llx="107" lly="-19" urx="823" ury="936"/>
+ <char name="Ocircumflex" width="778" llx="107" lly="-19" urx="823" ury="936"/>
+ <char name="Odieresis" width="778" llx="107" lly="-19" urx="823" ury="915"/>
+ <char name="Ograve" width="778" llx="107" lly="-19" urx="823" ury="936"/>
+ <char name="Oslash" width="778" llx="35" lly="-27" urx="894" ury="745"/>
+ <char name="Otilde" width="778" llx="107" lly="-19" urx="823" ury="923"/>
+ <char name="P" width="667" llx="76" lly="0" urx="738" ury="718"/>
+ <char name="Q" width="778" llx="107" lly="-52" urx="823" ury="737"/>
+ <char name="R" width="722" llx="76" lly="0" urx="778" ury="718"/>
+ <char name="S" width="667" llx="81" lly="-19" urx="718" ury="737"/>
+ <char name="Scaron" width="667" llx="81" lly="-19" urx="718" ury="936"/>
+ <char name="T" width="611" llx="140" lly="0" urx="751" ury="718"/>
+ <char name="Thorn" width="667" llx="76" lly="0" urx="716" ury="718"/>
+ <char name="U" width="722" llx="116" lly="-19" urx="804" ury="718"/>
+ <char name="Uacute" width="722" llx="116" lly="-19" urx="804" ury="936"/>
+ <char name="Ucircumflex" width="722" llx="116" lly="-19" urx="804" ury="936"/>
+ <char name="Udieresis" width="722" llx="116" lly="-19" urx="804" ury="915"/>
+ <char name="Ugrave" width="722" llx="116" lly="-19" urx="804" ury="936"/>
+ <char name="V" width="667" llx="172" lly="0" urx="801" ury="718"/>
+ <char name="W" width="944" llx="169" lly="0" urx="1082" ury="718"/>
+ <char name="X" width="667" llx="14" lly="0" urx="791" ury="718"/>
+ <char name="Y" width="667" llx="168" lly="0" urx="806" ury="718"/>
+ <char name="Yacute" width="667" llx="168" lly="0" urx="806" ury="936"/>
+ <char name="Ydieresis" width="667" llx="168" lly="0" urx="806" ury="915"/>
+ <char name="Z" width="611" llx="25" lly="0" urx="737" ury="718"/>
+ <char name="Zcaron" width="611" llx="25" lly="0" urx="737" ury="936"/>
+ <char name="a" width="556" llx="55" lly="-14" urx="583" ury="546"/>
+ <char name="aacute" width="556" llx="55" lly="-14" urx="627" ury="750"/>
+ <char name="acircumflex" width="556" llx="55" lly="-14" urx="583" ury="750"/>
+ <char name="acute" width="333" llx="236" lly="604" urx="515" ury="750"/>
+ <char name="adieresis" width="556" llx="55" lly="-14" urx="594" ury="729"/>
+ <char name="ae" width="889" llx="56" lly="-14" urx="923" ury="546"/>
+ <char name="agrave" width="556" llx="55" lly="-14" urx="583" ury="750"/>
+ <char name="ampersand" width="722" llx="89" lly="-19" urx="732" ury="718"/>
+ <char name="aring" width="556" llx="55" lly="-14" urx="583" ury="776"/>
+ <char name="asciicircum" width="584" llx="131" lly="323" urx="591" ury="698"/>
+ <char name="asciitilde" width="584" llx="115" lly="163" urx="577" ury="343"/>
+ <char name="asterisk" width="389" llx="146" lly="387" urx="481" ury="718"/>
+ <char name="at" width="975" llx="186" lly="-19" urx="954" ury="737"/>
+ <char name="atilde" width="556" llx="55" lly="-14" urx="619" ury="737"/>
+ <char name="b" width="611" llx="61" lly="-14" urx="645" ury="718"/>
+ <char name="backslash" width="278" llx="124" lly="-19" urx="307" ury="737"/>
+ <char name="bar" width="280" llx="36" lly="-225" urx="361" ury="775"/>
+ <char name="braceleft" width="389" llx="94" lly="-196" urx="518" ury="722"/>
+ <char name="braceright" width="389" llx="-18" lly="-196" urx="407" ury="722"/>
+ <char name="bracketleft" width="333" llx="21" lly="-196" urx="462" ury="722"/>
+ <char name="bracketright" width="333" llx="-18" lly="-196" urx="423" ury="722"/>
+ <char name="breve" width="333" llx="156" lly="604" urx="494" ury="750"/>
+ <char name="brokenbar" width="280" llx="52" lly="-150" urx="345" ury="700"/>
+ <char name="bullet" width="350" llx="83" lly="194" urx="420" ury="524"/>
+ <char name="c" width="556" llx="79" lly="-14" urx="599" ury="546"/>
+ <char name="caron" width="333" llx="149" lly="604" urx="502" ury="750"/>
+ <char name="ccedilla" width="556" llx="79" lly="-228" urx="599" ury="546"/>
+ <char name="cedilla" width="333" llx="-37" lly="-228" urx="220" ury="0"/>
+ <char name="cent" width="556" llx="79" lly="-118" urx="599" ury="628"/>
+ <char name="circumflex" width="333" llx="118" lly="604" urx="471" ury="750"/>
+ <char name="colon" width="333" llx="92" lly="0" urx="351" ury="512"/>
+ <char name="comma" width="278" llx="28" lly="-168" urx="245" ury="146"/>
+ <char name="copyright" width="737" llx="56" lly="-19" urx="835" ury="737"/>
+ <char name="currency" width="556" llx="27" lly="76" urx="680" ury="636"/>
+ <char name="d" width="611" llx="82" lly="-14" urx="704" ury="718"/>
+ <char name="dagger" width="556" llx="118" lly="-171" urx="626" ury="718"/>
+ <char name="daggerdbl" width="556" llx="46" lly="-171" urx="628" ury="718"/>
+ <char name="degree" width="400" llx="175" lly="426" urx="467" ury="712"/>
+ <char name="dieresis" width="333" llx="137" lly="614" urx="482" ury="729"/>
+ <char name="divide" width="584" llx="82" lly="-42" urx="610" ury="548"/>
+ <char name="dollar" width="556" llx="67" lly="-115" urx="622" ury="775"/>
+ <char name="dotaccent" width="333" llx="235" lly="614" urx="385" ury="729"/>
+ <char name="dotlessi" width="278" llx="69" lly="0" urx="322" ury="532"/>
+ <char name="e" width="556" llx="70" lly="-14" urx="593" ury="546"/>
+ <char name="eacute" width="556" llx="70" lly="-14" urx="627" ury="750"/>
+ <char name="ecircumflex" width="556" llx="70" lly="-14" urx="593" ury="750"/>
+ <char name="edieresis" width="556" llx="70" lly="-14" urx="594" ury="729"/>
+ <char name="egrave" width="556" llx="70" lly="-14" urx="593" ury="750"/>
+ <char name="eight" width="556" llx="69" lly="-19" urx="616" ury="710"/>
+ <char name="ellipsis" width="1000" llx="92" lly="0" urx="939" ury="146"/>
+ <char name="emdash" width="1000" llx="48" lly="227" urx="1071" ury="333"/>
+ <char name="endash" width="556" llx="48" lly="227" urx="627" ury="333"/>
+ <char name="equal" width="584" llx="58" lly="87" urx="633" ury="419"/>
+ <char name="eth" width="611" llx="82" lly="-14" urx="670" ury="737"/>
+ <char name="exclam" width="333" llx="94" lly="0" urx="397" ury="718"/>
+ <char name="exclamdown" width="333" llx="50" lly="-186" urx="353" ury="532"/>
+ <char name="f" width="333" llx="87" lly="0" urx="469" ury="727"/>
+ <char name="fi" width="611" llx="87" lly="0" urx="696" ury="727"/>
+ <char name="five" width="556" llx="64" lly="-19" urx="636" ury="698"/>
+ <char name="fl" width="611" llx="87" lly="0" urx="695" ury="727"/>
+ <char name="florin" width="556" llx="-50" lly="-210" urx="669" ury="737"/>
+ <char name="four" width="556" llx="60" lly="0" urx="598" ury="710"/>
+ <char name="fraction" width="167" llx="-174" lly="-19" urx="487" ury="710"/>
+ <char name="g" width="611" llx="38" lly="-217" urx="666" ury="546"/>
+ <char name="germandbls" width="611" llx="69" lly="-14" urx="657" ury="731"/>
+ <char name="grave" width="333" llx="136" lly="604" urx="353" ury="750"/>
+ <char name="greater" width="584" llx="36" lly="-8" urx="609" ury="514"/>
+ <char name="guillemotleft" width="556" llx="135" lly="76" urx="571" ury="484"/>
+ <char name="guillemotright" width="556" llx="104" lly="76" urx="540" ury="484"/>
+ <char name="guilsinglleft" width="333" llx="130" lly="76" urx="353" ury="484"/>
+ <char name="guilsinglright" width="333" llx="99" lly="76" urx="322" ury="484"/>
+ <char name="h" width="611" llx="65" lly="0" urx="629" ury="718"/>
+ <char name="hungarumlaut" width="333" llx="137" lly="604" urx="645" ury="750"/>
+ <char name="hyphen" width="333" llx="73" lly="215" urx="379" ury="345"/>
+ <char name="i" width="278" llx="69" lly="0" urx="363" ury="725"/>
+ <char name="iacute" width="278" llx="69" lly="0" urx="488" ury="750"/>
+ <char name="icircumflex" width="278" llx="69" lly="0" urx="444" ury="750"/>
+ <char name="idieresis" width="278" llx="69" lly="0" urx="455" ury="729"/>
+ <char name="igrave" width="278" llx="69" lly="0" urx="326" ury="750"/>
+ <char name="j" width="278" llx="-42" lly="-214" urx="363" ury="725"/>
+ <char name="k" width="556" llx="69" lly="0" urx="670" ury="718"/>
+ <char name="l" width="278" llx="69" lly="0" urx="362" ury="718"/>
+ <char name="less" width="584" llx="82" lly="-8" urx="655" ury="514"/>
+ <char name="logicalnot" width="584" llx="105" lly="108" urx="633" ury="419"/>
+ <char name="lslash" width="278" llx="40" lly="0" urx="407" ury="718"/>
+ <char name="m" width="889" llx="64" lly="0" urx="909" ury="546"/>
+ <char name="macron" width="333" llx="122" lly="604" urx="483" ury="678"/>
+ <char name="minus" width="324" llx="82" lly="197" urx="610" ury="309"/>
+ <char name="mu" width="611" llx="22" lly="-207" urx="658" ury="532"/>
+ <char name="multiply" width="584" llx="57" lly="1" urx="635" ury="505"/>
+ <char name="n" width="611" llx="65" lly="0" urx="629" ury="546"/>
+ <char name="nine" width="556" llx="78" lly="-19" urx="615" ury="710"/>
+ <char name="ntilde" width="611" llx="65" lly="0" urx="646" ury="737"/>
+ <char name="numbersign" width="556" llx="60" lly="0" urx="644" ury="698"/>
+ <char name="o" width="611" llx="82" lly="-14" urx="643" ury="546"/>
+ <char name="oacute" width="611" llx="82" lly="-14" urx="654" ury="750"/>
+ <char name="ocircumflex" width="611" llx="82" lly="-14" urx="643" ury="750"/>
+ <char name="odieresis" width="611" llx="82" lly="-14" urx="643" ury="729"/>
+ <char name="oe" width="944" llx="82" lly="-14" urx="977" ury="546"/>
+ <char name="ogonek" width="333" llx="41" lly="-228" urx="264" ury="0"/>
+ <char name="ograve" width="611" llx="82" lly="-14" urx="643" ury="750"/>
+ <char name="one" width="556" llx="173" lly="0" urx="529" ury="710"/>
+ <char name="onehalf" width="834" llx="132" lly="-19" urx="858" ury="710"/>
+ <char name="onequarter" width="834" llx="132" lly="-19" urx="806" ury="710"/>
+ <char name="onesuperior" width="333" llx="148" lly="283" urx="388" ury="710"/>
+ <char name="ordfeminine" width="370" llx="125" lly="401" urx="465" ury="737"/>
+ <char name="ordmasculine" width="365" llx="123" lly="401" urx="485" ury="737"/>
+ <char name="oslash" width="611" llx="22" lly="-29" urx="701" ury="560"/>
+ <char name="otilde" width="611" llx="82" lly="-14" urx="646" ury="737"/>
+ <char name="p" width="611" llx="18" lly="-207" urx="645" ury="546"/>
+ <char name="paragraph" width="556" llx="98" lly="-191" urx="688" ury="700"/>
+ <char name="parenleft" width="333" llx="76" lly="-208" urx="470" ury="734"/>
+ <char name="parenright" width="333" llx="-25" lly="-208" urx="369" ury="734"/>
+ <char name="percent" width="889" llx="136" lly="-19" urx="901" ury="710"/>
+ <char name="period" width="278" llx="64" lly="0" urx="245" ury="146"/>
+ <char name="periodcentered" width="278" llx="110" lly="172" urx="276" ury="334"/>
+ <char name="perthousand" width="1000" llx="76" lly="-19" urx="1038" ury="710"/>
+ <char name="plus" width="584" llx="82" lly="0" urx="610" ury="506"/>
+ <char name="plusminus" width="584" llx="40" lly="0" urx="625" ury="506"/>
+ <char name="q" width="611" llx="80" lly="-207" urx="665" ury="546"/>
+ <char name="question" width="611" llx="165" lly="0" urx="671" ury="727"/>
+ <char name="questiondown" width="611" llx="53" lly="-195" urx="559" ury="532"/>
+ <char name="quotedbl" width="474" llx="193" lly="447" urx="529" ury="718"/>
+ <char name="quotedblbase" width="500" llx="36" lly="-146" urx="463" ury="127"/>
+ <char name="quotedblleft" width="500" llx="160" lly="454" urx="588" ury="727"/>
+ <char name="quotedblright" width="500" llx="162" lly="445" urx="589" ury="718"/>
+ <char name="quoteleft" width="278" llx="165" lly="454" urx="361" ury="727"/>
+ <char name="quoteright" width="278" llx="167" lly="445" urx="362" ury="718"/>
+ <char name="quotesinglbase" width="278" llx="41" lly="-146" urx="236" ury="127"/>
+ <char name="quotesingle" width="238" llx="165" lly="447" urx="321" ury="718"/>
+ <char name="r" width="389" llx="64" lly="0" urx="489" ury="546"/>
+ <char name="registered" width="737" llx="55" lly="-19" urx="834" ury="737"/>
+ <char name="ring" width="333" llx="200" lly="568" urx="420" ury="776"/>
+ <char name="s" width="556" llx="63" lly="-14" urx="584" ury="546"/>
+ <char name="scaron" width="556" llx="63" lly="-14" urx="614" ury="750"/>
+ <char name="section" width="556" llx="61" lly="-184" urx="598" ury="727"/>
+ <char name="semicolon" width="333" llx="56" lly="-168" urx="351" ury="512"/>
+ <char name="seven" width="556" llx="125" lly="0" urx="676" ury="698"/>
+ <char name="six" width="556" llx="85" lly="-19" urx="619" ury="710"/>
+ <char name="slash" width="278" llx="-37" lly="-19" urx="468" ury="737"/>
+ <char name="space" width="278" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="278"/>
- <char name="sterling" width="556"/>
- <char name="t" width="333"/>
- <char name="thorn" width="611"/>
- <char name="three" width="556"/>
- <char name="threequarters" width="834"/>
- <char name="threesuperior" width="333"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="1000"/>
- <char name="two" width="556"/>
- <char name="twosuperior" width="333"/>
- <char name="u" width="611"/>
- <char name="uacute" width="611"/>
- <char name="ucircumflex" width="611"/>
- <char name="udieresis" width="611"/>
- <char name="ugrave" width="611"/>
- <char name="underscore" width="556"/>
- <char name="v" width="556"/>
- <char name="w" width="778"/>
- <char name="x" width="556"/>
- <char name="y" width="556"/>
- <char name="yacute" width="556"/>
- <char name="ydieresis" width="556"/>
- <char name="yen" width="556"/>
- <char name="z" width="500"/>
- <char name="zcaron" width="500"/>
- <char name="zero" width="556"/>
- </widths>
+ <char name="sterling" width="556" llx="50" lly="-16" urx="635" ury="718"/>
+ <char name="t" width="333" llx="100" lly="-6" urx="422" ury="676"/>
+ <char name="thorn" width="611" llx="18" lly="-208" urx="645" ury="718"/>
+ <char name="three" width="556" llx="65" lly="-19" urx="608" ury="710"/>
+ <char name="threequarters" width="834" llx="99" lly="-19" urx="839" ury="710"/>
+ <char name="threesuperior" width="333" llx="91" lly="271" urx="441" ury="710"/>
+ <char name="tilde" width="333" llx="113" lly="610" urx="507" ury="737"/>
+ <char name="trademark" width="1000" llx="179" lly="306" urx="1109" ury="718"/>
+ <char name="two" width="556" llx="26" lly="0" urx="619" ury="710"/>
+ <char name="twosuperior" width="333" llx="69" lly="283" urx="449" ury="710"/>
+ <char name="u" width="611" llx="98" lly="-14" urx="658" ury="532"/>
+ <char name="uacute" width="611" llx="98" lly="-14" urx="658" ury="750"/>
+ <char name="ucircumflex" width="611" llx="98" lly="-14" urx="658" ury="750"/>
+ <char name="udieresis" width="611" llx="98" lly="-14" urx="658" ury="729"/>
+ <char name="ugrave" width="611" llx="98" lly="-14" urx="658" ury="750"/>
+ <char name="underscore" width="556" llx="-27" lly="-125" urx="540" ury="-75"/>
+ <char name="v" width="556" llx="126" lly="0" urx="656" ury="532"/>
+ <char name="w" width="778" llx="123" lly="0" urx="882" ury="532"/>
+ <char name="x" width="556" llx="15" lly="0" urx="648" ury="532"/>
+ <char name="y" width="556" llx="42" lly="-214" urx="652" ury="532"/>
+ <char name="yacute" width="556" llx="42" lly="-214" urx="652" ury="750"/>
+ <char name="ydieresis" width="556" llx="42" lly="-214" urx="652" ury="729"/>
+ <char name="yen" width="556" llx="60" lly="0" urx="713" ury="698"/>
+ <char name="z" width="500" llx="20" lly="0" urx="583" ury="532"/>
+ <char name="zcaron" width="500" llx="20" lly="0" urx="586" ury="750"/>
+ <char name="zero" width="556" llx="86" lly="-19" urx="617" ury="710"/>
+ </char-metrics>
<kerning kpx1="107">
<pair kern="-15" kpx2="111"/>
</kerning>
diff --git a/src/codegen/fonts/HelveticaOblique.xml b/src/codegen/fonts/HelveticaOblique.xml
index 7f651a5d9..efa1165f2 100644
--- a/src/codegen/fonts/HelveticaOblique.xml
+++ b/src/codegen/fonts/HelveticaOblique.xml
@@ -22,245 +22,246 @@
<family-name>Helvetica</family-name>
<class-name>HelveticaOblique</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>718</cap-height>
<x-height>523</x-height>
<ascender>718</ascender>
<descender>-207</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="667"/>
- <char name="AE" width="1000"/>
- <char name="Aacute" width="667"/>
- <char name="Acircumflex" width="667"/>
- <char name="Adieresis" width="667"/>
- <char name="Agrave" width="667"/>
- <char name="Aring" width="667"/>
- <char name="Atilde" width="667"/>
- <char name="B" width="667"/>
- <char name="C" width="722"/>
- <char name="Ccedilla" width="722"/>
- <char name="D" width="722"/>
- <char name="E" width="667"/>
- <char name="Eacute" width="667"/>
- <char name="Ecircumflex" width="667"/>
- <char name="Edieresis" width="667"/>
- <char name="Egrave" width="667"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="556"/>
- <char name="F" width="611"/>
- <char name="G" width="778"/>
- <char name="H" width="722"/>
- <char name="I" width="278"/>
- <char name="Iacute" width="278"/>
- <char name="Icircumflex" width="278"/>
- <char name="Idieresis" width="278"/>
- <char name="Igrave" width="278"/>
- <char name="J" width="500"/>
- <char name="K" width="667"/>
- <char name="L" width="556"/>
- <char name="Lslash" width="556"/>
- <char name="M" width="833"/>
- <char name="N" width="722"/>
- <char name="Ntilde" width="722"/>
- <char name="O" width="778"/>
- <char name="OE" width="1000"/>
- <char name="Oacute" width="778"/>
- <char name="Ocircumflex" width="778"/>
- <char name="Odieresis" width="778"/>
- <char name="Ograve" width="778"/>
- <char name="Oslash" width="778"/>
- <char name="Otilde" width="778"/>
- <char name="P" width="667"/>
- <char name="Q" width="778"/>
- <char name="R" width="722"/>
- <char name="S" width="667"/>
- <char name="Scaron" width="667"/>
- <char name="T" width="611"/>
- <char name="Thorn" width="667"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="667"/>
- <char name="W" width="944"/>
- <char name="X" width="667"/>
- <char name="Y" width="667"/>
- <char name="Yacute" width="667"/>
- <char name="Ydieresis" width="667"/>
- <char name="Z" width="611"/>
- <char name="Zcaron" width="611"/>
- <char name="a" width="556"/>
- <char name="aacute" width="556"/>
- <char name="acircumflex" width="556"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="556"/>
- <char name="ae" width="889"/>
- <char name="agrave" width="556"/>
- <char name="ampersand" width="667"/>
- <char name="aring" width="556"/>
- <char name="asciicircum" width="469"/>
- <char name="asciitilde" width="584"/>
- <char name="asterisk" width="389"/>
- <char name="at" width="1015"/>
- <char name="atilde" width="556"/>
- <char name="b" width="556"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="260"/>
- <char name="braceleft" width="334"/>
- <char name="braceright" width="334"/>
- <char name="bracketleft" width="278"/>
- <char name="bracketright" width="278"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="260"/>
- <char name="bullet" width="350"/>
- <char name="c" width="500"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="500"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="556"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="278"/>
- <char name="comma" width="278"/>
- <char name="copyright" width="737"/>
- <char name="currency" width="556"/>
- <char name="d" width="556"/>
- <char name="dagger" width="556"/>
- <char name="daggerdbl" width="556"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="584"/>
- <char name="dollar" width="556"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="556"/>
- <char name="eacute" width="556"/>
- <char name="ecircumflex" width="556"/>
- <char name="edieresis" width="556"/>
- <char name="egrave" width="556"/>
- <char name="eight" width="556"/>
- <char name="ellipsis" width="1000"/>
- <char name="emdash" width="1000"/>
- <char name="endash" width="556"/>
- <char name="equal" width="584"/>
- <char name="eth" width="556"/>
- <char name="exclam" width="278"/>
- <char name="exclamdown" width="333"/>
- <char name="f" width="278"/>
- <char name="fi" width="500"/>
- <char name="five" width="556"/>
- <char name="fl" width="500"/>
- <char name="florin" width="556"/>
- <char name="four" width="556"/>
- <char name="fraction" width="167"/>
- <char name="g" width="556"/>
- <char name="germandbls" width="611"/>
- <char name="grave" width="333"/>
- <char name="greater" width="584"/>
- <char name="guillemotleft" width="556"/>
- <char name="guillemotright" width="556"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="556"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="222"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="222"/>
- <char name="k" width="500"/>
- <char name="l" width="222"/>
- <char name="less" width="584"/>
- <char name="logicalnot" width="584"/>
- <char name="lslash" width="222"/>
- <char name="m" width="833"/>
- <char name="macron" width="333"/>
- <char name="minus" width="584"/>
- <char name="mu" width="556"/>
- <char name="multiply" width="584"/>
- <char name="n" width="556"/>
- <char name="nine" width="556"/>
- <char name="ntilde" width="556"/>
- <char name="numbersign" width="556"/>
- <char name="o" width="556"/>
- <char name="oacute" width="556"/>
- <char name="ocircumflex" width="556"/>
- <char name="odieresis" width="556"/>
- <char name="oe" width="944"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="556"/>
- <char name="one" width="556"/>
- <char name="onehalf" width="834"/>
- <char name="onequarter" width="834"/>
- <char name="onesuperior" width="333"/>
- <char name="ordfeminine" width="370"/>
- <char name="ordmasculine" width="365"/>
- <char name="oslash" width="611"/>
- <char name="otilde" width="556"/>
- <char name="p" width="556"/>
- <char name="paragraph" width="537"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="889"/>
- <char name="period" width="278"/>
- <char name="periodcentered" width="278"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="584"/>
- <char name="plusminus" width="584"/>
- <char name="q" width="556"/>
- <char name="question" width="556"/>
- <char name="questiondown" width="611"/>
- <char name="quotedbl" width="355"/>
- <char name="quotedblbase" width="333"/>
- <char name="quotedblleft" width="333"/>
- <char name="quotedblright" width="333"/>
- <char name="quoteleft" width="222"/>
- <char name="quoteright" width="222"/>
- <char name="quotesinglbase" width="222"/>
- <char name="quotesingle" width="191"/>
- <char name="r" width="333"/>
- <char name="registered" width="737"/>
- <char name="ring" width="333"/>
- <char name="s" width="500"/>
- <char name="scaron" width="500"/>
- <char name="section" width="556"/>
- <char name="semicolon" width="278"/>
- <char name="seven" width="556"/>
- <char name="six" width="556"/>
- <char name="slash" width="278"/>
- <char name="space" width="278"/>
+ <char-metrics>
+ <char name="A" width="667" llx="14" lly="0" urx="654" ury="718"/>
+ <char name="AE" width="1000" llx="8" lly="0" urx="1097" ury="718"/>
+ <char name="Aacute" width="667" llx="14" lly="0" urx="683" ury="929"/>
+ <char name="Acircumflex" width="667" llx="14" lly="0" urx="654" ury="929"/>
+ <char name="Adieresis" width="667" llx="14" lly="0" urx="654" ury="901"/>
+ <char name="Agrave" width="667" llx="14" lly="0" urx="654" ury="929"/>
+ <char name="Aring" width="667" llx="14" lly="0" urx="654" ury="931"/>
+ <char name="Atilde" width="667" llx="14" lly="0" urx="699" ury="917"/>
+ <char name="B" width="667" llx="74" lly="0" urx="712" ury="718"/>
+ <char name="C" width="722" llx="108" lly="-19" urx="782" ury="737"/>
+ <char name="Ccedilla" width="722" llx="108" lly="-225" urx="782" ury="737"/>
+ <char name="D" width="722" llx="81" lly="0" urx="764" ury="718"/>
+ <char name="E" width="667" llx="86" lly="0" urx="762" ury="718"/>
+ <char name="Eacute" width="667" llx="86" lly="0" urx="762" ury="929"/>
+ <char name="Ecircumflex" width="667" llx="86" lly="0" urx="762" ury="929"/>
+ <char name="Edieresis" width="667" llx="86" lly="0" urx="762" ury="901"/>
+ <char name="Egrave" width="667" llx="86" lly="0" urx="762" ury="929"/>
+ <char name="Eth" width="722" llx="69" lly="0" urx="764" ury="718"/>
+ <char name="Euro" width="556" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="611" llx="86" lly="0" urx="736" ury="718"/>
+ <char name="G" width="778" llx="111" lly="-19" urx="799" ury="737"/>
+ <char name="H" width="722" llx="77" lly="0" urx="799" ury="718"/>
+ <char name="I" width="278" llx="91" lly="0" urx="341" ury="718"/>
+ <char name="Iacute" width="278" llx="91" lly="0" urx="489" ury="929"/>
+ <char name="Icircumflex" width="278" llx="91" lly="0" urx="452" ury="929"/>
+ <char name="Idieresis" width="278" llx="91" lly="0" urx="458" ury="901"/>
+ <char name="Igrave" width="278" llx="91" lly="0" urx="351" ury="929"/>
+ <char name="J" width="500" llx="47" lly="-19" urx="581" ury="718"/>
+ <char name="K" width="667" llx="76" lly="0" urx="808" ury="718"/>
+ <char name="L" width="556" llx="76" lly="0" urx="555" ury="718"/>
+ <char name="Lslash" width="556" llx="41" lly="0" urx="555" ury="718"/>
+ <char name="M" width="833" llx="73" lly="0" urx="914" ury="718"/>
+ <char name="N" width="722" llx="76" lly="0" urx="799" ury="718"/>
+ <char name="Ntilde" width="722" llx="76" lly="0" urx="799" ury="917"/>
+ <char name="O" width="778" llx="105" lly="-19" urx="826" ury="737"/>
+ <char name="OE" width="1000" llx="98" lly="-19" urx="1116" ury="737"/>
+ <char name="Oacute" width="778" llx="105" lly="-19" urx="826" ury="929"/>
+ <char name="Ocircumflex" width="778" llx="105" lly="-19" urx="826" ury="929"/>
+ <char name="Odieresis" width="778" llx="105" lly="-19" urx="826" ury="901"/>
+ <char name="Ograve" width="778" llx="105" lly="-19" urx="826" ury="929"/>
+ <char name="Oslash" width="778" llx="43" lly="-19" urx="890" ury="737"/>
+ <char name="Otilde" width="778" llx="105" lly="-19" urx="826" ury="917"/>
+ <char name="P" width="667" llx="86" lly="0" urx="737" ury="718"/>
+ <char name="Q" width="778" llx="105" lly="-56" urx="826" ury="737"/>
+ <char name="R" width="722" llx="88" lly="0" urx="773" ury="718"/>
+ <char name="S" width="667" llx="90" lly="-19" urx="713" ury="737"/>
+ <char name="Scaron" width="667" llx="90" lly="-19" urx="713" ury="929"/>
+ <char name="T" width="611" llx="148" lly="0" urx="750" ury="718"/>
+ <char name="Thorn" width="667" llx="86" lly="0" urx="712" ury="718"/>
+ <char name="U" width="722" llx="123" lly="-19" urx="797" ury="718"/>
+ <char name="Uacute" width="722" llx="123" lly="-19" urx="797" ury="929"/>
+ <char name="Ucircumflex" width="722" llx="123" lly="-19" urx="797" ury="929"/>
+ <char name="Udieresis" width="722" llx="123" lly="-19" urx="797" ury="901"/>
+ <char name="Ugrave" width="722" llx="123" lly="-19" urx="797" ury="929"/>
+ <char name="V" width="667" llx="173" lly="0" urx="800" ury="718"/>
+ <char name="W" width="944" llx="169" lly="0" urx="1081" ury="718"/>
+ <char name="X" width="667" llx="19" lly="0" urx="790" ury="718"/>
+ <char name="Y" width="667" llx="167" lly="0" urx="806" ury="718"/>
+ <char name="Yacute" width="667" llx="167" lly="0" urx="806" ury="929"/>
+ <char name="Ydieresis" width="667" llx="167" lly="0" urx="806" ury="901"/>
+ <char name="Z" width="611" llx="23" lly="0" urx="741" ury="718"/>
+ <char name="Zcaron" width="611" llx="23" lly="0" urx="741" ury="929"/>
+ <char name="a" width="556" llx="61" lly="-15" urx="559" ury="538"/>
+ <char name="aacute" width="556" llx="61" lly="-15" urx="587" ury="734"/>
+ <char name="acircumflex" width="556" llx="61" lly="-15" urx="559" ury="734"/>
+ <char name="acute" width="333" llx="248" lly="593" urx="475" ury="734"/>
+ <char name="adieresis" width="556" llx="61" lly="-15" urx="559" ury="706"/>
+ <char name="ae" width="889" llx="61" lly="-15" urx="909" ury="538"/>
+ <char name="agrave" width="556" llx="61" lly="-15" urx="559" ury="734"/>
+ <char name="ampersand" width="667" llx="77" lly="-15" urx="647" ury="718"/>
+ <char name="aring" width="556" llx="61" lly="-15" urx="559" ury="756"/>
+ <char name="asciicircum" width="469" llx="42" lly="264" urx="539" ury="688"/>
+ <char name="asciitilde" width="584" llx="111" lly="180" urx="580" ury="326"/>
+ <char name="asterisk" width="389" llx="165" lly="431" urx="475" ury="718"/>
+ <char name="at" width="1015" llx="215" lly="-19" urx="965" ury="737"/>
+ <char name="atilde" width="556" llx="61" lly="-15" urx="592" ury="722"/>
+ <char name="b" width="556" llx="58" lly="-15" urx="584" ury="718"/>
+ <char name="backslash" width="278" llx="140" lly="-19" urx="291" ury="737"/>
+ <char name="bar" width="260" llx="46" lly="-225" urx="332" ury="775"/>
+ <char name="braceleft" width="334" llx="92" lly="-196" urx="445" ury="722"/>
+ <char name="braceright" width="334" llx="0" lly="-196" urx="354" ury="722"/>
+ <char name="bracketleft" width="278" llx="21" lly="-196" urx="403" ury="722"/>
+ <char name="bracketright" width="278" llx="-14" lly="-196" urx="368" ury="722"/>
+ <char name="breve" width="333" llx="167" lly="595" urx="476" ury="731"/>
+ <char name="brokenbar" width="260" llx="62" lly="-150" urx="316" ury="700"/>
+ <char name="bullet" width="350" llx="91" lly="202" urx="413" ury="517"/>
+ <char name="c" width="500" llx="74" lly="-15" urx="553" ury="538"/>
+ <char name="caron" width="333" llx="177" lly="593" urx="468" ury="734"/>
+ <char name="ccedilla" width="500" llx="74" lly="-225" urx="553" ury="538"/>
+ <char name="cedilla" width="333" llx="2" lly="-225" urx="232" ury="0"/>
+ <char name="cent" width="556" llx="95" lly="-115" urx="584" ury="623"/>
+ <char name="circumflex" width="333" llx="147" lly="593" urx="438" ury="734"/>
+ <char name="colon" width="278" llx="87" lly="0" urx="301" ury="516"/>
+ <char name="comma" width="278" llx="56" lly="-147" urx="214" ury="106"/>
+ <char name="copyright" width="737" llx="54" lly="-19" urx="837" ury="737"/>
+ <char name="currency" width="556" llx="60" lly="99" urx="646" ury="603"/>
+ <char name="d" width="556" llx="84" lly="-15" urx="652" ury="718"/>
+ <char name="dagger" width="556" llx="135" lly="-159" urx="622" ury="718"/>
+ <char name="daggerdbl" width="556" llx="52" lly="-159" urx="623" ury="718"/>
+ <char name="degree" width="400" llx="169" lly="411" urx="468" ury="703"/>
+ <char name="dieresis" width="333" llx="168" lly="604" urx="443" ury="706"/>
+ <char name="divide" width="584" llx="85" lly="-19" urx="606" ury="524"/>
+ <char name="dollar" width="556" llx="69" lly="-115" urx="617" ury="775"/>
+ <char name="dotaccent" width="333" llx="249" lly="604" urx="362" ury="706"/>
+ <char name="dotlessi" width="278" llx="95" lly="0" urx="294" ury="523"/>
+ <char name="e" width="556" llx="84" lly="-15" urx="578" ury="538"/>
+ <char name="eacute" width="556" llx="84" lly="-15" urx="587" ury="734"/>
+ <char name="ecircumflex" width="556" llx="84" lly="-15" urx="578" ury="734"/>
+ <char name="edieresis" width="556" llx="84" lly="-15" urx="578" ury="706"/>
+ <char name="egrave" width="556" llx="84" lly="-15" urx="578" ury="734"/>
+ <char name="eight" width="556" llx="74" lly="-19" urx="607" ury="703"/>
+ <char name="ellipsis" width="1000" llx="115" lly="0" urx="908" ury="106"/>
+ <char name="emdash" width="1000" llx="51" lly="240" urx="1067" ury="313"/>
+ <char name="endash" width="556" llx="51" lly="240" urx="623" ury="313"/>
+ <char name="equal" width="584" llx="63" lly="115" urx="628" ury="390"/>
+ <char name="eth" width="556" llx="81" lly="-15" urx="617" ury="737"/>
+ <char name="exclam" width="278" llx="90" lly="0" urx="340" ury="718"/>
+ <char name="exclamdown" width="333" llx="77" lly="-195" urx="326" ury="523"/>
+ <char name="f" width="278" llx="86" lly="0" urx="416" ury="728"/>
+ <char name="fi" width="500" llx="86" lly="0" urx="587" ury="728"/>
+ <char name="five" width="556" llx="68" lly="-19" urx="621" ury="688"/>
+ <char name="fl" width="500" llx="86" lly="0" urx="585" ury="728"/>
+ <char name="florin" width="556" llx="-52" lly="-207" urx="654" ury="737"/>
+ <char name="four" width="556" llx="61" lly="0" urx="576" ury="703"/>
+ <char name="fraction" width="167" llx="-170" lly="-19" urx="482" ury="703"/>
+ <char name="g" width="556" llx="42" lly="-220" urx="610" ury="538"/>
+ <char name="germandbls" width="611" llx="67" lly="-15" urx="658" ury="728"/>
+ <char name="grave" width="333" llx="170" lly="593" urx="337" ury="734"/>
+ <char name="greater" width="584" llx="50" lly="11" urx="597" ury="495"/>
+ <char name="guillemotleft" width="556" llx="146" lly="108" urx="554" ury="446"/>
+ <char name="guillemotright" width="556" llx="120" lly="108" urx="528" ury="446"/>
+ <char name="guilsinglleft" width="333" llx="137" lly="108" urx="340" ury="446"/>
+ <char name="guilsinglright" width="333" llx="111" lly="108" urx="314" ury="446"/>
+ <char name="h" width="556" llx="65" lly="0" urx="573" ury="718"/>
+ <char name="hungarumlaut" width="333" llx="157" lly="593" urx="565" ury="734"/>
+ <char name="hyphen" width="333" llx="93" lly="232" urx="357" ury="322"/>
+ <char name="i" width="222" llx="67" lly="0" urx="308" ury="718"/>
+ <char name="iacute" width="278" llx="95" lly="0" urx="448" ury="734"/>
+ <char name="icircumflex" width="278" llx="95" lly="0" urx="411" ury="734"/>
+ <char name="idieresis" width="278" llx="95" lly="0" urx="416" ury="706"/>
+ <char name="igrave" width="278" llx="95" lly="0" urx="310" ury="734"/>
+ <char name="j" width="222" llx="-60" lly="-210" urx="308" ury="718"/>
+ <char name="k" width="500" llx="67" lly="0" urx="600" ury="718"/>
+ <char name="l" width="222" llx="67" lly="0" urx="308" ury="718"/>
+ <char name="less" width="584" llx="94" lly="11" urx="641" ury="495"/>
+ <char name="logicalnot" width="584" llx="106" lly="108" urx="628" ury="390"/>
+ <char name="lslash" width="222" llx="41" lly="0" urx="347" ury="718"/>
+ <char name="m" width="833" llx="65" lly="0" urx="852" ury="538"/>
+ <char name="macron" width="333" llx="143" lly="627" urx="468" ury="684"/>
+ <char name="minus" width="584" llx="85" lly="216" urx="606" ury="289"/>
+ <char name="mu" width="556" llx="24" lly="-207" urx="600" ury="523"/>
+ <char name="multiply" width="584" llx="50" lly="0" urx="642" ury="506"/>
+ <char name="n" width="556" llx="65" lly="0" urx="573" ury="538"/>
+ <char name="nine" width="556" llx="82" lly="-19" urx="609" ury="703"/>
+ <char name="ntilde" width="556" llx="65" lly="0" urx="592" ury="722"/>
+ <char name="numbersign" width="556" llx="73" lly="0" urx="631" ury="688"/>
+ <char name="o" width="556" llx="83" lly="-14" urx="585" ury="538"/>
+ <char name="oacute" width="556" llx="83" lly="-14" urx="587" ury="734"/>
+ <char name="ocircumflex" width="556" llx="83" lly="-14" urx="585" ury="734"/>
+ <char name="odieresis" width="556" llx="83" lly="-14" urx="585" ury="706"/>
+ <char name="oe" width="944" llx="83" lly="-15" urx="964" ury="538"/>
+ <char name="ogonek" width="333" llx="43" lly="-225" urx="249" ury="0"/>
+ <char name="ograve" width="556" llx="83" lly="-14" urx="585" ury="734"/>
+ <char name="one" width="556" llx="207" lly="0" urx="508" ury="703"/>
+ <char name="onehalf" width="834" llx="114" lly="-19" urx="839" ury="703"/>
+ <char name="onequarter" width="834" llx="150" lly="-19" urx="802" ury="703"/>
+ <char name="onesuperior" width="333" llx="166" lly="281" urx="371" ury="703"/>
+ <char name="ordfeminine" width="370" llx="127" lly="405" urx="449" ury="737"/>
+ <char name="ordmasculine" width="365" llx="141" lly="405" urx="468" ury="737"/>
+ <char name="oslash" width="611" llx="29" lly="-22" urx="647" ury="545"/>
+ <char name="otilde" width="556" llx="83" lly="-14" urx="602" ury="722"/>
+ <char name="p" width="556" llx="14" lly="-207" urx="584" ury="538"/>
+ <char name="paragraph" width="537" llx="126" lly="-173" urx="650" ury="718"/>
+ <char name="parenleft" width="333" llx="108" lly="-207" urx="454" ury="733"/>
+ <char name="parenright" width="333" llx="-9" lly="-207" urx="337" ury="733"/>
+ <char name="percent" width="889" llx="147" lly="-19" urx="889" ury="703"/>
+ <char name="period" width="278" llx="87" lly="0" urx="214" ury="106"/>
+ <char name="periodcentered" width="278" llx="129" lly="190" urx="257" ury="315"/>
+ <char name="perthousand" width="1000" llx="88" lly="-19" urx="1029" ury="703"/>
+ <char name="plus" width="584" llx="85" lly="0" urx="606" ury="505"/>
+ <char name="plusminus" width="584" llx="39" lly="0" urx="618" ury="506"/>
+ <char name="q" width="556" llx="84" lly="-207" urx="605" ury="538"/>
+ <char name="question" width="556" llx="161" lly="0" urx="610" ury="727"/>
+ <char name="questiondown" width="611" llx="85" lly="-201" urx="534" ury="525"/>
+ <char name="quotedbl" width="355" llx="168" lly="463" urx="438" ury="718"/>
+ <char name="quotedblbase" width="333" llx="-6" lly="-149" urx="318" ury="106"/>
+ <char name="quotedblleft" width="333" llx="138" lly="470" urx="461" ury="725"/>
+ <char name="quotedblright" width="333" llx="124" lly="463" urx="448" ury="718"/>
+ <char name="quoteleft" width="222" llx="165" lly="470" urx="323" ury="725"/>
+ <char name="quoteright" width="222" llx="151" lly="463" urx="310" ury="718"/>
+ <char name="quotesinglbase" width="222" llx="21" lly="-149" urx="180" ury="106"/>
+ <char name="quotesingle" width="191" llx="157" lly="463" urx="285" ury="718"/>
+ <char name="r" width="333" llx="77" lly="0" urx="446" ury="538"/>
+ <char name="registered" width="737" llx="54" lly="-19" urx="837" ury="737"/>
+ <char name="ring" width="333" llx="214" lly="572" urx="402" ury="756"/>
+ <char name="s" width="500" llx="63" lly="-15" urx="529" ury="538"/>
+ <char name="scaron" width="500" llx="63" lly="-15" urx="552" ury="734"/>
+ <char name="section" width="556" llx="76" lly="-191" urx="584" ury="737"/>
+ <char name="semicolon" width="278" llx="56" lly="-147" urx="301" ury="516"/>
+ <char name="seven" width="556" llx="137" lly="0" urx="669" ury="688"/>
+ <char name="six" width="556" llx="91" lly="-19" urx="615" ury="703"/>
+ <char name="slash" width="278" llx="-21" lly="-19" urx="452" ury="737"/>
+ <char name="space" width="278" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="278"/>
- <char name="sterling" width="556"/>
- <char name="t" width="278"/>
- <char name="thorn" width="556"/>
- <char name="three" width="556"/>
- <char name="threequarters" width="834"/>
- <char name="threesuperior" width="333"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="1000"/>
- <char name="two" width="556"/>
- <char name="twosuperior" width="333"/>
- <char name="u" width="556"/>
- <char name="uacute" width="556"/>
- <char name="ucircumflex" width="556"/>
- <char name="udieresis" width="556"/>
- <char name="ugrave" width="556"/>
- <char name="underscore" width="556"/>
- <char name="v" width="500"/>
- <char name="w" width="722"/>
- <char name="x" width="500"/>
- <char name="y" width="500"/>
- <char name="yacute" width="500"/>
- <char name="ydieresis" width="500"/>
- <char name="yen" width="556"/>
- <char name="z" width="500"/>
- <char name="zcaron" width="500"/>
- <char name="zero" width="556"/>
- </widths>
+ <char name="sterling" width="556" llx="49" lly="-16" urx="634" ury="718"/>
+ <char name="t" width="278" llx="102" lly="-7" urx="368" ury="669"/>
+ <char name="thorn" width="556" llx="14" lly="-207" urx="584" ury="718"/>
+ <char name="three" width="556" llx="75" lly="-19" urx="610" ury="703"/>
+ <char name="threequarters" width="834" llx="130" lly="-19" urx="861" ury="703"/>
+ <char name="threesuperior" width="333" llx="90" lly="270" urx="436" ury="703"/>
+ <char name="tilde" width="333" llx="125" lly="606" urx="490" ury="722"/>
+ <char name="trademark" width="1000" llx="186" lly="306" urx="1056" ury="718"/>
+ <char name="two" width="556" llx="26" lly="0" urx="617" ury="703"/>
+ <char name="twosuperior" width="333" llx="64" lly="281" urx="449" ury="703"/>
+ <char name="u" width="556" llx="94" lly="-15" urx="600" ury="523"/>
+ <char name="uacute" width="556" llx="94" lly="-15" urx="600" ury="734"/>
+ <char name="ucircumflex" width="556" llx="94" lly="-15" urx="600" ury="734"/>
+ <char name="udieresis" width="556" llx="94" lly="-15" urx="600" ury="706"/>
+ <char name="ugrave" width="556" llx="94" lly="-15" urx="600" ury="734"/>
+ <char name="underscore" width="556" llx="-27" lly="-125" urx="540" ury="-75"/>
+ <char name="v" width="500" llx="119" lly="0" urx="603" ury="523"/>
+ <char name="w" width="722" llx="125" lly="0" urx="820" ury="523"/>
+ <char name="x" width="500" llx="11" lly="0" urx="594" ury="523"/>
+ <char name="y" width="500" llx="15" lly="-214" urx="600" ury="523"/>
+ <char name="yacute" width="500" llx="15" lly="-214" urx="600" ury="734"/>
+ <char name="ydieresis" width="500" llx="15" lly="-214" urx="600" ury="706"/>
+ <char name="yen" width="556" llx="81" lly="0" urx="699" ury="688"/>
+ <char name="z" width="500" llx="31" lly="0" urx="571" ury="523"/>
+ <char name="zcaron" width="500" llx="31" lly="0" urx="571" ury="734"/>
+ <char name="zero" width="556" llx="93" lly="-19" urx="608" ury="703"/>
+ </char-metrics>
<kerning kpx1="107">
<pair kern="-20" kpx2="111"/>
<pair kern="-20" kpx2="101"/>
@@ -602,4 +603,4 @@
<pair kern="-30" kpx2="118"/>
<pair kern="-15" kpx2="44"/>
</kerning>
-</font-metrics> \ No newline at end of file
+</font-metrics>
diff --git a/src/codegen/fonts/Symbol.xml b/src/codegen/fonts/Symbol.xml
index c0dce043a..7493b7e7f 100644
--- a/src/codegen/fonts/Symbol.xml
+++ b/src/codegen/fonts/Symbol.xml
@@ -21,203 +21,204 @@
<family-name>Symbol</family-name>
<class-name>Symbol</class-name>
<encoding>SymbolEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>1010</cap-height>
<x-height>520</x-height>
<ascender>1010</ascender>
<descender>-293</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="space" width="250"/>
- <char name="exclam" width="333"/>
- <char name="universal" width="713"/>
- <char name="numbersign" width="500"/>
- <char name="existential" width="549"/>
- <char name="percent" width="833"/>
- <char name="ampersand" width="778"/>
- <char name="suchthat" width="439"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="asteriskmath" width="500"/>
- <char name="plus" width="549"/>
- <char name="comma" width="250"/>
- <char name="minus" width="549"/>
- <char name="period" width="250"/>
- <char name="slash" width="278"/>
- <char name="zero" width="500"/>
- <char name="one" width="500"/>
- <char name="two" width="500"/>
- <char name="three" width="500"/>
- <char name="four" width="500"/>
- <char name="five" width="500"/>
- <char name="six" width="500"/>
- <char name="seven" width="500"/>
- <char name="eight" width="500"/>
- <char name="nine" width="500"/>
- <char name="colon" width="278"/>
- <char name="semicolon" width="278"/>
- <char name="less" width="549"/>
- <char name="equal" width="549"/>
- <char name="greater" width="549"/>
- <char name="question" width="444"/>
- <char name="congruent" width="549"/>
- <char name="Alpha" width="722"/>
- <char name="Beta" width="667"/>
- <char name="Chi" width="722"/>
- <char name="Delta" width="612"/>
- <char name="Epsilon" width="611"/>
- <char name="Phi" width="763"/>
- <char name="Gamma" width="603"/>
- <char name="Eta" width="722"/>
- <char name="Iota" width="333"/>
- <char name="theta1" width="631"/>
- <char name="Kappa" width="722"/>
- <char name="Lambda" width="686"/>
- <char name="Mu" width="889"/>
- <char name="Nu" width="722"/>
- <char name="Omicron" width="722"/>
- <char name="Pi" width="768"/>
- <char name="Theta" width="741"/>
- <char name="Rho" width="556"/>
- <char name="Sigma" width="592"/>
- <char name="Tau" width="611"/>
- <char name="Upsilon" width="690"/>
- <char name="sigma1" width="439"/>
- <char name="Omega" width="768"/>
- <char name="Xi" width="645"/>
- <char name="Psi" width="795"/>
- <char name="Zeta" width="611"/>
- <char name="bracketleft" width="333"/>
- <char name="therefore" width="863"/>
- <char name="bracketright" width="333"/>
- <char name="perpendicular" width="658"/>
- <char name="underscore" width="500"/>
- <char name="radicalex" width="500"/>
- <char name="alpha" width="631"/>
- <char name="beta" width="549"/>
- <char name="chi" width="549"/>
- <char name="delta" width="494"/>
- <char name="epsilon" width="439"/>
- <char name="phi" width="521"/>
- <char name="gamma" width="411"/>
- <char name="eta" width="603"/>
- <char name="iota" width="329"/>
- <char name="phi1" width="603"/>
- <char name="kappa" width="549"/>
- <char name="lambda" width="549"/>
- <char name="m" width="576"/>
- <char name="mu" width="576"/>
- <char name="nu" width="521"/>
- <char name="omicron" width="549"/>
- <char name="pi" width="549"/>
- <char name="theta" width="521"/>
- <char name="rho" width="549"/>
- <char name="sigma" width="603"/>
- <char name="tau" width="439"/>
- <char name="upsilon" width="576"/>
- <char name="omega1" width="713"/>
- <char name="omega" width="686"/>
- <char name="xi" width="493"/>
- <char name="psi" width="686"/>
- <char name="zeta" width="494"/>
- <char name="braceleft" width="480"/>
- <char name="bar" width="200"/>
- <char name="braceright" width="480"/>
- <char name="similar" width="549"/>
- <char name="Euro" width="750"/>
- <char name="Upsilon1" width="620"/>
- <char name="minute" width="247"/>
- <char name="lessequal" width="549"/>
- <char name="fraction" width="167"/>
- <char name="infinity" width="713"/>
- <char name="florin" width="500"/>
- <char name="club" width="753"/>
- <char name="diamond" width="753"/>
- <char name="heart" width="753"/>
- <char name="spade" width="753"/>
- <char name="arrowboth" width="1042"/>
- <char name="arrowleft" width="987"/>
- <char name="arrowup" width="603"/>
- <char name="arrowright" width="987"/>
- <char name="arrowdown" width="603"/>
- <char name="degree" width="400"/>
- <char name="plusminus" width="549"/>
- <char name="second" width="411"/>
- <char name="greaterequal" width="549"/>
- <char name="multiply" width="549"/>
- <char name="proportional" width="713"/>
- <char name="partialdiff" width="494"/>
- <char name="bullet" width="460"/>
- <char name="divide" width="549"/>
- <char name="notequal" width="549"/>
- <char name="equivalence" width="549"/>
- <char name="approxequal" width="549"/>
- <char name="ellipsis" width="1000"/>
- <char name="arrowvertex" width="603"/>
- <char name="arrowhorizex" width="1000"/>
- <char name="carriagereturn" width="658"/>
- <char name="aleph" width="823"/>
- <char name="Ifraktur" width="686"/>
- <char name="Rfraktur" width="795"/>
- <char name="weierstrass" width="987"/>
- <char name="circlemultiply" width="768"/>
- <char name="circleplus" width="768"/>
- <char name="emptyset" width="823"/>
- <char name="intersection" width="768"/>
- <char name="union" width="768"/>
- <char name="propersuperset" width="713"/>
- <char name="reflexsuperset" width="713"/>
- <char name="notsubset" width="713"/>
- <char name="propersubset" width="713"/>
- <char name="reflexsubset" width="713"/>
- <char name="element" width="713"/>
- <char name="notelement" width="713"/>
- <char name="angle" width="768"/>
- <char name="gradient" width="713"/>
- <char name="registerserif" width="790"/>
- <char name="copyrightserif" width="790"/>
- <char name="trademarkserif" width="890"/>
- <char name="product" width="823"/>
- <char name="radical" width="549"/>
- <char name="dotmath" width="250"/>
- <char name="logicalnot" width="713"/>
- <char name="logicaland" width="603"/>
- <char name="logicalor" width="603"/>
- <char name="arrowdblboth" width="1042"/>
- <char name="arrowdblleft" width="987"/>
- <char name="arrowdblup" width="603"/>
- <char name="arrowdblright" width="987"/>
- <char name="arrowdbldown" width="603"/>
- <char name="lozenge" width="494"/>
- <char name="angleleft" width="329"/>
- <char name="registersans" width="790"/>
- <char name="copyrightsans" width="790"/>
- <char name="trademarksans" width="786"/>
- <char name="summation" width="713"/>
- <char name="parenlefttp" width="384"/>
- <char name="parenleftex" width="384"/>
- <char name="parenleftbt" width="384"/>
- <char name="bracketlefttp" width="384"/>
- <char name="bracketleftex" width="384"/>
- <char name="bracketleftbt" width="384"/>
- <char name="bracelefttp" width="494"/>
- <char name="braceleftmid" width="494"/>
- <char name="braceleftbt" width="494"/>
- <char name="braceex" width="494"/>
- <char name="angleright" width="329"/>
- <char name="integral" width="274"/>
- <char name="integraltp" width="686"/>
- <char name="integralex" width="686"/>
- <char name="integralbt" width="686"/>
- <char name="parenrighttp" width="384"/>
- <char name="parenrightex" width="384"/>
- <char name="parenrightbt" width="384"/>
- <char name="bracketrighttp" width="384"/>
- <char name="bracketrightex" width="384"/>
- <char name="bracketrightbt" width="384"/>
- <char name="bracerighttp" width="494"/>
- <char name="bracerightmid" width="494"/>
- <char name="bracerightbt" width="494"/>
- <char name="apple" width="790"/>
- </widths>
+ <char-metrics>
+ <char name="space" width="250" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="exclam" width="333" llx="128" lly="-17" urx="240" ury="672"/>
+ <char name="universal" width="713" llx="31" lly="0" urx="681" ury="705"/>
+ <char name="numbersign" width="500" llx="20" lly="-16" urx="481" ury="673"/>
+ <char name="existential" width="549" llx="25" lly="0" urx="478" ury="707"/>
+ <char name="percent" width="833" llx="63" lly="-36" urx="771" ury="655"/>
+ <char name="ampersand" width="778" llx="41" lly="-18" urx="750" ury="661"/>
+ <char name="suchthat" width="439" llx="48" lly="-17" urx="414" ury="500"/>
+ <char name="parenleft" width="333" llx="53" lly="-191" urx="300" ury="673"/>
+ <char name="parenright" width="333" llx="30" lly="-191" urx="277" ury="673"/>
+ <char name="asteriskmath" width="500" llx="65" lly="134" urx="427" ury="551"/>
+ <char name="plus" width="549" llx="10" lly="0" urx="539" ury="533"/>
+ <char name="comma" width="250" llx="56" lly="-152" urx="194" ury="104"/>
+ <char name="minus" width="549" llx="11" lly="233" urx="535" ury="288"/>
+ <char name="period" width="250" llx="69" lly="-17" urx="181" ury="95"/>
+ <char name="slash" width="278" llx="0" lly="-18" urx="254" ury="646"/>
+ <char name="zero" width="500" llx="24" lly="-14" urx="476" ury="685"/>
+ <char name="one" width="500" llx="117" lly="0" urx="390" ury="673"/>
+ <char name="two" width="500" llx="25" lly="0" urx="475" ury="685"/>
+ <char name="three" width="500" llx="43" lly="-14" urx="435" ury="685"/>
+ <char name="four" width="500" llx="15" lly="0" urx="469" ury="685"/>
+ <char name="five" width="500" llx="32" lly="-14" urx="445" ury="690"/>
+ <char name="six" width="500" llx="34" lly="-14" urx="468" ury="685"/>
+ <char name="seven" width="500" llx="24" lly="-16" urx="448" ury="673"/>
+ <char name="eight" width="500" llx="56" lly="-14" urx="445" ury="685"/>
+ <char name="nine" width="500" llx="30" lly="-18" urx="459" ury="685"/>
+ <char name="colon" width="278" llx="81" lly="-17" urx="193" ury="460"/>
+ <char name="semicolon" width="278" llx="83" lly="-152" urx="221" ury="460"/>
+ <char name="less" width="549" llx="26" lly="0" urx="523" ury="522"/>
+ <char name="equal" width="549" llx="11" lly="141" urx="537" ury="390"/>
+ <char name="greater" width="549" llx="26" lly="0" urx="523" ury="522"/>
+ <char name="question" width="444" llx="70" lly="-17" urx="412" ury="686"/>
+ <char name="congruent" width="549" llx="11" lly="0" urx="537" ury="475"/>
+ <char name="Alpha" width="722" llx="4" lly="0" urx="684" ury="673"/>
+ <char name="Beta" width="667" llx="29" lly="0" urx="592" ury="673"/>
+ <char name="Chi" width="722" llx="-9" lly="0" urx="704" ury="673"/>
+ <char name="Delta" width="612" llx="6" lly="0" urx="608" ury="688"/>
+ <char name="Epsilon" width="611" llx="32" lly="0" urx="617" ury="673"/>
+ <char name="Phi" width="763" llx="26" lly="0" urx="741" ury="673"/>
+ <char name="Gamma" width="603" llx="24" lly="0" urx="609" ury="673"/>
+ <char name="Eta" width="722" llx="39" lly="0" urx="729" ury="673"/>
+ <char name="Iota" width="333" llx="32" lly="0" urx="316" ury="673"/>
+ <char name="theta1" width="631" llx="18" lly="-18" urx="623" ury="689"/>
+ <char name="Kappa" width="722" llx="35" lly="0" urx="722" ury="673"/>
+ <char name="Lambda" width="686" llx="6" lly="0" urx="680" ury="688"/>
+ <char name="Mu" width="889" llx="28" lly="0" urx="887" ury="673"/>
+ <char name="Nu" width="722" llx="29" lly="-8" urx="720" ury="673"/>
+ <char name="Omicron" width="722" llx="41" lly="-17" urx="715" ury="685"/>
+ <char name="Pi" width="768" llx="25" lly="0" urx="745" ury="673"/>
+ <char name="Theta" width="741" llx="41" lly="-17" urx="715" ury="685"/>
+ <char name="Rho" width="556" llx="28" lly="0" urx="563" ury="673"/>
+ <char name="Sigma" width="592" llx="5" lly="0" urx="589" ury="673"/>
+ <char name="Tau" width="611" llx="33" lly="0" urx="607" ury="673"/>
+ <char name="Upsilon" width="690" llx="-8" lly="0" urx="694" ury="673"/>
+ <char name="sigma1" width="439" llx="40" lly="-233" urx="436" ury="500"/>
+ <char name="Omega" width="768" llx="34" lly="0" urx="736" ury="688"/>
+ <char name="Xi" width="645" llx="40" lly="0" urx="599" ury="673"/>
+ <char name="Psi" width="795" llx="15" lly="0" urx="781" ury="684"/>
+ <char name="Zeta" width="611" llx="44" lly="0" urx="636" ury="673"/>
+ <char name="bracketleft" width="333" llx="86" lly="-155" urx="299" ury="674"/>
+ <char name="therefore" width="863" llx="163" lly="0" urx="701" ury="487"/>
+ <char name="bracketright" width="333" llx="33" lly="-155" urx="246" ury="674"/>
+ <char name="perpendicular" width="658" llx="15" lly="0" urx="652" ury="674"/>
+ <char name="underscore" width="500" llx="-2" lly="-125" urx="502" ury="-75"/>
+ <char name="radicalex" width="500" llx="480" lly="881" urx="1090" ury="917"/>
+ <char name="alpha" width="631" llx="41" lly="-18" urx="622" ury="500"/>
+ <char name="beta" width="549" llx="61" lly="-223" urx="515" ury="741"/>
+ <char name="chi" width="549" llx="12" lly="-231" urx="522" ury="499"/>
+ <char name="delta" width="494" llx="40" lly="-19" urx="481" ury="740"/>
+ <char name="epsilon" width="439" llx="22" lly="-19" urx="427" ury="502"/>
+ <char name="phi" width="521" llx="28" lly="-224" urx="492" ury="673"/>
+ <char name="gamma" width="411" llx="5" lly="-225" urx="484" ury="499"/>
+ <char name="eta" width="603" llx="0" lly="-202" urx="527" ury="514"/>
+ <char name="iota" width="329" llx="0" lly="-17" urx="301" ury="503"/>
+ <char name="phi1" width="603" llx="36" lly="-224" urx="587" ury="499"/>
+ <char name="kappa" width="549" llx="33" lly="0" urx="558" ury="501"/>
+ <char name="lambda" width="549" llx="24" lly="-17" urx="548" ury="739"/>
+ <char name="mu" width="576" llx="33" lly="-223" urx="567" ury="500"/>
+ <char name="nu" width="521" llx="-9" lly="-16" urx="475" ury="507"/>
+ <char name="omicron" width="549" llx="35" lly="-19" urx="501" ury="499"/>
+ <char name="pi" width="549" llx="10" lly="-19" urx="530" ury="487"/>
+ <char name="theta" width="521" llx="43" lly="-17" urx="485" ury="690"/>
+ <char name="rho" width="549" llx="50" lly="-230" urx="490" ury="499"/>
+ <char name="sigma" width="603" llx="30" lly="-21" urx="588" ury="500"/>
+ <char name="tau" width="439" llx="10" lly="-19" urx="418" ury="500"/>
+ <char name="upsilon" width="576" llx="7" lly="-18" urx="535" ury="507"/>
+ <char name="omega1" width="713" llx="12" lly="-18" urx="671" ury="583"/>
+ <char name="omega" width="686" llx="42" lly="-17" urx="684" ury="500"/>
+ <char name="xi" width="493" llx="27" lly="-224" urx="469" ury="766"/>
+ <char name="psi" width="686" llx="12" lly="-228" urx="701" ury="500"/>
+ <char name="zeta" width="494" llx="60" lly="-225" urx="467" ury="756"/>
+ <char name="braceleft" width="480" llx="58" lly="-183" urx="397" ury="673"/>
+ <char name="bar" width="200" llx="65" lly="-293" urx="135" ury="707"/>
+ <char name="braceright" width="480" llx="79" lly="-183" urx="418" ury="673"/>
+ <char name="similar" width="549" llx="17" lly="203" urx="529" ury="307"/>
+ <char name="Euro" width="750" llx="20" lly="-12" urx="714" ury="685"/>
+ <char name="Upsilon1" width="620" llx="-2" lly="0" urx="610" ury="685"/>
+ <char name="minute" width="247" llx="27" lly="459" urx="228" ury="735"/>
+ <char name="lessequal" width="549" llx="29" lly="0" urx="526" ury="639"/>
+ <char name="fraction" width="167" llx="-180" lly="-12" urx="340" ury="677"/>
+ <char name="infinity" width="713" llx="26" lly="124" urx="688" ury="404"/>
+ <char name="florin" width="500" llx="2" lly="-193" urx="494" ury="686"/>
+ <char name="club" width="753" llx="86" lly="-26" urx="660" ury="533"/>
+ <char name="diamond" width="753" llx="142" lly="-36" urx="600" ury="550"/>
+ <char name="heart" width="753" llx="117" lly="-33" urx="631" ury="532"/>
+ <char name="spade" width="753" llx="113" lly="-36" urx="629" ury="548"/>
+ <char name="arrowboth" width="1042" llx="24" lly="-15" urx="1024" ury="511"/>
+ <char name="arrowleft" width="987" llx="32" lly="-15" urx="942" ury="511"/>
+ <char name="arrowup" width="603" llx="45" lly="0" urx="571" ury="910"/>
+ <char name="arrowright" width="987" llx="49" lly="-15" urx="959" ury="511"/>
+ <char name="arrowdown" width="603" llx="45" lly="-22" urx="571" ury="888"/>
+ <char name="degree" width="400" llx="50" lly="385" urx="350" ury="685"/>
+ <char name="plusminus" width="549" llx="10" lly="0" urx="539" ury="645"/>
+ <char name="second" width="411" llx="20" lly="459" urx="413" ury="737"/>
+ <char name="greaterequal" width="549" llx="29" lly="0" urx="526" ury="639"/>
+ <char name="multiply" width="549" llx="17" lly="8" urx="533" ury="524"/>
+ <char name="proportional" width="713" llx="27" lly="123" urx="639" ury="404"/>
+ <char name="partialdiff" width="494" llx="26" lly="-20" urx="462" ury="746"/>
+ <char name="bullet" width="460" llx="50" lly="113" urx="410" ury="473"/>
+ <char name="divide" width="549" llx="10" lly="71" urx="536" ury="456"/>
+ <char name="notequal" width="549" llx="15" lly="-25" urx="540" ury="549"/>
+ <char name="equivalence" width="549" llx="14" lly="82" urx="538" ury="443"/>
+ <char name="approxequal" width="549" llx="14" lly="135" urx="527" ury="394"/>
+ <char name="ellipsis" width="1000" llx="111" lly="-17" urx="889" ury="95"/>
+ <char name="arrowvertex" width="603" llx="280" lly="-120" urx="336" ury="1010"/>
+ <char name="arrowhorizex" width="1000" llx="-60" lly="220" urx="1050" ury="276"/>
+ <char name="carriagereturn" width="658" llx="15" lly="-16" urx="602" ury="629"/>
+ <char name="aleph" width="823" llx="175" lly="-18" urx="661" ury="658"/>
+ <char name="Ifraktur" width="686" llx="10" lly="-53" urx="578" ury="740"/>
+ <char name="Rfraktur" width="795" llx="26" lly="-15" urx="759" ury="734"/>
+ <char name="weierstrass" width="987" llx="159" lly="-211" urx="870" ury="573"/>
+ <char name="circlemultiply" width="768" llx="43" lly="-17" urx="733" ury="673"/>
+ <char name="circleplus" width="768" llx="43" lly="-15" urx="733" ury="675"/>
+ <char name="emptyset" width="823" llx="39" lly="-24" urx="781" ury="719"/>
+ <char name="intersection" width="768" llx="40" lly="0" urx="732" ury="509"/>
+ <char name="union" width="768" llx="40" lly="-17" urx="732" ury="492"/>
+ <char name="propersuperset" width="713" llx="20" lly="0" urx="673" ury="470"/>
+ <char name="reflexsuperset" width="713" llx="20" lly="-125" urx="673" ury="470"/>
+ <char name="notsubset" width="713" llx="36" lly="-70" urx="690" ury="540"/>
+ <char name="propersubset" width="713" llx="37" lly="0" urx="690" ury="470"/>
+ <char name="reflexsubset" width="713" llx="37" lly="-125" urx="690" ury="470"/>
+ <char name="element" width="713" llx="45" lly="0" urx="505" ury="468"/>
+ <char name="notelement" width="713" llx="45" lly="-58" urx="505" ury="555"/>
+ <char name="angle" width="768" llx="26" lly="0" urx="738" ury="673"/>
+ <char name="gradient" width="713" llx="36" lly="-19" urx="681" ury="718"/>
+ <char name="registerserif" width="790" llx="50" lly="-17" urx="740" ury="673"/>
+ <char name="copyrightserif" width="790" llx="51" lly="-15" urx="741" ury="675"/>
+ <char name="trademarkserif" width="890" llx="18" lly="293" urx="855" ury="673"/>
+ <char name="product" width="823" llx="25" lly="-101" urx="803" ury="751"/>
+ <char name="radical" width="549" llx="10" lly="-38" urx="515" ury="917"/>
+ <char name="dotmath" width="250" llx="69" lly="210" urx="169" ury="310"/>
+ <char name="logicalnot" width="713" llx="15" lly="0" urx="680" ury="288"/>
+ <char name="logicaland" width="603" llx="23" lly="0" urx="583" ury="454"/>
+ <char name="logicalor" width="603" llx="30" lly="0" urx="578" ury="477"/>
+ <char name="arrowdblboth" width="1042" llx="27" lly="-20" urx="1023" ury="510"/>
+ <char name="arrowdblleft" width="987" llx="30" lly="-15" urx="939" ury="513"/>
+ <char name="arrowdblup" width="603" llx="39" lly="2" urx="567" ury="911"/>
+ <char name="arrowdblright" width="987" llx="45" lly="-20" urx="954" ury="508"/>
+ <char name="arrowdbldown" width="603" llx="44" lly="-19" urx="572" ury="890"/>
+ <char name="lozenge" width="494" llx="18" lly="0" urx="466" ury="745"/>
+ <char name="angleleft" width="329" llx="25" lly="-198" urx="306" ury="746"/>
+ <char name="registersans" width="790" llx="50" lly="-20" urx="740" ury="670"/>
+ <char name="copyrightsans" width="790" llx="49" lly="-15" urx="739" ury="675"/>
+ <char name="trademarksans" width="786" llx="5" lly="293" urx="725" ury="673"/>
+ <char name="summation" width="713" llx="14" lly="-108" urx="695" ury="752"/>
+ <char name="parenlefttp" width="384" llx="24" lly="-293" urx="436" ury="926"/>
+ <char name="parenleftex" width="384" llx="24" lly="-85" urx="108" ury="925"/>
+ <char name="parenleftbt" width="384" llx="24" lly="-293" urx="436" ury="926"/>
+ <char name="bracketlefttp" width="384" llx="0" lly="-80" urx="349" ury="926"/>
+ <char name="bracketleftex" width="384" llx="0" lly="-79" urx="77" ury="925"/>
+ <char name="bracketleftbt" width="384" llx="0" lly="-80" urx="349" ury="926"/>
+ <char name="bracelefttp" width="494" llx="209" lly="-85" urx="445" ury="925"/>
+ <char name="braceleftmid" width="494" llx="20" lly="-85" urx="284" ury="935"/>
+ <char name="braceleftbt" width="494" llx="209" lly="-75" urx="445" ury="935"/>
+ <char name="braceex" width="494" llx="209" lly="-85" urx="284" ury="935"/>
+ <char name="angleright" width="329" llx="21" lly="-198" urx="302" ury="746"/>
+ <char name="integral" width="274" llx="2" lly="-107" urx="291" ury="916"/>
+ <char name="integraltp" width="686" llx="308" lly="-88" urx="675" ury="920"/>
+ <char name="integralex" width="686" llx="308" lly="-88" urx="378" ury="975"/>
+ <char name="integralbt" width="686" llx="11" lly="-87" urx="378" ury="921"/>
+ <char name="parenrighttp" width="384" llx="54" lly="-293" urx="466" ury="926"/>
+ <char name="parenrightex" width="384" llx="382" lly="-85" urx="466" ury="925"/>
+ <char name="parenrightbt" width="384" llx="54" lly="-293" urx="466" ury="926"/>
+ <char name="bracketrighttp" width="384" llx="22" lly="-80" urx="371" ury="926"/>
+ <char name="bracketrightex" width="384" llx="294" lly="-79" urx="371" ury="925"/>
+ <char name="bracketrightbt" width="384" llx="22" lly="-80" urx="371" ury="926"/>
+ <char name="bracerighttp" width="494" llx="48" lly="-85" urx="284" ury="925"/>
+ <char name="bracerightmid" width="494" llx="209" lly="-85" urx="473" ury="935"/>
+ <char name="bracerightbt" width="494" llx="48" lly="-75" urx="284" ury="935"/>
+ <char name="apple" width="790" llx="56" lly="-3" urx="733" ury="808"/>
+ </char-metrics>
</font-metrics>
diff --git a/src/codegen/fonts/TimesBold.xml b/src/codegen/fonts/TimesBold.xml
index 20d50fdd8..ef5d6bcc0 100644
--- a/src/codegen/fonts/TimesBold.xml
+++ b/src/codegen/fonts/TimesBold.xml
@@ -22,245 +22,246 @@
<family-name>Times</family-name>
<class-name>TimesBold</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>676</cap-height>
<x-height>461</x-height>
<ascender>676</ascender>
<descender>-205</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="722"/>
- <char name="AE" width="1000"/>
- <char name="Aacute" width="722"/>
- <char name="Acircumflex" width="722"/>
- <char name="Adieresis" width="722"/>
- <char name="Agrave" width="722"/>
- <char name="Aring" width="722"/>
- <char name="Atilde" width="722"/>
- <char name="B" width="667"/>
- <char name="C" width="722"/>
- <char name="Ccedilla" width="722"/>
- <char name="D" width="722"/>
- <char name="E" width="667"/>
- <char name="Eacute" width="667"/>
- <char name="Ecircumflex" width="667"/>
- <char name="Edieresis" width="667"/>
- <char name="Egrave" width="667"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="500"/>
- <char name="F" width="611"/>
- <char name="G" width="778"/>
- <char name="H" width="778"/>
- <char name="I" width="389"/>
- <char name="Iacute" width="389"/>
- <char name="Icircumflex" width="389"/>
- <char name="Idieresis" width="389"/>
- <char name="Igrave" width="389"/>
- <char name="J" width="500"/>
- <char name="K" width="778"/>
- <char name="L" width="667"/>
- <char name="Lslash" width="667"/>
- <char name="M" width="944"/>
- <char name="N" width="722"/>
- <char name="Ntilde" width="722"/>
- <char name="O" width="778"/>
- <char name="OE" width="1000"/>
- <char name="Oacute" width="778"/>
- <char name="Ocircumflex" width="778"/>
- <char name="Odieresis" width="778"/>
- <char name="Ograve" width="778"/>
- <char name="Oslash" width="778"/>
- <char name="Otilde" width="778"/>
- <char name="P" width="611"/>
- <char name="Q" width="778"/>
- <char name="R" width="722"/>
- <char name="S" width="556"/>
- <char name="Scaron" width="556"/>
- <char name="T" width="667"/>
- <char name="Thorn" width="611"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="722"/>
- <char name="W" width="1000"/>
- <char name="X" width="722"/>
- <char name="Y" width="722"/>
- <char name="Yacute" width="722"/>
- <char name="Ydieresis" width="722"/>
- <char name="Z" width="667"/>
- <char name="Zcaron" width="667"/>
- <char name="a" width="500"/>
- <char name="aacute" width="500"/>
- <char name="acircumflex" width="500"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="500"/>
- <char name="ae" width="722"/>
- <char name="agrave" width="500"/>
- <char name="ampersand" width="833"/>
- <char name="aring" width="500"/>
- <char name="asciicircum" width="581"/>
- <char name="asciitilde" width="520"/>
- <char name="asterisk" width="500"/>
- <char name="at" width="930"/>
- <char name="atilde" width="500"/>
- <char name="b" width="556"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="220"/>
- <char name="braceleft" width="394"/>
- <char name="braceright" width="394"/>
- <char name="bracketleft" width="333"/>
- <char name="bracketright" width="333"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="220"/>
- <char name="bullet" width="350"/>
- <char name="c" width="444"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="444"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="500"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="333"/>
- <char name="comma" width="250"/>
- <char name="copyright" width="747"/>
- <char name="currency" width="500"/>
- <char name="d" width="556"/>
- <char name="dagger" width="500"/>
- <char name="daggerdbl" width="500"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="570"/>
- <char name="dollar" width="500"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="444"/>
- <char name="eacute" width="444"/>
- <char name="ecircumflex" width="444"/>
- <char name="edieresis" width="444"/>
- <char name="egrave" width="444"/>
- <char name="eight" width="500"/>
- <char name="ellipsis" width="1000"/>
- <char name="emdash" width="1000"/>
- <char name="endash" width="500"/>
- <char name="equal" width="570"/>
- <char name="eth" width="500"/>
- <char name="exclam" width="333"/>
- <char name="exclamdown" width="333"/>
- <char name="f" width="333"/>
- <char name="fi" width="556"/>
- <char name="five" width="500"/>
- <char name="fl" width="556"/>
- <char name="florin" width="500"/>
- <char name="four" width="500"/>
- <char name="fraction" width="167"/>
- <char name="g" width="500"/>
- <char name="germandbls" width="556"/>
- <char name="grave" width="333"/>
- <char name="greater" width="570"/>
- <char name="guillemotleft" width="500"/>
- <char name="guillemotright" width="500"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="556"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="278"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="333"/>
- <char name="k" width="556"/>
- <char name="l" width="278"/>
- <char name="less" width="570"/>
- <char name="logicalnot" width="570"/>
- <char name="lslash" width="278"/>
- <char name="m" width="833"/>
- <char name="macron" width="333"/>
- <char name="minus" width="324"/>
- <char name="mu" width="556"/>
- <char name="multiply" width="570"/>
- <char name="n" width="556"/>
- <char name="nine" width="500"/>
- <char name="ntilde" width="556"/>
- <char name="numbersign" width="500"/>
- <char name="o" width="500"/>
- <char name="oacute" width="500"/>
- <char name="ocircumflex" width="500"/>
- <char name="odieresis" width="500"/>
- <char name="oe" width="722"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="500"/>
- <char name="one" width="500"/>
- <char name="onehalf" width="750"/>
- <char name="onequarter" width="750"/>
- <char name="onesuperior" width="300"/>
- <char name="ordfeminine" width="300"/>
- <char name="ordmasculine" width="330"/>
- <char name="oslash" width="500"/>
- <char name="otilde" width="500"/>
- <char name="p" width="556"/>
- <char name="paragraph" width="540"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="1000"/>
- <char name="period" width="250"/>
- <char name="periodcentered" width="250"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="570"/>
- <char name="plusminus" width="570"/>
- <char name="q" width="556"/>
- <char name="question" width="500"/>
- <char name="questiondown" width="500"/>
- <char name="quotedbl" width="555"/>
- <char name="quotedblbase" width="500"/>
- <char name="quotedblleft" width="500"/>
- <char name="quotedblright" width="500"/>
- <char name="quoteleft" width="333"/>
- <char name="quoteright" width="333"/>
- <char name="quotesinglbase" width="333"/>
- <char name="quotesingle" width="278"/>
- <char name="r" width="444"/>
- <char name="registered" width="747"/>
- <char name="ring" width="333"/>
- <char name="s" width="389"/>
- <char name="scaron" width="389"/>
- <char name="section" width="500"/>
- <char name="semicolon" width="333"/>
- <char name="seven" width="500"/>
- <char name="six" width="500"/>
- <char name="slash" width="278"/>
- <char name="space" width="250"/>
+ <char-metrics>
+ <char name="A" width="722" llx="9" lly="0" urx="689" ury="690"/>
+ <char name="AE" width="1000" llx="4" lly="0" urx="951" ury="676"/>
+ <char name="Aacute" width="722" llx="9" lly="0" urx="689" ury="923"/>
+ <char name="Acircumflex" width="722" llx="9" lly="0" urx="689" ury="914"/>
+ <char name="Adieresis" width="722" llx="9" lly="0" urx="689" ury="877"/>
+ <char name="Agrave" width="722" llx="9" lly="0" urx="689" ury="923"/>
+ <char name="Aring" width="722" llx="9" lly="0" urx="689" ury="935"/>
+ <char name="Atilde" width="722" llx="9" lly="0" urx="689" ury="884"/>
+ <char name="B" width="667" llx="16" lly="0" urx="619" ury="676"/>
+ <char name="C" width="722" llx="49" lly="-19" urx="687" ury="691"/>
+ <char name="Ccedilla" width="722" llx="49" lly="-218" urx="687" ury="691"/>
+ <char name="D" width="722" llx="14" lly="0" urx="690" ury="676"/>
+ <char name="E" width="667" llx="16" lly="0" urx="641" ury="676"/>
+ <char name="Eacute" width="667" llx="16" lly="0" urx="641" ury="923"/>
+ <char name="Ecircumflex" width="667" llx="16" lly="0" urx="641" ury="914"/>
+ <char name="Edieresis" width="667" llx="16" lly="0" urx="641" ury="877"/>
+ <char name="Egrave" width="667" llx="16" lly="0" urx="641" ury="923"/>
+ <char name="Eth" width="722" llx="6" lly="0" urx="690" ury="676"/>
+ <char name="Euro" width="500" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="611" llx="16" lly="0" urx="583" ury="676"/>
+ <char name="G" width="778" llx="37" lly="-19" urx="755" ury="691"/>
+ <char name="H" width="778" llx="21" lly="0" urx="759" ury="676"/>
+ <char name="I" width="389" llx="20" lly="0" urx="370" ury="676"/>
+ <char name="Iacute" width="389" llx="20" lly="0" urx="370" ury="923"/>
+ <char name="Icircumflex" width="389" llx="20" lly="0" urx="370" ury="914"/>
+ <char name="Idieresis" width="389" llx="20" lly="0" urx="370" ury="877"/>
+ <char name="Igrave" width="389" llx="20" lly="0" urx="370" ury="923"/>
+ <char name="J" width="500" llx="3" lly="-96" urx="479" ury="676"/>
+ <char name="K" width="778" llx="30" lly="0" urx="769" ury="676"/>
+ <char name="L" width="667" llx="19" lly="0" urx="638" ury="676"/>
+ <char name="Lslash" width="667" llx="19" lly="0" urx="638" ury="676"/>
+ <char name="M" width="944" llx="14" lly="0" urx="921" ury="676"/>
+ <char name="N" width="722" llx="16" lly="-18" urx="701" ury="676"/>
+ <char name="Ntilde" width="722" llx="16" lly="-18" urx="701" ury="884"/>
+ <char name="O" width="778" llx="35" lly="-19" urx="743" ury="691"/>
+ <char name="OE" width="1000" llx="22" lly="-5" urx="981" ury="684"/>
+ <char name="Oacute" width="778" llx="35" lly="-19" urx="743" ury="923"/>
+ <char name="Ocircumflex" width="778" llx="35" lly="-19" urx="743" ury="914"/>
+ <char name="Odieresis" width="778" llx="35" lly="-19" urx="743" ury="877"/>
+ <char name="Ograve" width="778" llx="35" lly="-19" urx="743" ury="923"/>
+ <char name="Oslash" width="778" llx="35" lly="-74" urx="743" ury="737"/>
+ <char name="Otilde" width="778" llx="35" lly="-19" urx="743" ury="884"/>
+ <char name="P" width="611" llx="16" lly="0" urx="600" ury="676"/>
+ <char name="Q" width="778" llx="35" lly="-176" urx="743" ury="691"/>
+ <char name="R" width="722" llx="26" lly="0" urx="715" ury="676"/>
+ <char name="S" width="556" llx="35" lly="-19" urx="513" ury="692"/>
+ <char name="Scaron" width="556" llx="35" lly="-19" urx="513" ury="914"/>
+ <char name="T" width="667" llx="31" lly="0" urx="636" ury="676"/>
+ <char name="Thorn" width="611" llx="16" lly="0" urx="600" ury="676"/>
+ <char name="U" width="722" llx="16" lly="-19" urx="701" ury="676"/>
+ <char name="Uacute" width="722" llx="16" lly="-19" urx="701" ury="923"/>
+ <char name="Ucircumflex" width="722" llx="16" lly="-19" urx="701" ury="914"/>
+ <char name="Udieresis" width="722" llx="16" lly="-19" urx="701" ury="877"/>
+ <char name="Ugrave" width="722" llx="16" lly="-19" urx="701" ury="923"/>
+ <char name="V" width="722" llx="16" lly="-18" urx="701" ury="676"/>
+ <char name="W" width="1000" llx="19" lly="-15" urx="981" ury="676"/>
+ <char name="X" width="722" llx="16" lly="0" urx="699" ury="676"/>
+ <char name="Y" width="722" llx="15" lly="0" urx="699" ury="676"/>
+ <char name="Yacute" width="722" llx="15" lly="0" urx="699" ury="923"/>
+ <char name="Ydieresis" width="722" llx="15" lly="0" urx="699" ury="877"/>
+ <char name="Z" width="667" llx="28" lly="0" urx="634" ury="676"/>
+ <char name="Zcaron" width="667" llx="28" lly="0" urx="634" ury="914"/>
+ <char name="a" width="500" llx="25" lly="-14" urx="488" ury="473"/>
+ <char name="aacute" width="500" llx="25" lly="-14" urx="488" ury="713"/>
+ <char name="acircumflex" width="500" llx="25" lly="-14" urx="488" ury="704"/>
+ <char name="acute" width="333" llx="86" lly="528" urx="324" ury="713"/>
+ <char name="adieresis" width="500" llx="25" lly="-14" urx="488" ury="667"/>
+ <char name="ae" width="722" llx="33" lly="-14" urx="693" ury="473"/>
+ <char name="agrave" width="500" llx="25" lly="-14" urx="488" ury="713"/>
+ <char name="ampersand" width="833" llx="62" lly="-16" urx="787" ury="691"/>
+ <char name="aring" width="500" llx="25" lly="-14" urx="488" ury="740"/>
+ <char name="asciicircum" width="581" llx="73" lly="311" urx="509" ury="676"/>
+ <char name="asciitilde" width="520" llx="29" lly="173" urx="491" ury="333"/>
+ <char name="asterisk" width="500" llx="56" lly="255" urx="447" ury="691"/>
+ <char name="at" width="930" llx="108" lly="-19" urx="822" ury="691"/>
+ <char name="atilde" width="500" llx="25" lly="-14" urx="488" ury="674"/>
+ <char name="b" width="556" llx="17" lly="-14" urx="521" ury="676"/>
+ <char name="backslash" width="278" llx="-25" lly="-19" urx="303" ury="691"/>
+ <char name="bar" width="220" llx="66" lly="-218" urx="154" ury="782"/>
+ <char name="braceleft" width="394" llx="22" lly="-175" urx="340" ury="698"/>
+ <char name="braceright" width="394" llx="54" lly="-175" urx="372" ury="698"/>
+ <char name="bracketleft" width="333" llx="67" lly="-149" urx="301" ury="678"/>
+ <char name="bracketright" width="333" llx="32" lly="-149" urx="266" ury="678"/>
+ <char name="breve" width="333" llx="15" lly="528" urx="318" ury="691"/>
+ <char name="brokenbar" width="220" llx="66" lly="-143" urx="154" ury="707"/>
+ <char name="bullet" width="350" llx="35" lly="198" urx="315" ury="478"/>
+ <char name="c" width="444" llx="25" lly="-14" urx="430" ury="473"/>
+ <char name="caron" width="333" llx="-2" lly="528" urx="335" ury="704"/>
+ <char name="ccedilla" width="444" llx="25" lly="-218" urx="430" ury="473"/>
+ <char name="cedilla" width="333" llx="68" lly="-218" urx="294" ury="0"/>
+ <char name="cent" width="500" llx="53" lly="-140" urx="458" ury="588"/>
+ <char name="circumflex" width="333" llx="-2" lly="528" urx="335" ury="704"/>
+ <char name="colon" width="333" llx="82" lly="-13" urx="251" ury="472"/>
+ <char name="comma" width="250" llx="39" lly="-180" urx="223" ury="155"/>
+ <char name="copyright" width="747" llx="26" lly="-19" urx="721" ury="691"/>
+ <char name="currency" width="500" llx="-26" lly="61" urx="526" ury="613"/>
+ <char name="d" width="556" llx="25" lly="-14" urx="534" ury="676"/>
+ <char name="dagger" width="500" llx="47" lly="-134" urx="453" ury="691"/>
+ <char name="daggerdbl" width="500" llx="45" lly="-132" urx="456" ury="691"/>
+ <char name="degree" width="400" llx="57" lly="402" urx="343" ury="688"/>
+ <char name="dieresis" width="333" llx="-2" lly="537" urx="335" ury="667"/>
+ <char name="divide" width="570" llx="33" lly="-31" urx="537" ury="537"/>
+ <char name="dollar" width="500" llx="29" lly="-99" urx="472" ury="750"/>
+ <char name="dotaccent" width="333" llx="103" lly="536" urx="258" ury="691"/>
+ <char name="dotlessi" width="278" llx="16" lly="0" urx="255" ury="461"/>
+ <char name="e" width="444" llx="25" lly="-14" urx="426" ury="473"/>
+ <char name="eacute" width="444" llx="25" lly="-14" urx="426" ury="713"/>
+ <char name="ecircumflex" width="444" llx="25" lly="-14" urx="426" ury="704"/>
+ <char name="edieresis" width="444" llx="25" lly="-14" urx="426" ury="667"/>
+ <char name="egrave" width="444" llx="25" lly="-14" urx="426" ury="713"/>
+ <char name="eight" width="500" llx="28" lly="-13" urx="472" ury="688"/>
+ <char name="ellipsis" width="1000" llx="82" lly="-13" urx="917" ury="156"/>
+ <char name="emdash" width="1000" llx="0" lly="181" urx="1000" ury="271"/>
+ <char name="endash" width="500" llx="0" lly="181" urx="500" ury="271"/>
+ <char name="equal" width="570" llx="33" lly="107" urx="537" ury="399"/>
+ <char name="eth" width="500" llx="25" lly="-14" urx="476" ury="691"/>
+ <char name="exclam" width="333" llx="81" lly="-13" urx="251" ury="691"/>
+ <char name="exclamdown" width="333" llx="82" lly="-203" urx="252" ury="501"/>
+ <char name="f" width="333" llx="14" lly="0" urx="389" ury="691"/>
+ <char name="fi" width="556" llx="14" lly="0" urx="536" ury="691"/>
+ <char name="five" width="500" llx="22" lly="-8" urx="470" ury="676"/>
+ <char name="fl" width="556" llx="14" lly="0" urx="536" ury="691"/>
+ <char name="florin" width="500" llx="0" lly="-155" urx="498" ury="706"/>
+ <char name="four" width="500" llx="19" lly="0" urx="475" ury="688"/>
+ <char name="fraction" width="167" llx="-168" lly="-12" urx="329" ury="688"/>
+ <char name="g" width="500" llx="28" lly="-206" urx="483" ury="473"/>
+ <char name="germandbls" width="556" llx="19" lly="-12" urx="517" ury="691"/>
+ <char name="grave" width="333" llx="8" lly="528" urx="246" ury="713"/>
+ <char name="greater" width="570" llx="31" lly="-8" urx="539" ury="514"/>
+ <char name="guillemotleft" width="500" llx="23" lly="36" urx="473" ury="415"/>
+ <char name="guillemotright" width="500" llx="27" lly="36" urx="477" ury="415"/>
+ <char name="guilsinglleft" width="333" llx="51" lly="36" urx="305" ury="415"/>
+ <char name="guilsinglright" width="333" llx="28" lly="36" urx="282" ury="415"/>
+ <char name="h" width="556" llx="16" lly="0" urx="534" ury="676"/>
+ <char name="hungarumlaut" width="333" llx="-13" lly="528" urx="425" ury="713"/>
+ <char name="hyphen" width="333" llx="44" lly="171" urx="287" ury="287"/>
+ <char name="i" width="278" llx="16" lly="0" urx="255" ury="691"/>
+ <char name="iacute" width="278" llx="16" lly="0" urx="289" ury="713"/>
+ <char name="icircumflex" width="278" llx="-37" lly="0" urx="300" ury="704"/>
+ <char name="idieresis" width="278" llx="-37" lly="0" urx="300" ury="667"/>
+ <char name="igrave" width="278" llx="-27" lly="0" urx="255" ury="713"/>
+ <char name="j" width="333" llx="-57" lly="-203" urx="263" ury="691"/>
+ <char name="k" width="556" llx="22" lly="0" urx="543" ury="676"/>
+ <char name="l" width="278" llx="16" lly="0" urx="255" ury="676"/>
+ <char name="less" width="570" llx="31" lly="-8" urx="539" ury="514"/>
+ <char name="logicalnot" width="570" llx="33" lly="108" urx="537" ury="399"/>
+ <char name="lslash" width="278" llx="-22" lly="0" urx="303" ury="676"/>
+ <char name="m" width="833" llx="16" lly="0" urx="814" ury="473"/>
+ <char name="macron" width="333" llx="1" lly="565" urx="331" ury="637"/>
+ <char name="minus" width="324" llx="33" lly="209" urx="537" ury="297"/>
+ <char name="mu" width="556" llx="33" lly="-206" urx="536" ury="461"/>
+ <char name="multiply" width="570" llx="48" lly="16" urx="522" ury="490"/>
+ <char name="n" width="556" llx="21" lly="0" urx="539" ury="473"/>
+ <char name="nine" width="500" llx="26" lly="-13" urx="473" ury="688"/>
+ <char name="ntilde" width="556" llx="21" lly="0" urx="539" ury="674"/>
+ <char name="numbersign" width="500" llx="4" lly="0" urx="496" ury="700"/>
+ <char name="o" width="500" llx="25" lly="-14" urx="476" ury="473"/>
+ <char name="oacute" width="500" llx="25" lly="-14" urx="476" ury="713"/>
+ <char name="ocircumflex" width="500" llx="25" lly="-14" urx="476" ury="704"/>
+ <char name="odieresis" width="500" llx="25" lly="-14" urx="476" ury="667"/>
+ <char name="oe" width="722" llx="22" lly="-14" urx="696" ury="473"/>
+ <char name="ogonek" width="333" llx="90" lly="-193" urx="319" ury="24"/>
+ <char name="ograve" width="500" llx="25" lly="-14" urx="476" ury="713"/>
+ <char name="one" width="500" llx="65" lly="0" urx="442" ury="688"/>
+ <char name="onehalf" width="750" llx="-7" lly="-12" urx="775" ury="688"/>
+ <char name="onequarter" width="750" llx="28" lly="-12" urx="743" ury="688"/>
+ <char name="onesuperior" width="300" llx="28" lly="275" urx="273" ury="688"/>
+ <char name="ordfeminine" width="300" llx="-1" lly="397" urx="301" ury="688"/>
+ <char name="ordmasculine" width="330" llx="18" lly="397" urx="312" ury="688"/>
+ <char name="oslash" width="500" llx="25" lly="-92" urx="476" ury="549"/>
+ <char name="otilde" width="500" llx="25" lly="-14" urx="476" ury="674"/>
+ <char name="p" width="556" llx="19" lly="-205" urx="524" ury="473"/>
+ <char name="paragraph" width="540" llx="0" lly="-186" urx="519" ury="676"/>
+ <char name="parenleft" width="333" llx="46" lly="-168" urx="306" ury="694"/>
+ <char name="parenright" width="333" llx="27" lly="-168" urx="287" ury="694"/>
+ <char name="percent" width="1000" llx="124" lly="-14" urx="877" ury="692"/>
+ <char name="period" width="250" llx="41" lly="-13" urx="210" ury="156"/>
+ <char name="periodcentered" width="250" llx="41" lly="248" urx="210" ury="417"/>
+ <char name="perthousand" width="1000" llx="7" lly="-29" urx="995" ury="706"/>
+ <char name="plus" width="570" llx="33" lly="0" urx="537" ury="506"/>
+ <char name="plusminus" width="570" llx="33" lly="0" urx="537" ury="506"/>
+ <char name="q" width="556" llx="34" lly="-205" urx="536" ury="473"/>
+ <char name="question" width="500" llx="57" lly="-13" urx="445" ury="689"/>
+ <char name="questiondown" width="500" llx="55" lly="-201" urx="443" ury="501"/>
+ <char name="quotedbl" width="555" llx="83" lly="404" urx="472" ury="691"/>
+ <char name="quotedblbase" width="500" llx="14" lly="-180" urx="468" ury="155"/>
+ <char name="quotedblleft" width="500" llx="32" lly="356" urx="486" ury="691"/>
+ <char name="quotedblright" width="500" llx="14" lly="356" urx="468" ury="691"/>
+ <char name="quoteleft" width="333" llx="70" lly="356" urx="254" ury="691"/>
+ <char name="quoteright" width="333" llx="79" lly="356" urx="263" ury="691"/>
+ <char name="quotesinglbase" width="333" llx="79" lly="-180" urx="263" ury="155"/>
+ <char name="quotesingle" width="278" llx="75" lly="404" urx="204" ury="691"/>
+ <char name="r" width="444" llx="29" lly="0" urx="434" ury="473"/>
+ <char name="registered" width="747" llx="26" lly="-19" urx="721" ury="691"/>
+ <char name="ring" width="333" llx="60" lly="527" urx="273" ury="740"/>
+ <char name="s" width="389" llx="25" lly="-14" urx="361" ury="473"/>
+ <char name="scaron" width="389" llx="25" lly="-14" urx="363" ury="704"/>
+ <char name="section" width="500" llx="57" lly="-132" urx="443" ury="691"/>
+ <char name="semicolon" width="333" llx="82" lly="-180" urx="266" ury="472"/>
+ <char name="seven" width="500" llx="17" lly="0" urx="477" ury="676"/>
+ <char name="six" width="500" llx="28" lly="-13" urx="475" ury="688"/>
+ <char name="slash" width="278" llx="-24" lly="-19" urx="302" ury="691"/>
+ <char name="space" width="250" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="250"/>
- <char name="sterling" width="500"/>
- <char name="t" width="333"/>
- <char name="thorn" width="556"/>
- <char name="three" width="500"/>
- <char name="threequarters" width="750"/>
- <char name="threesuperior" width="300"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="1000"/>
- <char name="two" width="500"/>
- <char name="twosuperior" width="300"/>
- <char name="u" width="556"/>
- <char name="uacute" width="556"/>
- <char name="ucircumflex" width="556"/>
- <char name="udieresis" width="556"/>
- <char name="ugrave" width="556"/>
- <char name="underscore" width="500"/>
- <char name="v" width="500"/>
- <char name="w" width="722"/>
- <char name="x" width="500"/>
- <char name="y" width="500"/>
- <char name="yacute" width="500"/>
- <char name="ydieresis" width="500"/>
- <char name="yen" width="500"/>
- <char name="z" width="444"/>
- <char name="zcaron" width="444"/>
- <char name="zero" width="500"/>
- </widths>
+ <char name="sterling" width="500" llx="21" lly="-14" urx="477" ury="684"/>
+ <char name="t" width="333" llx="20" lly="-12" urx="332" ury="630"/>
+ <char name="thorn" width="556" llx="19" lly="-205" urx="524" ury="676"/>
+ <char name="three" width="500" llx="16" lly="-14" urx="468" ury="688"/>
+ <char name="threequarters" width="750" llx="23" lly="-12" urx="733" ury="688"/>
+ <char name="threesuperior" width="300" llx="3" lly="268" urx="297" ury="688"/>
+ <char name="tilde" width="333" llx="-16" lly="547" urx="349" ury="674"/>
+ <char name="trademark" width="1000" llx="24" lly="271" urx="977" ury="676"/>
+ <char name="two" width="500" llx="17" lly="0" urx="478" ury="688"/>
+ <char name="twosuperior" width="300" llx="0" lly="275" urx="300" ury="688"/>
+ <char name="u" width="556" llx="16" lly="-14" urx="537" ury="461"/>
+ <char name="uacute" width="556" llx="16" lly="-14" urx="537" ury="713"/>
+ <char name="ucircumflex" width="556" llx="16" lly="-14" urx="537" ury="704"/>
+ <char name="udieresis" width="556" llx="16" lly="-14" urx="537" ury="667"/>
+ <char name="ugrave" width="556" llx="16" lly="-14" urx="537" ury="713"/>
+ <char name="underscore" width="500" llx="0" lly="-125" urx="500" ury="-75"/>
+ <char name="v" width="500" llx="21" lly="-14" urx="485" ury="461"/>
+ <char name="w" width="722" llx="23" lly="-14" urx="707" ury="461"/>
+ <char name="x" width="500" llx="12" lly="0" urx="484" ury="461"/>
+ <char name="y" width="500" llx="16" lly="-205" urx="480" ury="461"/>
+ <char name="yacute" width="500" llx="16" lly="-205" urx="480" ury="713"/>
+ <char name="ydieresis" width="500" llx="16" lly="-205" urx="480" ury="667"/>
+ <char name="yen" width="500" llx="-64" lly="0" urx="547" ury="676"/>
+ <char name="z" width="444" llx="21" lly="0" urx="420" ury="461"/>
+ <char name="zcaron" width="444" llx="21" lly="0" urx="420" ury="704"/>
+ <char name="zero" width="500" llx="24" lly="-13" urx="476" ury="688"/>
+ </char-metrics>
<kerning kpx1="79">
<pair kern="-40" kpx2="65"/>
<pair kern="-50" kpx2="87"/>
diff --git a/src/codegen/fonts/TimesBoldItalic.xml b/src/codegen/fonts/TimesBoldItalic.xml
index d1b70872a..ec67369f2 100644
--- a/src/codegen/fonts/TimesBoldItalic.xml
+++ b/src/codegen/fonts/TimesBoldItalic.xml
@@ -22,245 +22,246 @@
<family-name>Times</family-name>
<class-name>TimesBoldItalic</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>669</cap-height>
<x-height>462</x-height>
<ascender>699</ascender>
<descender>-205</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="667"/>
- <char name="AE" width="944"/>
- <char name="Aacute" width="667"/>
- <char name="Acircumflex" width="667"/>
- <char name="Adieresis" width="667"/>
- <char name="Agrave" width="667"/>
- <char name="Aring" width="667"/>
- <char name="Atilde" width="667"/>
- <char name="B" width="667"/>
- <char name="C" width="667"/>
- <char name="Ccedilla" width="667"/>
- <char name="D" width="722"/>
- <char name="E" width="667"/>
- <char name="Eacute" width="667"/>
- <char name="Ecircumflex" width="667"/>
- <char name="Edieresis" width="667"/>
- <char name="Egrave" width="667"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="500"/>
- <char name="F" width="667"/>
- <char name="G" width="722"/>
- <char name="H" width="778"/>
- <char name="I" width="389"/>
- <char name="Iacute" width="389"/>
- <char name="Icircumflex" width="389"/>
- <char name="Idieresis" width="389"/>
- <char name="Igrave" width="389"/>
- <char name="J" width="500"/>
- <char name="K" width="667"/>
- <char name="L" width="611"/>
- <char name="Lslash" width="611"/>
- <char name="M" width="889"/>
- <char name="N" width="722"/>
- <char name="Ntilde" width="722"/>
- <char name="O" width="722"/>
- <char name="OE" width="944"/>
- <char name="Oacute" width="722"/>
- <char name="Ocircumflex" width="722"/>
- <char name="Odieresis" width="722"/>
- <char name="Ograve" width="722"/>
- <char name="Oslash" width="722"/>
- <char name="Otilde" width="722"/>
- <char name="P" width="611"/>
- <char name="Q" width="722"/>
- <char name="R" width="667"/>
- <char name="S" width="556"/>
- <char name="Scaron" width="556"/>
- <char name="T" width="611"/>
- <char name="Thorn" width="611"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="667"/>
- <char name="W" width="889"/>
- <char name="X" width="667"/>
- <char name="Y" width="611"/>
- <char name="Yacute" width="611"/>
- <char name="Ydieresis" width="611"/>
- <char name="Z" width="611"/>
- <char name="Zcaron" width="611"/>
- <char name="a" width="500"/>
- <char name="aacute" width="500"/>
- <char name="acircumflex" width="500"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="500"/>
- <char name="ae" width="722"/>
- <char name="agrave" width="500"/>
- <char name="ampersand" width="778"/>
- <char name="aring" width="500"/>
- <char name="asciicircum" width="570"/>
- <char name="asciitilde" width="570"/>
- <char name="asterisk" width="500"/>
- <char name="at" width="832"/>
- <char name="atilde" width="500"/>
- <char name="b" width="500"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="220"/>
- <char name="braceleft" width="348"/>
- <char name="braceright" width="348"/>
- <char name="bracketleft" width="333"/>
- <char name="bracketright" width="333"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="220"/>
- <char name="bullet" width="350"/>
- <char name="c" width="444"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="444"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="500"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="333"/>
- <char name="comma" width="250"/>
- <char name="copyright" width="747"/>
- <char name="currency" width="500"/>
- <char name="d" width="500"/>
- <char name="dagger" width="500"/>
- <char name="daggerdbl" width="500"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="570"/>
- <char name="dollar" width="500"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="444"/>
- <char name="eacute" width="444"/>
- <char name="ecircumflex" width="444"/>
- <char name="edieresis" width="444"/>
- <char name="egrave" width="444"/>
- <char name="eight" width="500"/>
- <char name="ellipsis" width="1000"/>
- <char name="emdash" width="1000"/>
- <char name="endash" width="500"/>
- <char name="equal" width="570"/>
- <char name="eth" width="500"/>
- <char name="exclam" width="389"/>
- <char name="exclamdown" width="389"/>
- <char name="f" width="333"/>
- <char name="fi" width="556"/>
- <char name="five" width="500"/>
- <char name="fl" width="556"/>
- <char name="florin" width="500"/>
- <char name="four" width="500"/>
- <char name="fraction" width="167"/>
- <char name="g" width="500"/>
- <char name="germandbls" width="500"/>
- <char name="grave" width="333"/>
- <char name="greater" width="570"/>
- <char name="guillemotleft" width="500"/>
- <char name="guillemotright" width="500"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="556"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="278"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="278"/>
- <char name="k" width="500"/>
- <char name="l" width="278"/>
- <char name="less" width="570"/>
- <char name="logicalnot" width="606"/>
- <char name="lslash" width="278"/>
- <char name="m" width="778"/>
- <char name="macron" width="333"/>
- <char name="minus" width="330"/>
- <char name="mu" width="576"/>
- <char name="multiply" width="570"/>
- <char name="n" width="556"/>
- <char name="nine" width="500"/>
- <char name="ntilde" width="556"/>
- <char name="numbersign" width="500"/>
- <char name="o" width="500"/>
- <char name="oacute" width="500"/>
- <char name="ocircumflex" width="500"/>
- <char name="odieresis" width="500"/>
- <char name="oe" width="722"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="500"/>
- <char name="one" width="500"/>
- <char name="onehalf" width="750"/>
- <char name="onequarter" width="750"/>
- <char name="onesuperior" width="300"/>
- <char name="ordfeminine" width="266"/>
- <char name="ordmasculine" width="300"/>
- <char name="oslash" width="500"/>
- <char name="otilde" width="500"/>
- <char name="p" width="500"/>
- <char name="paragraph" width="500"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="833"/>
- <char name="period" width="250"/>
- <char name="periodcentered" width="250"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="570"/>
- <char name="plusminus" width="570"/>
- <char name="q" width="500"/>
- <char name="question" width="500"/>
- <char name="questiondown" width="500"/>
- <char name="quotedbl" width="555"/>
- <char name="quotedblbase" width="500"/>
- <char name="quotedblleft" width="500"/>
- <char name="quotedblright" width="500"/>
- <char name="quoteleft" width="333"/>
- <char name="quoteright" width="333"/>
- <char name="quotesinglbase" width="333"/>
- <char name="quotesingle" width="278"/>
- <char name="r" width="389"/>
- <char name="registered" width="747"/>
- <char name="ring" width="333"/>
- <char name="s" width="389"/>
- <char name="scaron" width="389"/>
- <char name="section" width="500"/>
- <char name="semicolon" width="333"/>
- <char name="seven" width="500"/>
- <char name="six" width="500"/>
- <char name="slash" width="278"/>
- <char name="space" width="250"/>
+ <char-metrics>
+ <char name="A" width="667" llx="-67" lly="0" urx="593" ury="683"/>
+ <char name="AE" width="944" llx="-64" lly="0" urx="918" ury="669"/>
+ <char name="Aacute" width="667" llx="-67" lly="0" urx="593" ury="904"/>
+ <char name="Acircumflex" width="667" llx="-67" lly="0" urx="593" ury="897"/>
+ <char name="Adieresis" width="667" llx="-67" lly="0" urx="593" ury="862"/>
+ <char name="Agrave" width="667" llx="-67" lly="0" urx="593" ury="904"/>
+ <char name="Aring" width="667" llx="-67" lly="0" urx="593" ury="921"/>
+ <char name="Atilde" width="667" llx="-67" lly="0" urx="593" ury="862"/>
+ <char name="B" width="667" llx="-24" lly="0" urx="624" ury="669"/>
+ <char name="C" width="667" llx="32" lly="-18" urx="677" ury="685"/>
+ <char name="Ccedilla" width="667" llx="32" lly="-218" urx="677" ury="685"/>
+ <char name="D" width="722" llx="-46" lly="0" urx="685" ury="669"/>
+ <char name="E" width="667" llx="-27" lly="0" urx="653" ury="669"/>
+ <char name="Eacute" width="667" llx="-27" lly="0" urx="653" ury="904"/>
+ <char name="Ecircumflex" width="667" llx="-27" lly="0" urx="653" ury="897"/>
+ <char name="Edieresis" width="667" llx="-27" lly="0" urx="653" ury="862"/>
+ <char name="Egrave" width="667" llx="-27" lly="0" urx="653" ury="904"/>
+ <char name="Eth" width="722" llx="-31" lly="0" urx="700" ury="669"/>
+ <char name="Euro" width="500" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="667" llx="-13" lly="0" urx="660" ury="669"/>
+ <char name="G" width="722" llx="21" lly="-18" urx="706" ury="685"/>
+ <char name="H" width="778" llx="-24" lly="0" urx="799" ury="669"/>
+ <char name="I" width="389" llx="-32" lly="0" urx="406" ury="669"/>
+ <char name="Iacute" width="389" llx="-32" lly="0" urx="432" ury="904"/>
+ <char name="Icircumflex" width="389" llx="-32" lly="0" urx="450" ury="897"/>
+ <char name="Idieresis" width="389" llx="-32" lly="0" urx="450" ury="862"/>
+ <char name="Igrave" width="389" llx="-32" lly="0" urx="406" ury="904"/>
+ <char name="J" width="500" llx="-46" lly="-99" urx="524" ury="669"/>
+ <char name="K" width="667" llx="-21" lly="0" urx="702" ury="669"/>
+ <char name="L" width="611" llx="-22" lly="0" urx="590" ury="669"/>
+ <char name="Lslash" width="611" llx="-22" lly="0" urx="590" ury="669"/>
+ <char name="M" width="889" llx="-29" lly="-12" urx="917" ury="669"/>
+ <char name="N" width="722" llx="-27" lly="-15" urx="748" ury="669"/>
+ <char name="Ntilde" width="722" llx="-27" lly="-15" urx="748" ury="862"/>
+ <char name="O" width="722" llx="27" lly="-18" urx="691" ury="685"/>
+ <char name="OE" width="944" llx="23" lly="-8" urx="946" ury="677"/>
+ <char name="Oacute" width="722" llx="27" lly="-18" urx="691" ury="904"/>
+ <char name="Ocircumflex" width="722" llx="27" lly="-18" urx="691" ury="897"/>
+ <char name="Odieresis" width="722" llx="27" lly="-18" urx="691" ury="862"/>
+ <char name="Ograve" width="722" llx="27" lly="-18" urx="691" ury="904"/>
+ <char name="Oslash" width="722" llx="27" lly="-125" urx="691" ury="764"/>
+ <char name="Otilde" width="722" llx="27" lly="-18" urx="691" ury="862"/>
+ <char name="P" width="611" llx="-27" lly="0" urx="613" ury="669"/>
+ <char name="Q" width="722" llx="27" lly="-208" urx="691" ury="685"/>
+ <char name="R" width="667" llx="-29" lly="0" urx="623" ury="669"/>
+ <char name="S" width="556" llx="2" lly="-18" urx="526" ury="685"/>
+ <char name="Scaron" width="556" llx="2" lly="-18" urx="553" ury="897"/>
+ <char name="T" width="611" llx="50" lly="0" urx="650" ury="669"/>
+ <char name="Thorn" width="611" llx="-27" lly="0" urx="573" ury="669"/>
+ <char name="U" width="722" llx="67" lly="-18" urx="744" ury="669"/>
+ <char name="Uacute" width="722" llx="67" lly="-18" urx="744" ury="904"/>
+ <char name="Ucircumflex" width="722" llx="67" lly="-18" urx="744" ury="897"/>
+ <char name="Udieresis" width="722" llx="67" lly="-18" urx="744" ury="862"/>
+ <char name="Ugrave" width="722" llx="67" lly="-18" urx="744" ury="904"/>
+ <char name="V" width="667" llx="65" lly="-18" urx="715" ury="669"/>
+ <char name="W" width="889" llx="65" lly="-18" urx="940" ury="669"/>
+ <char name="X" width="667" llx="-24" lly="0" urx="694" ury="669"/>
+ <char name="Y" width="611" llx="73" lly="0" urx="659" ury="669"/>
+ <char name="Yacute" width="611" llx="73" lly="0" urx="659" ury="904"/>
+ <char name="Ydieresis" width="611" llx="73" lly="0" urx="659" ury="862"/>
+ <char name="Z" width="611" llx="-11" lly="0" urx="590" ury="669"/>
+ <char name="Zcaron" width="611" llx="-11" lly="0" urx="590" ury="897"/>
+ <char name="a" width="500" llx="-21" lly="-14" urx="455" ury="462"/>
+ <char name="aacute" width="500" llx="-21" lly="-14" urx="463" ury="697"/>
+ <char name="acircumflex" width="500" llx="-21" lly="-14" urx="455" ury="690"/>
+ <char name="acute" width="333" llx="139" lly="516" urx="379" ury="697"/>
+ <char name="adieresis" width="500" llx="-21" lly="-14" urx="476" ury="655"/>
+ <char name="ae" width="722" llx="-5" lly="-13" urx="673" ury="462"/>
+ <char name="agrave" width="500" llx="-21" lly="-14" urx="455" ury="697"/>
+ <char name="ampersand" width="778" llx="5" lly="-19" urx="699" ury="682"/>
+ <char name="aring" width="500" llx="-21" lly="-14" urx="455" ury="729"/>
+ <char name="asciicircum" width="570" llx="67" lly="304" urx="503" ury="669"/>
+ <char name="asciitilde" width="570" llx="54" lly="173" urx="516" ury="333"/>
+ <char name="asterisk" width="500" llx="65" lly="249" urx="456" ury="685"/>
+ <char name="at" width="832" llx="63" lly="-18" urx="770" ury="685"/>
+ <char name="atilde" width="500" llx="-21" lly="-14" urx="491" ury="655"/>
+ <char name="b" width="500" llx="-14" lly="-13" urx="444" ury="699"/>
+ <char name="backslash" width="278" llx="-1" lly="-18" urx="279" ury="685"/>
+ <char name="bar" width="220" llx="66" lly="-218" urx="154" ury="782"/>
+ <char name="braceleft" width="348" llx="5" lly="-187" urx="436" ury="686"/>
+ <char name="braceright" width="348" llx="-129" lly="-187" urx="302" ury="686"/>
+ <char name="bracketleft" width="333" llx="-37" lly="-159" urx="362" ury="674"/>
+ <char name="bracketright" width="333" llx="-56" lly="-157" urx="343" ury="674"/>
+ <char name="breve" width="333" llx="71" lly="516" urx="387" ury="678"/>
+ <char name="brokenbar" width="220" llx="66" lly="-143" urx="154" ury="707"/>
+ <char name="bullet" width="350" llx="0" lly="175" urx="350" ury="525"/>
+ <char name="c" width="444" llx="-5" lly="-13" urx="392" ury="462"/>
+ <char name="caron" width="333" llx="79" lly="516" urx="411" ury="690"/>
+ <char name="ccedilla" width="444" llx="-5" lly="-218" urx="392" ury="462"/>
+ <char name="cedilla" width="333" llx="-80" lly="-218" urx="156" ury="5"/>
+ <char name="cent" width="500" llx="42" lly="-143" urx="439" ury="576"/>
+ <char name="circumflex" width="333" llx="40" lly="516" urx="367" ury="690"/>
+ <char name="colon" width="333" llx="23" lly="-13" urx="264" ury="459"/>
+ <char name="comma" width="250" llx="-60" lly="-182" urx="144" ury="134"/>
+ <char name="copyright" width="747" llx="30" lly="-18" urx="718" ury="685"/>
+ <char name="currency" width="500" llx="-26" lly="34" urx="526" ury="586"/>
+ <char name="d" width="500" llx="-21" lly="-13" urx="517" ury="699"/>
+ <char name="dagger" width="500" llx="91" lly="-145" urx="494" ury="685"/>
+ <char name="daggerdbl" width="500" llx="10" lly="-139" urx="493" ury="685"/>
+ <char name="degree" width="400" llx="83" lly="397" urx="369" ury="683"/>
+ <char name="dieresis" width="333" llx="55" lly="550" urx="402" ury="684"/>
+ <char name="divide" width="570" llx="33" lly="-29" urx="537" ury="535"/>
+ <char name="dollar" width="500" llx="-20" lly="-100" urx="497" ury="733"/>
+ <char name="dotaccent" width="333" llx="163" lly="550" urx="298" ury="684"/>
+ <char name="dotlessi" width="278" llx="2" lly="-9" urx="238" ury="462"/>
+ <char name="e" width="444" llx="5" lly="-13" urx="398" ury="462"/>
+ <char name="eacute" width="444" llx="5" lly="-13" urx="435" ury="697"/>
+ <char name="ecircumflex" width="444" llx="5" lly="-13" urx="423" ury="690"/>
+ <char name="edieresis" width="444" llx="5" lly="-13" urx="448" ury="655"/>
+ <char name="egrave" width="444" llx="5" lly="-13" urx="398" ury="697"/>
+ <char name="eight" width="500" llx="3" lly="-13" urx="476" ury="683"/>
+ <char name="ellipsis" width="1000" llx="40" lly="-13" urx="852" ury="135"/>
+ <char name="emdash" width="1000" llx="-40" lly="178" urx="977" ury="269"/>
+ <char name="endash" width="500" llx="-40" lly="178" urx="477" ury="269"/>
+ <char name="equal" width="570" llx="33" lly="107" urx="537" ury="399"/>
+ <char name="eth" width="500" llx="-3" lly="-13" urx="454" ury="699"/>
+ <char name="exclam" width="389" llx="67" lly="-13" urx="370" ury="684"/>
+ <char name="exclamdown" width="389" llx="19" lly="-205" urx="322" ury="492"/>
+ <char name="f" width="333" llx="-169" lly="-205" urx="446" ury="698"/>
+ <char name="fi" width="556" llx="-188" lly="-205" urx="514" ury="703"/>
+ <char name="five" width="500" llx="-11" lly="-13" urx="487" ury="669"/>
+ <char name="fl" width="556" llx="-186" lly="-205" urx="553" ury="704"/>
+ <char name="florin" width="500" llx="-87" lly="-156" urx="537" ury="707"/>
+ <char name="four" width="500" llx="-15" lly="0" urx="503" ury="683"/>
+ <char name="fraction" width="167" llx="-169" lly="-14" urx="324" ury="683"/>
+ <char name="g" width="500" llx="-52" lly="-203" urx="478" ury="462"/>
+ <char name="germandbls" width="500" llx="-200" lly="-200" urx="473" ury="705"/>
+ <char name="grave" width="333" llx="85" lly="516" urx="297" ury="697"/>
+ <char name="greater" width="570" llx="31" lly="-8" urx="539" ury="514"/>
+ <char name="guillemotleft" width="500" llx="12" lly="32" urx="468" ury="415"/>
+ <char name="guillemotright" width="500" llx="12" lly="32" urx="468" ury="415"/>
+ <char name="guilsinglleft" width="333" llx="32" lly="32" urx="303" ury="415"/>
+ <char name="guilsinglright" width="333" llx="10" lly="32" urx="281" ury="415"/>
+ <char name="h" width="556" llx="-13" lly="-9" urx="498" ury="699"/>
+ <char name="hungarumlaut" width="333" llx="69" lly="516" urx="498" ury="697"/>
+ <char name="hyphen" width="333" llx="2" lly="166" urx="271" ury="282"/>
+ <char name="i" width="278" llx="2" lly="-9" urx="263" ury="684"/>
+ <char name="iacute" width="278" llx="2" lly="-9" urx="352" ury="697"/>
+ <char name="icircumflex" width="278" llx="-3" lly="-9" urx="324" ury="690"/>
+ <char name="idieresis" width="278" llx="2" lly="-9" urx="364" ury="655"/>
+ <char name="igrave" width="278" llx="2" lly="-9" urx="259" ury="697"/>
+ <char name="j" width="278" llx="-189" lly="-207" urx="279" ury="684"/>
+ <char name="k" width="500" llx="-23" lly="-8" urx="483" ury="699"/>
+ <char name="l" width="278" llx="2" lly="-9" urx="290" ury="699"/>
+ <char name="less" width="570" llx="31" lly="-8" urx="539" ury="514"/>
+ <char name="logicalnot" width="606" llx="51" lly="108" urx="555" ury="399"/>
+ <char name="lslash" width="278" llx="-7" lly="-9" urx="307" ury="699"/>
+ <char name="m" width="778" llx="-14" lly="-9" urx="722" ury="462"/>
+ <char name="macron" width="333" llx="51" lly="553" urx="393" ury="623"/>
+ <char name="minus" width="330" llx="51" lly="209" urx="555" ury="297"/>
+ <char name="mu" width="576" llx="-60" lly="-207" urx="516" ury="449"/>
+ <char name="multiply" width="570" llx="48" lly="16" urx="522" ury="490"/>
+ <char name="n" width="556" llx="-6" lly="-9" urx="493" ury="462"/>
+ <char name="nine" width="500" llx="-12" lly="-10" urx="475" ury="683"/>
+ <char name="ntilde" width="556" llx="-6" lly="-9" urx="504" ury="655"/>
+ <char name="numbersign" width="500" llx="-33" lly="0" urx="533" ury="700"/>
+ <char name="o" width="500" llx="-3" lly="-13" urx="441" ury="462"/>
+ <char name="oacute" width="500" llx="-3" lly="-13" urx="463" ury="697"/>
+ <char name="ocircumflex" width="500" llx="-3" lly="-13" urx="451" ury="690"/>
+ <char name="odieresis" width="500" llx="-3" lly="-13" urx="471" ury="655"/>
+ <char name="oe" width="722" llx="6" lly="-13" urx="674" ury="462"/>
+ <char name="ogonek" width="333" llx="15" lly="-183" urx="244" ury="34"/>
+ <char name="ograve" width="500" llx="-3" lly="-13" urx="441" ury="697"/>
+ <char name="one" width="500" llx="5" lly="0" urx="419" ury="683"/>
+ <char name="onehalf" width="750" llx="-9" lly="-14" urx="723" ury="683"/>
+ <char name="onequarter" width="750" llx="7" lly="-14" urx="721" ury="683"/>
+ <char name="onesuperior" width="300" llx="30" lly="274" urx="301" ury="683"/>
+ <char name="ordfeminine" width="266" llx="16" lly="399" urx="330" ury="685"/>
+ <char name="ordmasculine" width="300" llx="56" lly="400" urx="347" ury="685"/>
+ <char name="oslash" width="500" llx="-3" lly="-119" urx="441" ury="560"/>
+ <char name="otilde" width="500" llx="-3" lly="-13" urx="491" ury="655"/>
+ <char name="p" width="500" llx="-120" lly="-205" urx="446" ury="462"/>
+ <char name="paragraph" width="500" llx="-57" lly="-193" urx="562" ury="669"/>
+ <char name="parenleft" width="333" llx="28" lly="-179" urx="344" ury="685"/>
+ <char name="parenright" width="333" llx="-44" lly="-179" urx="271" ury="685"/>
+ <char name="percent" width="833" llx="39" lly="-10" urx="793" ury="692"/>
+ <char name="period" width="250" llx="-9" lly="-13" urx="139" ury="135"/>
+ <char name="periodcentered" width="250" llx="51" lly="257" urx="199" ury="405"/>
+ <char name="perthousand" width="1000" llx="7" lly="-29" urx="996" ury="706"/>
+ <char name="plus" width="570" llx="33" lly="0" urx="537" ury="506"/>
+ <char name="plusminus" width="570" llx="33" lly="0" urx="537" ury="506"/>
+ <char name="q" width="500" llx="1" lly="-205" urx="471" ury="462"/>
+ <char name="question" width="500" llx="79" lly="-13" urx="470" ury="684"/>
+ <char name="questiondown" width="500" llx="30" lly="-205" urx="421" ury="492"/>
+ <char name="quotedbl" width="555" llx="136" lly="398" urx="536" ury="685"/>
+ <char name="quotedblbase" width="500" llx="-57" lly="-182" urx="403" ury="134"/>
+ <char name="quotedblleft" width="500" llx="53" lly="369" urx="513" ury="685"/>
+ <char name="quotedblright" width="500" llx="53" lly="369" urx="513" ury="685"/>
+ <char name="quoteleft" width="333" llx="128" lly="369" urx="332" ury="685"/>
+ <char name="quoteright" width="333" llx="98" lly="369" urx="302" ury="685"/>
+ <char name="quotesinglbase" width="333" llx="-5" lly="-182" urx="199" ury="134"/>
+ <char name="quotesingle" width="278" llx="128" lly="398" urx="268" ury="685"/>
+ <char name="r" width="389" llx="-21" lly="0" urx="389" ury="462"/>
+ <char name="registered" width="747" llx="30" lly="-18" urx="718" ury="685"/>
+ <char name="ring" width="333" llx="127" lly="516" urx="340" ury="729"/>
+ <char name="s" width="389" llx="-19" lly="-13" urx="333" ury="462"/>
+ <char name="scaron" width="389" llx="-19" lly="-13" urx="424" ury="690"/>
+ <char name="section" width="500" llx="36" lly="-143" urx="459" ury="685"/>
+ <char name="semicolon" width="333" llx="-25" lly="-183" urx="264" ury="459"/>
+ <char name="seven" width="500" llx="52" lly="0" urx="525" ury="669"/>
+ <char name="six" width="500" llx="23" lly="-15" urx="509" ury="679"/>
+ <char name="slash" width="278" llx="-64" lly="-18" urx="342" ury="685"/>
+ <char name="space" width="250" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="250"/>
- <char name="sterling" width="500"/>
- <char name="t" width="278"/>
- <char name="thorn" width="500"/>
- <char name="three" width="500"/>
- <char name="threequarters" width="750"/>
- <char name="threesuperior" width="300"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="1000"/>
- <char name="two" width="500"/>
- <char name="twosuperior" width="300"/>
- <char name="u" width="556"/>
- <char name="uacute" width="556"/>
- <char name="ucircumflex" width="556"/>
- <char name="udieresis" width="556"/>
- <char name="ugrave" width="556"/>
- <char name="underscore" width="500"/>
- <char name="v" width="444"/>
- <char name="w" width="667"/>
- <char name="x" width="500"/>
- <char name="y" width="444"/>
- <char name="yacute" width="444"/>
- <char name="ydieresis" width="444"/>
- <char name="yen" width="500"/>
- <char name="z" width="389"/>
- <char name="zcaron" width="389"/>
- <char name="zero" width="500"/>
- </widths>
+ <char name="sterling" width="500" llx="-32" lly="-12" urx="510" ury="683"/>
+ <char name="t" width="278" llx="-11" lly="-9" urx="281" ury="594"/>
+ <char name="thorn" width="500" llx="-120" lly="-205" urx="446" ury="699"/>
+ <char name="three" width="500" llx="-15" lly="-13" urx="450" ury="683"/>
+ <char name="threequarters" width="750" llx="7" lly="-14" urx="726" ury="683"/>
+ <char name="threesuperior" width="300" llx="17" lly="265" urx="321" ury="683"/>
+ <char name="tilde" width="333" llx="48" lly="536" urx="407" ury="655"/>
+ <char name="trademark" width="1000" llx="32" lly="263" urx="968" ury="669"/>
+ <char name="two" width="500" llx="-27" lly="0" urx="446" ury="683"/>
+ <char name="twosuperior" width="300" llx="2" lly="274" urx="313" ury="683"/>
+ <char name="u" width="556" llx="15" lly="-9" urx="492" ury="462"/>
+ <char name="uacute" width="556" llx="15" lly="-9" urx="492" ury="697"/>
+ <char name="ucircumflex" width="556" llx="15" lly="-9" urx="492" ury="690"/>
+ <char name="udieresis" width="556" llx="15" lly="-9" urx="499" ury="655"/>
+ <char name="ugrave" width="556" llx="15" lly="-9" urx="492" ury="697"/>
+ <char name="underscore" width="500" llx="0" lly="-125" urx="500" ury="-75"/>
+ <char name="v" width="444" llx="16" lly="-13" urx="401" ury="462"/>
+ <char name="w" width="667" llx="16" lly="-13" urx="614" ury="462"/>
+ <char name="x" width="500" llx="-46" lly="-13" urx="469" ury="462"/>
+ <char name="y" width="444" llx="-94" lly="-205" urx="392" ury="462"/>
+ <char name="yacute" width="444" llx="-94" lly="-205" urx="435" ury="697"/>
+ <char name="ydieresis" width="444" llx="-94" lly="-205" urx="443" ury="655"/>
+ <char name="yen" width="500" llx="33" lly="0" urx="628" ury="669"/>
+ <char name="z" width="389" llx="-43" lly="-78" urx="368" ury="449"/>
+ <char name="zcaron" width="389" llx="-43" lly="-78" urx="424" ury="690"/>
+ <char name="zero" width="500" llx="17" lly="-14" urx="477" ury="683"/>
+ </char-metrics>
<kerning kpx1="79">
<pair kern="-40" kpx2="65"/>
<pair kern="-50" kpx2="87"/>
diff --git a/src/codegen/fonts/TimesItalic.xml b/src/codegen/fonts/TimesItalic.xml
index 514aaa6dd..607f4d701 100644
--- a/src/codegen/fonts/TimesItalic.xml
+++ b/src/codegen/fonts/TimesItalic.xml
@@ -22,245 +22,246 @@
<family-name>Times</family-name>
<class-name>TimesItalic</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>653</cap-height>
<x-height>441</x-height>
<ascender>683</ascender>
<descender>-205</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="611"/>
- <char name="AE" width="889"/>
- <char name="Aacute" width="611"/>
- <char name="Acircumflex" width="611"/>
- <char name="Adieresis" width="611"/>
- <char name="Agrave" width="611"/>
- <char name="Aring" width="611"/>
- <char name="Atilde" width="611"/>
- <char name="B" width="611"/>
- <char name="C" width="667"/>
- <char name="Ccedilla" width="667"/>
- <char name="D" width="722"/>
- <char name="E" width="611"/>
- <char name="Eacute" width="611"/>
- <char name="Ecircumflex" width="611"/>
- <char name="Edieresis" width="611"/>
- <char name="Egrave" width="611"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="500"/>
- <char name="F" width="611"/>
- <char name="G" width="722"/>
- <char name="H" width="722"/>
- <char name="I" width="333"/>
- <char name="Iacute" width="333"/>
- <char name="Icircumflex" width="333"/>
- <char name="Idieresis" width="333"/>
- <char name="Igrave" width="333"/>
- <char name="J" width="444"/>
- <char name="K" width="667"/>
- <char name="L" width="556"/>
- <char name="Lslash" width="556"/>
- <char name="M" width="833"/>
- <char name="N" width="667"/>
- <char name="Ntilde" width="667"/>
- <char name="O" width="722"/>
- <char name="OE" width="944"/>
- <char name="Oacute" width="722"/>
- <char name="Ocircumflex" width="722"/>
- <char name="Odieresis" width="722"/>
- <char name="Ograve" width="722"/>
- <char name="Oslash" width="722"/>
- <char name="Otilde" width="722"/>
- <char name="P" width="611"/>
- <char name="Q" width="722"/>
- <char name="R" width="611"/>
- <char name="S" width="500"/>
- <char name="Scaron" width="500"/>
- <char name="T" width="556"/>
- <char name="Thorn" width="611"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="611"/>
- <char name="W" width="833"/>
- <char name="X" width="611"/>
- <char name="Y" width="556"/>
- <char name="Yacute" width="556"/>
- <char name="Ydieresis" width="556"/>
- <char name="Z" width="556"/>
- <char name="Zcaron" width="556"/>
- <char name="a" width="500"/>
- <char name="aacute" width="500"/>
- <char name="acircumflex" width="500"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="500"/>
- <char name="ae" width="667"/>
- <char name="agrave" width="500"/>
- <char name="ampersand" width="778"/>
- <char name="aring" width="500"/>
- <char name="asciicircum" width="422"/>
- <char name="asciitilde" width="541"/>
- <char name="asterisk" width="500"/>
- <char name="at" width="920"/>
- <char name="atilde" width="500"/>
- <char name="b" width="500"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="275"/>
- <char name="braceleft" width="400"/>
- <char name="braceright" width="400"/>
- <char name="bracketleft" width="389"/>
- <char name="bracketright" width="389"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="275"/>
- <char name="bullet" width="350"/>
- <char name="c" width="444"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="444"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="500"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="333"/>
- <char name="comma" width="250"/>
- <char name="copyright" width="760"/>
- <char name="currency" width="500"/>
- <char name="d" width="500"/>
- <char name="dagger" width="500"/>
- <char name="daggerdbl" width="500"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="675"/>
- <char name="dollar" width="500"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="444"/>
- <char name="eacute" width="444"/>
- <char name="ecircumflex" width="444"/>
- <char name="edieresis" width="444"/>
- <char name="egrave" width="444"/>
- <char name="eight" width="500"/>
- <char name="ellipsis" width="889"/>
- <char name="emdash" width="889"/>
- <char name="endash" width="500"/>
- <char name="equal" width="675"/>
- <char name="eth" width="500"/>
- <char name="exclam" width="333"/>
- <char name="exclamdown" width="389"/>
- <char name="f" width="278"/>
- <char name="fi" width="500"/>
- <char name="five" width="500"/>
- <char name="fl" width="500"/>
- <char name="florin" width="500"/>
- <char name="four" width="500"/>
- <char name="fraction" width="167"/>
- <char name="g" width="500"/>
- <char name="germandbls" width="500"/>
- <char name="grave" width="333"/>
- <char name="greater" width="675"/>
- <char name="guillemotleft" width="500"/>
- <char name="guillemotright" width="500"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="500"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="278"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="278"/>
- <char name="k" width="444"/>
- <char name="l" width="278"/>
- <char name="less" width="675"/>
- <char name="logicalnot" width="675"/>
- <char name="lslash" width="278"/>
- <char name="m" width="722"/>
- <char name="macron" width="333"/>
- <char name="minus" width="675"/>
- <char name="mu" width="500"/>
- <char name="multiply" width="675"/>
- <char name="n" width="500"/>
- <char name="nine" width="500"/>
- <char name="ntilde" width="500"/>
- <char name="numbersign" width="500"/>
- <char name="o" width="500"/>
- <char name="oacute" width="500"/>
- <char name="ocircumflex" width="500"/>
- <char name="odieresis" width="500"/>
- <char name="oe" width="667"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="500"/>
- <char name="one" width="500"/>
- <char name="onehalf" width="750"/>
- <char name="onequarter" width="750"/>
- <char name="onesuperior" width="300"/>
- <char name="ordfeminine" width="276"/>
- <char name="ordmasculine" width="310"/>
- <char name="oslash" width="500"/>
- <char name="otilde" width="500"/>
- <char name="p" width="500"/>
- <char name="paragraph" width="523"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="833"/>
- <char name="period" width="250"/>
- <char name="periodcentered" width="250"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="675"/>
- <char name="plusminus" width="675"/>
- <char name="q" width="500"/>
- <char name="question" width="500"/>
- <char name="questiondown" width="500"/>
- <char name="quotedbl" width="420"/>
- <char name="quotedblbase" width="556"/>
- <char name="quotedblleft" width="556"/>
- <char name="quotedblright" width="556"/>
- <char name="quoteleft" width="333"/>
- <char name="quoteright" width="333"/>
- <char name="quotesinglbase" width="333"/>
- <char name="quotesingle" width="214"/>
- <char name="r" width="389"/>
- <char name="registered" width="760"/>
- <char name="ring" width="333"/>
- <char name="s" width="389"/>
- <char name="scaron" width="389"/>
- <char name="section" width="500"/>
- <char name="semicolon" width="333"/>
- <char name="seven" width="500"/>
- <char name="six" width="500"/>
- <char name="slash" width="278"/>
- <char name="space" width="250"/>
+ <char-metrics>
+ <char name="A" width="611" llx="-51" lly="0" urx="564" ury="668"/>
+ <char name="AE" width="889" llx="-27" lly="0" urx="911" ury="653"/>
+ <char name="Aacute" width="611" llx="-51" lly="0" urx="564" ury="876"/>
+ <char name="Acircumflex" width="611" llx="-51" lly="0" urx="564" ury="873"/>
+ <char name="Adieresis" width="611" llx="-51" lly="0" urx="564" ury="818"/>
+ <char name="Agrave" width="611" llx="-51" lly="0" urx="564" ury="876"/>
+ <char name="Aring" width="611" llx="-51" lly="0" urx="564" ury="883"/>
+ <char name="Atilde" width="611" llx="-51" lly="0" urx="566" ury="836"/>
+ <char name="B" width="611" llx="-8" lly="0" urx="588" ury="653"/>
+ <char name="C" width="667" llx="66" lly="-18" urx="689" ury="666"/>
+ <char name="Ccedilla" width="667" llx="66" lly="-217" urx="689" ury="666"/>
+ <char name="D" width="722" llx="-8" lly="0" urx="700" ury="653"/>
+ <char name="E" width="611" llx="-1" lly="0" urx="634" ury="653"/>
+ <char name="Eacute" width="611" llx="-1" lly="0" urx="634" ury="876"/>
+ <char name="Ecircumflex" width="611" llx="-1" lly="0" urx="634" ury="873"/>
+ <char name="Edieresis" width="611" llx="-1" lly="0" urx="634" ury="818"/>
+ <char name="Egrave" width="611" llx="-1" lly="0" urx="634" ury="876"/>
+ <char name="Eth" width="722" llx="-8" lly="0" urx="700" ury="653"/>
+ <char name="Euro" width="500" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="611" llx="8" lly="0" urx="645" ury="653"/>
+ <char name="G" width="722" llx="52" lly="-18" urx="722" ury="666"/>
+ <char name="H" width="722" llx="-8" lly="0" urx="767" ury="653"/>
+ <char name="I" width="333" llx="-8" lly="0" urx="384" ury="653"/>
+ <char name="Iacute" width="333" llx="-8" lly="0" urx="433" ury="876"/>
+ <char name="Icircumflex" width="333" llx="-8" lly="0" urx="425" ury="873"/>
+ <char name="Idieresis" width="333" llx="-8" lly="0" urx="435" ury="818"/>
+ <char name="Igrave" width="333" llx="-8" lly="0" urx="384" ury="876"/>
+ <char name="J" width="444" llx="-6" lly="-18" urx="491" ury="653"/>
+ <char name="K" width="667" llx="7" lly="0" urx="722" ury="653"/>
+ <char name="L" width="556" llx="-8" lly="0" urx="559" ury="653"/>
+ <char name="Lslash" width="556" llx="-8" lly="0" urx="559" ury="653"/>
+ <char name="M" width="833" llx="-18" lly="0" urx="873" ury="653"/>
+ <char name="N" width="667" llx="-20" lly="-15" urx="727" ury="653"/>
+ <char name="Ntilde" width="667" llx="-20" lly="-15" urx="727" ury="836"/>
+ <char name="O" width="722" llx="60" lly="-18" urx="699" ury="666"/>
+ <char name="OE" width="944" llx="49" lly="-8" urx="964" ury="666"/>
+ <char name="Oacute" width="722" llx="60" lly="-18" urx="699" ury="876"/>
+ <char name="Ocircumflex" width="722" llx="60" lly="-18" urx="699" ury="873"/>
+ <char name="Odieresis" width="722" llx="60" lly="-18" urx="699" ury="818"/>
+ <char name="Ograve" width="722" llx="60" lly="-18" urx="699" ury="876"/>
+ <char name="Oslash" width="722" llx="60" lly="-105" urx="699" ury="722"/>
+ <char name="Otilde" width="722" llx="60" lly="-18" urx="699" ury="836"/>
+ <char name="P" width="611" llx="0" lly="0" urx="605" ury="653"/>
+ <char name="Q" width="722" llx="59" lly="-182" urx="699" ury="666"/>
+ <char name="R" width="611" llx="-13" lly="0" urx="588" ury="653"/>
+ <char name="S" width="500" llx="17" lly="-18" urx="508" ury="667"/>
+ <char name="Scaron" width="500" llx="17" lly="-18" urx="520" ury="873"/>
+ <char name="T" width="556" llx="59" lly="0" urx="633" ury="653"/>
+ <char name="Thorn" width="611" llx="0" lly="0" urx="569" ury="653"/>
+ <char name="U" width="722" llx="102" lly="-18" urx="765" ury="653"/>
+ <char name="Uacute" width="722" llx="102" lly="-18" urx="765" ury="876"/>
+ <char name="Ucircumflex" width="722" llx="102" lly="-18" urx="765" ury="873"/>
+ <char name="Udieresis" width="722" llx="102" lly="-18" urx="765" ury="818"/>
+ <char name="Ugrave" width="722" llx="102" lly="-18" urx="765" ury="876"/>
+ <char name="V" width="611" llx="76" lly="-18" urx="688" ury="653"/>
+ <char name="W" width="833" llx="71" lly="-18" urx="906" ury="653"/>
+ <char name="X" width="611" llx="-29" lly="0" urx="655" ury="653"/>
+ <char name="Y" width="556" llx="78" lly="0" urx="633" ury="653"/>
+ <char name="Yacute" width="556" llx="78" lly="0" urx="633" ury="876"/>
+ <char name="Ydieresis" width="556" llx="78" lly="0" urx="633" ury="818"/>
+ <char name="Z" width="556" llx="-6" lly="0" urx="606" ury="653"/>
+ <char name="Zcaron" width="556" llx="-6" lly="0" urx="606" ury="873"/>
+ <char name="a" width="500" llx="17" lly="-11" urx="476" ury="441"/>
+ <char name="aacute" width="500" llx="17" lly="-11" urx="487" ury="664"/>
+ <char name="acircumflex" width="500" llx="17" lly="-11" urx="476" ury="661"/>
+ <char name="acute" width="333" llx="180" lly="494" urx="403" ury="664"/>
+ <char name="adieresis" width="500" llx="17" lly="-11" urx="489" ury="606"/>
+ <char name="ae" width="667" llx="23" lly="-11" urx="640" ury="441"/>
+ <char name="agrave" width="500" llx="17" lly="-11" urx="476" ury="664"/>
+ <char name="ampersand" width="778" llx="76" lly="-18" urx="723" ury="666"/>
+ <char name="aring" width="500" llx="17" lly="-11" urx="476" ury="691"/>
+ <char name="asciicircum" width="422" llx="0" lly="301" urx="422" ury="666"/>
+ <char name="asciitilde" width="541" llx="40" lly="183" urx="502" ury="323"/>
+ <char name="asterisk" width="500" llx="128" lly="255" urx="492" ury="666"/>
+ <char name="at" width="920" llx="118" lly="-18" urx="806" ury="666"/>
+ <char name="atilde" width="500" llx="17" lly="-11" urx="511" ury="624"/>
+ <char name="b" width="500" llx="23" lly="-11" urx="473" ury="683"/>
+ <char name="backslash" width="278" llx="-41" lly="-18" urx="319" ury="666"/>
+ <char name="bar" width="275" llx="105" lly="-217" urx="171" ury="783"/>
+ <char name="braceleft" width="400" llx="51" lly="-177" urx="407" ury="687"/>
+ <char name="braceright" width="400" llx="-7" lly="-177" urx="349" ury="687"/>
+ <char name="bracketleft" width="389" llx="21" lly="-153" urx="391" ury="663"/>
+ <char name="bracketright" width="389" llx="12" lly="-153" urx="382" ury="663"/>
+ <char name="breve" width="333" llx="117" lly="492" urx="418" ury="650"/>
+ <char name="brokenbar" width="275" llx="105" lly="-142" urx="171" ury="708"/>
+ <char name="bullet" width="350" llx="40" lly="191" urx="310" ury="461"/>
+ <char name="c" width="444" llx="30" lly="-11" urx="425" ury="441"/>
+ <char name="caron" width="333" llx="121" lly="492" urx="426" ury="661"/>
+ <char name="ccedilla" width="444" llx="30" lly="-217" urx="425" ury="441"/>
+ <char name="cedilla" width="333" llx="-30" lly="-217" urx="182" ury="0"/>
+ <char name="cent" width="500" llx="77" lly="-143" urx="472" ury="560"/>
+ <char name="circumflex" width="333" llx="91" lly="492" urx="385" ury="661"/>
+ <char name="colon" width="333" llx="50" lly="-11" urx="261" ury="441"/>
+ <char name="comma" width="250" llx="-4" lly="-129" urx="135" ury="101"/>
+ <char name="copyright" width="760" llx="41" lly="-18" urx="719" ury="666"/>
+ <char name="currency" width="500" llx="-22" lly="53" urx="522" ury="597"/>
+ <char name="d" width="500" llx="15" lly="-13" urx="527" ury="683"/>
+ <char name="dagger" width="500" llx="101" lly="-159" urx="488" ury="666"/>
+ <char name="daggerdbl" width="500" llx="22" lly="-143" urx="491" ury="666"/>
+ <char name="degree" width="400" llx="101" lly="390" urx="387" ury="676"/>
+ <char name="dieresis" width="333" llx="107" lly="548" urx="405" ury="646"/>
+ <char name="divide" width="675" llx="86" lly="-11" urx="590" ury="517"/>
+ <char name="dollar" width="500" llx="31" lly="-89" urx="497" ury="731"/>
+ <char name="dotaccent" width="333" llx="207" lly="548" urx="305" ury="646"/>
+ <char name="dotlessi" width="278" llx="49" lly="-11" urx="235" ury="441"/>
+ <char name="e" width="444" llx="31" lly="-11" urx="412" ury="441"/>
+ <char name="eacute" width="444" llx="31" lly="-11" urx="459" ury="664"/>
+ <char name="ecircumflex" width="444" llx="31" lly="-11" urx="441" ury="661"/>
+ <char name="edieresis" width="444" llx="31" lly="-11" urx="451" ury="606"/>
+ <char name="egrave" width="444" llx="31" lly="-11" urx="412" ury="664"/>
+ <char name="eight" width="500" llx="30" lly="-7" urx="493" ury="676"/>
+ <char name="ellipsis" width="889" llx="57" lly="-11" urx="762" ury="100"/>
+ <char name="emdash" width="889" llx="-6" lly="197" urx="894" ury="243"/>
+ <char name="endash" width="500" llx="-6" lly="197" urx="505" ury="243"/>
+ <char name="equal" width="675" llx="86" lly="120" urx="590" ury="386"/>
+ <char name="eth" width="500" llx="27" lly="-11" urx="482" ury="683"/>
+ <char name="exclam" width="333" llx="39" lly="-11" urx="302" ury="667"/>
+ <char name="exclamdown" width="389" llx="59" lly="-205" urx="322" ury="473"/>
+ <char name="f" width="278" llx="-147" lly="-207" urx="424" ury="678"/>
+ <char name="fi" width="500" llx="-141" lly="-207" urx="481" ury="681"/>
+ <char name="five" width="500" llx="15" lly="-7" urx="491" ury="666"/>
+ <char name="fl" width="500" llx="-141" lly="-204" urx="518" ury="682"/>
+ <char name="florin" width="500" llx="25" lly="-182" urx="507" ury="682"/>
+ <char name="four" width="500" llx="1" lly="0" urx="479" ury="676"/>
+ <char name="fraction" width="167" llx="-169" lly="-10" urx="337" ury="676"/>
+ <char name="g" width="500" llx="8" lly="-206" urx="472" ury="441"/>
+ <char name="germandbls" width="500" llx="-168" lly="-207" urx="493" ury="679"/>
+ <char name="grave" width="333" llx="121" lly="492" urx="311" ury="664"/>
+ <char name="greater" width="675" llx="84" lly="-8" urx="592" ury="514"/>
+ <char name="guillemotleft" width="500" llx="53" lly="37" urx="445" ury="403"/>
+ <char name="guillemotright" width="500" llx="55" lly="37" urx="447" ury="403"/>
+ <char name="guilsinglleft" width="333" llx="51" lly="37" urx="281" ury="403"/>
+ <char name="guilsinglright" width="333" llx="52" lly="37" urx="282" ury="403"/>
+ <char name="h" width="500" llx="19" lly="-9" urx="478" ury="683"/>
+ <char name="hungarumlaut" width="333" llx="93" lly="494" urx="486" ury="664"/>
+ <char name="hyphen" width="333" llx="49" lly="192" urx="282" ury="255"/>
+ <char name="i" width="278" llx="49" lly="-11" urx="264" ury="654"/>
+ <char name="iacute" width="278" llx="49" lly="-11" urx="355" ury="664"/>
+ <char name="icircumflex" width="278" llx="33" lly="-11" urx="327" ury="661"/>
+ <char name="idieresis" width="278" llx="49" lly="-11" urx="352" ury="606"/>
+ <char name="igrave" width="278" llx="49" lly="-11" urx="284" ury="664"/>
+ <char name="j" width="278" llx="-124" lly="-207" urx="276" ury="654"/>
+ <char name="k" width="444" llx="14" lly="-11" urx="461" ury="683"/>
+ <char name="l" width="278" llx="41" lly="-11" urx="279" ury="683"/>
+ <char name="less" width="675" llx="84" lly="-8" urx="592" ury="514"/>
+ <char name="logicalnot" width="675" llx="86" lly="108" urx="590" ury="386"/>
+ <char name="lslash" width="278" llx="41" lly="-11" urx="312" ury="683"/>
+ <char name="m" width="722" llx="12" lly="-9" urx="704" ury="441"/>
+ <char name="macron" width="333" llx="99" lly="532" urx="411" ury="583"/>
+ <char name="minus" width="675" llx="86" lly="220" urx="590" ury="286"/>
+ <char name="mu" width="500" llx="-30" lly="-209" urx="497" ury="428"/>
+ <char name="multiply" width="675" llx="93" lly="8" urx="582" ury="497"/>
+ <char name="n" width="500" llx="14" lly="-9" urx="474" ury="441"/>
+ <char name="nine" width="500" llx="23" lly="-17" urx="492" ury="676"/>
+ <char name="ntilde" width="500" llx="14" lly="-9" urx="476" ury="624"/>
+ <char name="numbersign" width="500" llx="2" lly="0" urx="540" ury="676"/>
+ <char name="o" width="500" llx="27" lly="-11" urx="468" ury="441"/>
+ <char name="oacute" width="500" llx="27" lly="-11" urx="487" ury="664"/>
+ <char name="ocircumflex" width="500" llx="27" lly="-11" urx="468" ury="661"/>
+ <char name="odieresis" width="500" llx="27" lly="-11" urx="489" ury="606"/>
+ <char name="oe" width="667" llx="20" lly="-12" urx="646" ury="441"/>
+ <char name="ogonek" width="333" llx="20" lly="-169" urx="203" ury="40"/>
+ <char name="ograve" width="500" llx="27" lly="-11" urx="468" ury="664"/>
+ <char name="one" width="500" llx="49" lly="0" urx="409" ury="676"/>
+ <char name="onehalf" width="750" llx="34" lly="-10" urx="749" ury="676"/>
+ <char name="onequarter" width="750" llx="33" lly="-10" urx="736" ury="676"/>
+ <char name="onesuperior" width="300" llx="43" lly="271" urx="284" ury="676"/>
+ <char name="ordfeminine" width="276" llx="42" lly="406" urx="352" ury="676"/>
+ <char name="ordmasculine" width="310" llx="67" lly="406" urx="362" ury="676"/>
+ <char name="oslash" width="500" llx="28" lly="-135" urx="469" ury="554"/>
+ <char name="otilde" width="500" llx="27" lly="-11" urx="496" ury="624"/>
+ <char name="p" width="500" llx="-75" lly="-205" urx="469" ury="441"/>
+ <char name="paragraph" width="523" llx="55" lly="-123" urx="616" ury="653"/>
+ <char name="parenleft" width="333" llx="42" lly="-181" urx="315" ury="669"/>
+ <char name="parenright" width="333" llx="16" lly="-180" urx="289" ury="669"/>
+ <char name="percent" width="833" llx="79" lly="-13" urx="790" ury="676"/>
+ <char name="period" width="250" llx="27" lly="-11" urx="138" ury="100"/>
+ <char name="periodcentered" width="250" llx="70" lly="199" urx="181" ury="310"/>
+ <char name="perthousand" width="1000" llx="25" lly="-19" urx="1010" ury="706"/>
+ <char name="plus" width="675" llx="86" lly="0" urx="590" ury="506"/>
+ <char name="plusminus" width="675" llx="86" lly="0" urx="590" ury="506"/>
+ <char name="q" width="500" llx="25" lly="-209" urx="483" ury="441"/>
+ <char name="question" width="500" llx="132" lly="-12" urx="472" ury="664"/>
+ <char name="questiondown" width="500" llx="28" lly="-205" urx="368" ury="471"/>
+ <char name="quotedbl" width="420" llx="144" lly="421" urx="432" ury="666"/>
+ <char name="quotedblbase" width="556" llx="57" lly="-129" urx="405" ury="101"/>
+ <char name="quotedblleft" width="556" llx="166" lly="436" urx="514" ury="666"/>
+ <char name="quotedblright" width="556" llx="151" lly="436" urx="499" ury="666"/>
+ <char name="quoteleft" width="333" llx="171" lly="436" urx="310" ury="666"/>
+ <char name="quoteright" width="333" llx="151" lly="436" urx="290" ury="666"/>
+ <char name="quotesinglbase" width="333" llx="44" lly="-129" urx="183" ury="101"/>
+ <char name="quotesingle" width="214" llx="132" lly="421" urx="241" ury="666"/>
+ <char name="r" width="389" llx="45" lly="0" urx="412" ury="441"/>
+ <char name="registered" width="760" llx="41" lly="-18" urx="719" ury="666"/>
+ <char name="ring" width="333" llx="155" lly="492" urx="355" ury="691"/>
+ <char name="s" width="389" llx="16" lly="-13" urx="366" ury="442"/>
+ <char name="scaron" width="389" llx="16" lly="-13" urx="454" ury="661"/>
+ <char name="section" width="500" llx="53" lly="-162" urx="461" ury="666"/>
+ <char name="semicolon" width="333" llx="27" lly="-129" urx="261" ury="441"/>
+ <char name="seven" width="500" llx="75" lly="-8" urx="537" ury="666"/>
+ <char name="six" width="500" llx="30" lly="-7" urx="521" ury="686"/>
+ <char name="slash" width="278" llx="-65" lly="-18" urx="386" ury="666"/>
+ <char name="space" width="250" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="250"/>
- <char name="sterling" width="500"/>
- <char name="t" width="278"/>
- <char name="thorn" width="500"/>
- <char name="three" width="500"/>
- <char name="threequarters" width="750"/>
- <char name="threesuperior" width="300"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="980"/>
- <char name="two" width="500"/>
- <char name="twosuperior" width="300"/>
- <char name="u" width="500"/>
- <char name="uacute" width="500"/>
- <char name="ucircumflex" width="500"/>
- <char name="udieresis" width="500"/>
- <char name="ugrave" width="500"/>
- <char name="underscore" width="500"/>
- <char name="v" width="444"/>
- <char name="w" width="667"/>
- <char name="x" width="444"/>
- <char name="y" width="444"/>
- <char name="yacute" width="444"/>
- <char name="ydieresis" width="444"/>
- <char name="yen" width="500"/>
- <char name="z" width="389"/>
- <char name="zcaron" width="389"/>
- <char name="zero" width="500"/>
- </widths>
+ <char name="sterling" width="500" llx="10" lly="-6" urx="517" ury="670"/>
+ <char name="t" width="278" llx="37" lly="-11" urx="296" ury="546"/>
+ <char name="thorn" width="500" llx="-75" lly="-205" urx="469" ury="683"/>
+ <char name="three" width="500" llx="15" lly="-7" urx="465" ury="676"/>
+ <char name="threequarters" width="750" llx="23" lly="-10" urx="736" ury="676"/>
+ <char name="threesuperior" width="300" llx="43" lly="268" urx="339" ury="676"/>
+ <char name="tilde" width="333" llx="100" lly="517" urx="427" ury="624"/>
+ <char name="trademark" width="980" llx="30" lly="247" urx="957" ury="653"/>
+ <char name="two" width="500" llx="12" lly="0" urx="452" ury="676"/>
+ <char name="twosuperior" width="300" llx="33" lly="271" urx="324" ury="676"/>
+ <char name="u" width="500" llx="42" lly="-11" urx="475" ury="441"/>
+ <char name="uacute" width="500" llx="42" lly="-11" urx="477" ury="664"/>
+ <char name="ucircumflex" width="500" llx="42" lly="-11" urx="475" ury="661"/>
+ <char name="udieresis" width="500" llx="42" lly="-11" urx="479" ury="606"/>
+ <char name="ugrave" width="500" llx="42" lly="-11" urx="475" ury="664"/>
+ <char name="underscore" width="500" llx="0" lly="-125" urx="500" ury="-75"/>
+ <char name="v" width="444" llx="21" lly="-18" urx="426" ury="441"/>
+ <char name="w" width="667" llx="16" lly="-18" urx="648" ury="441"/>
+ <char name="x" width="444" llx="-27" lly="-11" urx="447" ury="441"/>
+ <char name="y" width="444" llx="-24" lly="-206" urx="426" ury="441"/>
+ <char name="yacute" width="444" llx="-24" lly="-206" urx="459" ury="664"/>
+ <char name="ydieresis" width="444" llx="-24" lly="-206" urx="441" ury="606"/>
+ <char name="yen" width="500" llx="27" lly="0" urx="603" ury="653"/>
+ <char name="z" width="389" llx="-2" lly="-81" urx="380" ury="428"/>
+ <char name="zcaron" width="389" llx="-2" lly="-81" urx="434" ury="661"/>
+ <char name="zero" width="500" llx="32" lly="-7" urx="497" ury="676"/>
+ </char-metrics>
<kerning kpx1="79">
<pair kern="-55" kpx2="65"/>
<pair kern="-50" kpx2="87"/>
@@ -641,4 +642,4 @@
<pair kern="0" kpx2="121"/>
<pair kern="-40" kpx2="118"/>
</kerning>
-</font-metrics> \ No newline at end of file
+</font-metrics>
diff --git a/src/codegen/fonts/TimesRoman.xml b/src/codegen/fonts/TimesRoman.xml
index 9ddadfddd..9e16b6a3b 100644
--- a/src/codegen/fonts/TimesRoman.xml
+++ b/src/codegen/fonts/TimesRoman.xml
@@ -22,245 +22,246 @@
<family-name>Times</family-name>
<class-name>TimesRoman</class-name>
<encoding>StandardEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>662</cap-height>
<x-height>450</x-height>
<ascender>683</ascender>
<descender>-217</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="A" width="722"/>
- <char name="AE" width="889"/>
- <char name="Aacute" width="722"/>
- <char name="Acircumflex" width="722"/>
- <char name="Adieresis" width="722"/>
- <char name="Agrave" width="722"/>
- <char name="Aring" width="722"/>
- <char name="Atilde" width="722"/>
- <char name="B" width="667"/>
- <char name="C" width="667"/>
- <char name="Ccedilla" width="667"/>
- <char name="D" width="722"/>
- <char name="E" width="611"/>
- <char name="Eacute" width="611"/>
- <char name="Ecircumflex" width="611"/>
- <char name="Edieresis" width="611"/>
- <char name="Egrave" width="611"/>
- <char name="Eth" width="722"/>
- <char name="Euro" width="500"/>
- <char name="F" width="556"/>
- <char name="G" width="722"/>
- <char name="H" width="722"/>
- <char name="I" width="333"/>
- <char name="Iacute" width="333"/>
- <char name="Icircumflex" width="333"/>
- <char name="Idieresis" width="333"/>
- <char name="Igrave" width="333"/>
- <char name="J" width="389"/>
- <char name="K" width="722"/>
- <char name="L" width="611"/>
- <char name="Lslash" width="611"/>
- <char name="M" width="889"/>
- <char name="N" width="722"/>
- <char name="Ntilde" width="722"/>
- <char name="O" width="722"/>
- <char name="OE" width="889"/>
- <char name="Oacute" width="722"/>
- <char name="Ocircumflex" width="722"/>
- <char name="Odieresis" width="722"/>
- <char name="Ograve" width="722"/>
- <char name="Oslash" width="722"/>
- <char name="Otilde" width="722"/>
- <char name="P" width="556"/>
- <char name="Q" width="722"/>
- <char name="R" width="667"/>
- <char name="S" width="556"/>
- <char name="Scaron" width="556"/>
- <char name="T" width="611"/>
- <char name="Thorn" width="556"/>
- <char name="U" width="722"/>
- <char name="Uacute" width="722"/>
- <char name="Ucircumflex" width="722"/>
- <char name="Udieresis" width="722"/>
- <char name="Ugrave" width="722"/>
- <char name="V" width="722"/>
- <char name="W" width="944"/>
- <char name="X" width="722"/>
- <char name="Y" width="722"/>
- <char name="Yacute" width="722"/>
- <char name="Ydieresis" width="722"/>
- <char name="Z" width="611"/>
- <char name="Zcaron" width="611"/>
- <char name="a" width="444"/>
- <char name="aacute" width="444"/>
- <char name="acircumflex" width="444"/>
- <char name="acute" width="333"/>
- <char name="adieresis" width="444"/>
- <char name="ae" width="667"/>
- <char name="agrave" width="444"/>
- <char name="ampersand" width="778"/>
- <char name="aring" width="444"/>
- <char name="asciicircum" width="469"/>
- <char name="asciitilde" width="541"/>
- <char name="asterisk" width="500"/>
- <char name="at" width="921"/>
- <char name="atilde" width="444"/>
- <char name="b" width="500"/>
- <char name="backslash" width="278"/>
- <char name="bar" width="200"/>
- <char name="braceleft" width="480"/>
- <char name="braceright" width="480"/>
- <char name="bracketleft" width="333"/>
- <char name="bracketright" width="333"/>
- <char name="breve" width="333"/>
- <char name="brokenbar" width="200"/>
- <char name="bullet" width="350"/>
- <char name="c" width="444"/>
- <char name="caron" width="333"/>
- <char name="ccedilla" width="444"/>
- <char name="cedilla" width="333"/>
- <char name="cent" width="500"/>
- <char name="circumflex" width="333"/>
- <char name="colon" width="278"/>
- <char name="comma" width="250"/>
- <char name="copyright" width="760"/>
- <char name="currency" width="500"/>
- <char name="d" width="500"/>
- <char name="dagger" width="500"/>
- <char name="daggerdbl" width="500"/>
- <char name="degree" width="400"/>
- <char name="dieresis" width="333"/>
- <char name="divide" width="564"/>
- <char name="dollar" width="500"/>
- <char name="dotaccent" width="333"/>
- <char name="dotlessi" width="278"/>
- <char name="e" width="444"/>
- <char name="eacute" width="444"/>
- <char name="ecircumflex" width="444"/>
- <char name="edieresis" width="444"/>
- <char name="egrave" width="444"/>
- <char name="eight" width="500"/>
- <char name="ellipsis" width="1000"/>
- <char name="emdash" width="1000"/>
- <char name="endash" width="500"/>
- <char name="equal" width="564"/>
- <char name="eth" width="500"/>
- <char name="exclam" width="333"/>
- <char name="exclamdown" width="333"/>
- <char name="f" width="333"/>
- <char name="fi" width="556"/>
- <char name="five" width="500"/>
- <char name="fl" width="556"/>
- <char name="florin" width="500"/>
- <char name="four" width="500"/>
- <char name="fraction" width="167"/>
- <char name="g" width="500"/>
- <char name="germandbls" width="500"/>
- <char name="grave" width="333"/>
- <char name="greater" width="564"/>
- <char name="guillemotleft" width="500"/>
- <char name="guillemotright" width="500"/>
- <char name="guilsinglleft" width="333"/>
- <char name="guilsinglright" width="333"/>
- <char name="h" width="500"/>
- <char name="hungarumlaut" width="333"/>
- <char name="hyphen" width="333"/>
- <char name="i" width="278"/>
- <char name="iacute" width="278"/>
- <char name="icircumflex" width="278"/>
- <char name="idieresis" width="278"/>
- <char name="igrave" width="278"/>
- <char name="j" width="278"/>
- <char name="k" width="500"/>
- <char name="l" width="278"/>
- <char name="less" width="564"/>
- <char name="logicalnot" width="564"/>
- <char name="lslash" width="278"/>
- <char name="m" width="778"/>
- <char name="macron" width="333"/>
- <char name="minus" width="324"/>
- <char name="mu" width="500"/>
- <char name="multiply" width="564"/>
- <char name="n" width="500"/>
- <char name="nine" width="500"/>
- <char name="ntilde" width="500"/>
- <char name="numbersign" width="500"/>
- <char name="o" width="500"/>
- <char name="oacute" width="500"/>
- <char name="ocircumflex" width="500"/>
- <char name="odieresis" width="500"/>
- <char name="oe" width="722"/>
- <char name="ogonek" width="333"/>
- <char name="ograve" width="500"/>
- <char name="one" width="500"/>
- <char name="onehalf" width="750"/>
- <char name="onequarter" width="750"/>
- <char name="onesuperior" width="300"/>
- <char name="ordfeminine" width="276"/>
- <char name="ordmasculine" width="310"/>
- <char name="oslash" width="500"/>
- <char name="otilde" width="500"/>
- <char name="p" width="500"/>
- <char name="paragraph" width="453"/>
- <char name="parenleft" width="333"/>
- <char name="parenright" width="333"/>
- <char name="percent" width="833"/>
- <char name="period" width="250"/>
- <char name="periodcentered" width="250"/>
- <char name="perthousand" width="1000"/>
- <char name="plus" width="564"/>
- <char name="plusminus" width="564"/>
- <char name="q" width="500"/>
- <char name="question" width="444"/>
- <char name="questiondown" width="444"/>
- <char name="quotedbl" width="408"/>
- <char name="quotedblbase" width="444"/>
- <char name="quotedblleft" width="444"/>
- <char name="quotedblright" width="444"/>
- <char name="quoteleft" width="333"/>
- <char name="quoteright" width="333"/>
- <char name="quotesinglbase" width="333"/>
- <char name="quotesingle" width="180"/>
- <char name="r" width="333"/>
- <char name="registered" width="760"/>
- <char name="ring" width="333"/>
- <char name="s" width="389"/>
- <char name="scaron" width="389"/>
- <char name="section" width="500"/>
- <char name="semicolon" width="278"/>
- <char name="seven" width="500"/>
- <char name="six" width="500"/>
- <char name="slash" width="278"/>
- <char name="space" width="250"/>
+ <char-metrics>
+ <char name="A" width="722" llx="15" lly="0" urx="706" ury="674"/>
+ <char name="AE" width="889" llx="0" lly="0" urx="863" ury="662"/>
+ <char name="Aacute" width="722" llx="15" lly="0" urx="706" ury="890"/>
+ <char name="Acircumflex" width="722" llx="15" lly="0" urx="706" ury="886"/>
+ <char name="Adieresis" width="722" llx="15" lly="0" urx="706" ury="835"/>
+ <char name="Agrave" width="722" llx="15" lly="0" urx="706" ury="890"/>
+ <char name="Aring" width="722" llx="15" lly="0" urx="706" ury="898"/>
+ <char name="Atilde" width="722" llx="15" lly="0" urx="706" ury="850"/>
+ <char name="B" width="667" llx="17" lly="0" urx="593" ury="662"/>
+ <char name="C" width="667" llx="28" lly="-14" urx="633" ury="676"/>
+ <char name="Ccedilla" width="667" llx="28" lly="-215" urx="633" ury="676"/>
+ <char name="D" width="722" llx="16" lly="0" urx="685" ury="662"/>
+ <char name="E" width="611" llx="12" lly="0" urx="597" ury="662"/>
+ <char name="Eacute" width="611" llx="12" lly="0" urx="597" ury="890"/>
+ <char name="Ecircumflex" width="611" llx="12" lly="0" urx="597" ury="886"/>
+ <char name="Edieresis" width="611" llx="12" lly="0" urx="597" ury="835"/>
+ <char name="Egrave" width="611" llx="12" lly="0" urx="597" ury="890"/>
+ <char name="Eth" width="722" llx="16" lly="0" urx="685" ury="662"/>
+ <char name="Euro" width="500" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="F" width="556" llx="12" lly="0" urx="546" ury="662"/>
+ <char name="G" width="722" llx="32" lly="-14" urx="709" ury="676"/>
+ <char name="H" width="722" llx="19" lly="0" urx="702" ury="662"/>
+ <char name="I" width="333" llx="18" lly="0" urx="315" ury="662"/>
+ <char name="Iacute" width="333" llx="18" lly="0" urx="317" ury="890"/>
+ <char name="Icircumflex" width="333" llx="11" lly="0" urx="322" ury="886"/>
+ <char name="Idieresis" width="333" llx="18" lly="0" urx="315" ury="835"/>
+ <char name="Igrave" width="333" llx="18" lly="0" urx="315" ury="890"/>
+ <char name="J" width="389" llx="10" lly="-14" urx="370" ury="662"/>
+ <char name="K" width="722" llx="34" lly="0" urx="723" ury="662"/>
+ <char name="L" width="611" llx="12" lly="0" urx="598" ury="662"/>
+ <char name="Lslash" width="611" llx="12" lly="0" urx="598" ury="662"/>
+ <char name="M" width="889" llx="12" lly="0" urx="863" ury="662"/>
+ <char name="N" width="722" llx="12" lly="-11" urx="707" ury="662"/>
+ <char name="Ntilde" width="722" llx="12" lly="-11" urx="707" ury="850"/>
+ <char name="O" width="722" llx="34" lly="-14" urx="688" ury="676"/>
+ <char name="OE" width="889" llx="30" lly="-6" urx="885" ury="668"/>
+ <char name="Oacute" width="722" llx="34" lly="-14" urx="688" ury="890"/>
+ <char name="Ocircumflex" width="722" llx="34" lly="-14" urx="688" ury="886"/>
+ <char name="Odieresis" width="722" llx="34" lly="-14" urx="688" ury="835"/>
+ <char name="Ograve" width="722" llx="34" lly="-14" urx="688" ury="890"/>
+ <char name="Oslash" width="722" llx="34" lly="-80" urx="688" ury="734"/>
+ <char name="Otilde" width="722" llx="34" lly="-14" urx="688" ury="850"/>
+ <char name="P" width="556" llx="16" lly="0" urx="542" ury="662"/>
+ <char name="Q" width="722" llx="34" lly="-178" urx="701" ury="676"/>
+ <char name="R" width="667" llx="17" lly="0" urx="659" ury="662"/>
+ <char name="S" width="556" llx="42" lly="-14" urx="491" ury="676"/>
+ <char name="Scaron" width="556" llx="42" lly="-14" urx="491" ury="886"/>
+ <char name="T" width="611" llx="17" lly="0" urx="593" ury="662"/>
+ <char name="Thorn" width="556" llx="16" lly="0" urx="542" ury="662"/>
+ <char name="U" width="722" llx="14" lly="-14" urx="705" ury="662"/>
+ <char name="Uacute" width="722" llx="14" lly="-14" urx="705" ury="890"/>
+ <char name="Ucircumflex" width="722" llx="14" lly="-14" urx="705" ury="886"/>
+ <char name="Udieresis" width="722" llx="14" lly="-14" urx="705" ury="835"/>
+ <char name="Ugrave" width="722" llx="14" lly="-14" urx="705" ury="890"/>
+ <char name="V" width="722" llx="16" lly="-11" urx="697" ury="662"/>
+ <char name="W" width="944" llx="5" lly="-11" urx="932" ury="662"/>
+ <char name="X" width="722" llx="10" lly="0" urx="704" ury="662"/>
+ <char name="Y" width="722" llx="22" lly="0" urx="703" ury="662"/>
+ <char name="Yacute" width="722" llx="22" lly="0" urx="703" ury="890"/>
+ <char name="Ydieresis" width="722" llx="22" lly="0" urx="703" ury="835"/>
+ <char name="Z" width="611" llx="9" lly="0" urx="597" ury="662"/>
+ <char name="Zcaron" width="611" llx="9" lly="0" urx="597" ury="886"/>
+ <char name="a" width="444" llx="37" lly="-10" urx="442" ury="460"/>
+ <char name="aacute" width="444" llx="37" lly="-10" urx="442" ury="678"/>
+ <char name="acircumflex" width="444" llx="37" lly="-10" urx="442" ury="674"/>
+ <char name="acute" width="333" llx="93" lly="507" urx="317" ury="678"/>
+ <char name="adieresis" width="444" llx="37" lly="-10" urx="442" ury="623"/>
+ <char name="ae" width="667" llx="38" lly="-10" urx="632" ury="460"/>
+ <char name="agrave" width="444" llx="37" lly="-10" urx="442" ury="678"/>
+ <char name="ampersand" width="778" llx="42" lly="-13" urx="750" ury="676"/>
+ <char name="aring" width="444" llx="37" lly="-10" urx="442" ury="711"/>
+ <char name="asciicircum" width="469" llx="24" lly="297" urx="446" ury="662"/>
+ <char name="asciitilde" width="541" llx="40" lly="183" urx="502" ury="323"/>
+ <char name="asterisk" width="500" llx="69" lly="265" urx="432" ury="676"/>
+ <char name="at" width="921" llx="116" lly="-14" urx="809" ury="676"/>
+ <char name="atilde" width="444" llx="37" lly="-10" urx="442" ury="638"/>
+ <char name="b" width="500" llx="3" lly="-10" urx="468" ury="683"/>
+ <char name="backslash" width="278" llx="-9" lly="-14" urx="287" ury="676"/>
+ <char name="bar" width="200" llx="67" lly="-218" urx="133" ury="782"/>
+ <char name="braceleft" width="480" llx="100" lly="-181" urx="350" ury="680"/>
+ <char name="braceright" width="480" llx="130" lly="-181" urx="380" ury="680"/>
+ <char name="bracketleft" width="333" llx="88" lly="-156" urx="299" ury="662"/>
+ <char name="bracketright" width="333" llx="34" lly="-156" urx="245" ury="662"/>
+ <char name="breve" width="333" llx="26" lly="507" urx="307" ury="664"/>
+ <char name="brokenbar" width="200" llx="67" lly="-143" urx="133" ury="707"/>
+ <char name="bullet" width="350" llx="40" lly="196" urx="310" ury="466"/>
+ <char name="c" width="444" llx="25" lly="-10" urx="412" ury="460"/>
+ <char name="caron" width="333" llx="11" lly="507" urx="322" ury="674"/>
+ <char name="ccedilla" width="444" llx="25" lly="-215" urx="412" ury="460"/>
+ <char name="cedilla" width="333" llx="52" lly="-215" urx="261" ury="0"/>
+ <char name="cent" width="500" llx="53" lly="-138" urx="448" ury="579"/>
+ <char name="circumflex" width="333" llx="11" lly="507" urx="322" ury="674"/>
+ <char name="colon" width="278" llx="81" lly="-11" urx="192" ury="459"/>
+ <char name="comma" width="250" llx="56" lly="-141" urx="195" ury="102"/>
+ <char name="copyright" width="760" llx="38" lly="-14" urx="722" ury="676"/>
+ <char name="currency" width="500" llx="-22" lly="58" urx="522" ury="602"/>
+ <char name="d" width="500" llx="27" lly="-10" urx="491" ury="683"/>
+ <char name="dagger" width="500" llx="59" lly="-149" urx="442" ury="676"/>
+ <char name="daggerdbl" width="500" llx="58" lly="-153" urx="442" ury="676"/>
+ <char name="degree" width="400" llx="57" lly="390" urx="343" ury="676"/>
+ <char name="dieresis" width="333" llx="18" lly="581" urx="315" ury="681"/>
+ <char name="divide" width="564" llx="30" lly="-10" urx="534" ury="516"/>
+ <char name="dollar" width="500" llx="44" lly="-87" urx="457" ury="727"/>
+ <char name="dotaccent" width="333" llx="118" lly="581" urx="216" ury="681"/>
+ <char name="dotlessi" width="278" llx="16" lly="0" urx="253" ury="460"/>
+ <char name="e" width="444" llx="25" lly="-10" urx="424" ury="460"/>
+ <char name="eacute" width="444" llx="25" lly="-10" urx="424" ury="678"/>
+ <char name="ecircumflex" width="444" llx="25" lly="-10" urx="424" ury="674"/>
+ <char name="edieresis" width="444" llx="25" lly="-10" urx="424" ury="623"/>
+ <char name="egrave" width="444" llx="25" lly="-10" urx="424" ury="678"/>
+ <char name="eight" width="500" llx="56" lly="-14" urx="445" ury="676"/>
+ <char name="ellipsis" width="1000" llx="111" lly="-11" urx="888" ury="100"/>
+ <char name="emdash" width="1000" llx="0" lly="201" urx="1000" ury="250"/>
+ <char name="endash" width="500" llx="0" lly="201" urx="500" ury="250"/>
+ <char name="equal" width="564" llx="30" lly="120" urx="534" ury="386"/>
+ <char name="eth" width="500" llx="29" lly="-10" urx="471" ury="686"/>
+ <char name="exclam" width="333" llx="130" lly="-9" urx="238" ury="676"/>
+ <char name="exclamdown" width="333" llx="97" lly="-218" urx="205" ury="467"/>
+ <char name="f" width="333" llx="20" lly="0" urx="383" ury="683"/>
+ <char name="fi" width="556" llx="31" lly="0" urx="521" ury="683"/>
+ <char name="five" width="500" llx="32" lly="-14" urx="438" ury="688"/>
+ <char name="fl" width="556" llx="32" lly="0" urx="521" ury="683"/>
+ <char name="florin" width="500" llx="7" lly="-189" urx="490" ury="676"/>
+ <char name="four" width="500" llx="12" lly="0" urx="472" ury="676"/>
+ <char name="fraction" width="167" llx="-168" lly="-14" urx="331" ury="676"/>
+ <char name="g" width="500" llx="28" lly="-218" urx="470" ury="460"/>
+ <char name="germandbls" width="500" llx="12" lly="-9" urx="468" ury="683"/>
+ <char name="grave" width="333" llx="19" lly="507" urx="242" ury="678"/>
+ <char name="greater" width="564" llx="28" lly="-8" urx="536" ury="514"/>
+ <char name="guillemotleft" width="500" llx="42" lly="33" urx="456" ury="416"/>
+ <char name="guillemotright" width="500" llx="44" lly="33" urx="458" ury="416"/>
+ <char name="guilsinglleft" width="333" llx="63" lly="33" urx="285" ury="416"/>
+ <char name="guilsinglright" width="333" llx="48" lly="33" urx="270" ury="416"/>
+ <char name="h" width="500" llx="9" lly="0" urx="487" ury="683"/>
+ <char name="hungarumlaut" width="333" llx="-3" lly="507" urx="377" ury="678"/>
+ <char name="hyphen" width="333" llx="39" lly="194" urx="285" ury="257"/>
+ <char name="i" width="278" llx="16" lly="0" urx="253" ury="683"/>
+ <char name="iacute" width="278" llx="16" lly="0" urx="290" ury="678"/>
+ <char name="icircumflex" width="278" llx="-16" lly="0" urx="295" ury="674"/>
+ <char name="idieresis" width="278" llx="-9" lly="0" urx="288" ury="623"/>
+ <char name="igrave" width="278" llx="-8" lly="0" urx="253" ury="678"/>
+ <char name="j" width="278" llx="-70" lly="-218" urx="194" ury="683"/>
+ <char name="k" width="500" llx="7" lly="0" urx="505" ury="683"/>
+ <char name="l" width="278" llx="19" lly="0" urx="257" ury="683"/>
+ <char name="less" width="564" llx="28" lly="-8" urx="536" ury="514"/>
+ <char name="logicalnot" width="564" llx="30" lly="108" urx="534" ury="386"/>
+ <char name="lslash" width="278" llx="19" lly="0" urx="259" ury="683"/>
+ <char name="m" width="778" llx="16" lly="0" urx="775" ury="460"/>
+ <char name="macron" width="333" llx="11" lly="547" urx="322" ury="601"/>
+ <char name="minus" width="324" llx="30" lly="220" urx="534" ury="286"/>
+ <char name="mu" width="500" llx="36" lly="-218" urx="512" ury="450"/>
+ <char name="multiply" width="564" llx="38" lly="8" urx="527" ury="497"/>
+ <char name="n" width="500" llx="16" lly="0" urx="485" ury="460"/>
+ <char name="nine" width="500" llx="30" lly="-22" urx="459" ury="676"/>
+ <char name="ntilde" width="500" llx="16" lly="0" urx="485" ury="638"/>
+ <char name="numbersign" width="500" llx="5" lly="0" urx="496" ury="662"/>
+ <char name="o" width="500" llx="29" lly="-10" urx="470" ury="460"/>
+ <char name="oacute" width="500" llx="29" lly="-10" urx="470" ury="678"/>
+ <char name="ocircumflex" width="500" llx="29" lly="-10" urx="470" ury="674"/>
+ <char name="odieresis" width="500" llx="29" lly="-10" urx="470" ury="623"/>
+ <char name="oe" width="722" llx="30" lly="-10" urx="690" ury="460"/>
+ <char name="ogonek" width="333" llx="62" lly="-165" urx="243" ury="0"/>
+ <char name="ograve" width="500" llx="29" lly="-10" urx="470" ury="678"/>
+ <char name="one" width="500" llx="111" lly="0" urx="394" ury="676"/>
+ <char name="onehalf" width="750" llx="31" lly="-14" urx="746" ury="676"/>
+ <char name="onequarter" width="750" llx="37" lly="-14" urx="718" ury="676"/>
+ <char name="onesuperior" width="300" llx="57" lly="270" urx="248" ury="676"/>
+ <char name="ordfeminine" width="276" llx="4" lly="394" urx="270" ury="676"/>
+ <char name="ordmasculine" width="310" llx="6" lly="394" urx="304" ury="676"/>
+ <char name="oslash" width="500" llx="29" lly="-112" urx="470" ury="551"/>
+ <char name="otilde" width="500" llx="29" lly="-10" urx="470" ury="638"/>
+ <char name="p" width="500" llx="5" lly="-217" urx="470" ury="460"/>
+ <char name="paragraph" width="453" llx="-22" lly="-154" urx="450" ury="662"/>
+ <char name="parenleft" width="333" llx="48" lly="-177" urx="304" ury="676"/>
+ <char name="parenright" width="333" llx="29" lly="-177" urx="285" ury="676"/>
+ <char name="percent" width="833" llx="61" lly="-13" urx="772" ury="676"/>
+ <char name="period" width="250" llx="70" lly="-11" urx="181" ury="100"/>
+ <char name="periodcentered" width="250" llx="70" lly="199" urx="181" ury="310"/>
+ <char name="perthousand" width="1000" llx="7" lly="-19" urx="994" ury="706"/>
+ <char name="plus" width="564" llx="30" lly="0" urx="534" ury="506"/>
+ <char name="plusminus" width="564" llx="30" lly="0" urx="534" ury="506"/>
+ <char name="q" width="500" llx="24" lly="-217" urx="488" ury="460"/>
+ <char name="question" width="444" llx="68" lly="-8" urx="414" ury="676"/>
+ <char name="questiondown" width="444" llx="30" lly="-218" urx="376" ury="466"/>
+ <char name="quotedbl" width="408" llx="77" lly="431" urx="331" ury="676"/>
+ <char name="quotedblbase" width="444" llx="45" lly="-141" urx="416" ury="102"/>
+ <char name="quotedblleft" width="444" llx="43" lly="433" urx="414" ury="676"/>
+ <char name="quotedblright" width="444" llx="30" lly="433" urx="401" ury="676"/>
+ <char name="quoteleft" width="333" llx="115" lly="433" urx="254" ury="676"/>
+ <char name="quoteright" width="333" llx="79" lly="433" urx="218" ury="676"/>
+ <char name="quotesinglbase" width="333" llx="79" lly="-141" urx="218" ury="102"/>
+ <char name="quotesingle" width="180" llx="48" lly="431" urx="133" ury="676"/>
+ <char name="r" width="333" llx="5" lly="0" urx="335" ury="460"/>
+ <char name="registered" width="760" llx="38" lly="-14" urx="722" ury="676"/>
+ <char name="ring" width="333" llx="67" lly="512" urx="266" ury="711"/>
+ <char name="s" width="389" llx="51" lly="-10" urx="348" ury="460"/>
+ <char name="scaron" width="389" llx="39" lly="-10" urx="350" ury="674"/>
+ <char name="section" width="500" llx="70" lly="-148" urx="426" ury="676"/>
+ <char name="semicolon" width="278" llx="80" lly="-141" urx="219" ury="459"/>
+ <char name="seven" width="500" llx="20" lly="-8" urx="449" ury="662"/>
+ <char name="six" width="500" llx="34" lly="-14" urx="468" ury="684"/>
+ <char name="slash" width="278" llx="-9" lly="-14" urx="287" ury="676"/>
+ <char name="space" width="250" llx="0" lly="0" urx="0" ury="0"/>
<!-- JKT: the following has been manually added -->
- <char name="nbsp" width="250"/>
- <char name="sterling" width="500"/>
- <char name="t" width="278"/>
- <char name="thorn" width="500"/>
- <char name="three" width="500"/>
- <char name="threequarters" width="750"/>
- <char name="threesuperior" width="300"/>
- <char name="tilde" width="333"/>
- <char name="trademark" width="980"/>
- <char name="two" width="500"/>
- <char name="twosuperior" width="300"/>
- <char name="u" width="500"/>
- <char name="uacute" width="500"/>
- <char name="ucircumflex" width="500"/>
- <char name="udieresis" width="500"/>
- <char name="ugrave" width="500"/>
- <char name="underscore" width="500"/>
- <char name="v" width="500"/>
- <char name="w" width="722"/>
- <char name="x" width="500"/>
- <char name="y" width="500"/>
- <char name="yacute" width="500"/>
- <char name="ydieresis" width="500"/>
- <char name="yen" width="500"/>
- <char name="z" width="444"/>
- <char name="zcaron" width="444"/>
- <char name="zero" width="500"/>
- </widths>
+ <char name="sterling" width="500" llx="12" lly="-8" urx="490" ury="676"/>
+ <char name="t" width="278" llx="13" lly="-10" urx="279" ury="579"/>
+ <char name="thorn" width="500" llx="5" lly="-217" urx="470" ury="683"/>
+ <char name="three" width="500" llx="43" lly="-14" urx="431" ury="676"/>
+ <char name="threequarters" width="750" llx="15" lly="-14" urx="718" ury="676"/>
+ <char name="threesuperior" width="300" llx="15" lly="262" urx="291" ury="676"/>
+ <char name="tilde" width="333" llx="1" lly="532" urx="331" ury="638"/>
+ <char name="trademark" width="980" llx="30" lly="256" urx="957" ury="662"/>
+ <char name="two" width="500" llx="30" lly="0" urx="475" ury="676"/>
+ <char name="twosuperior" width="300" llx="1" lly="270" urx="296" ury="676"/>
+ <char name="u" width="500" llx="9" lly="-10" urx="479" ury="450"/>
+ <char name="uacute" width="500" llx="9" lly="-10" urx="479" ury="678"/>
+ <char name="ucircumflex" width="500" llx="9" lly="-10" urx="479" ury="674"/>
+ <char name="udieresis" width="500" llx="9" lly="-10" urx="479" ury="623"/>
+ <char name="ugrave" width="500" llx="9" lly="-10" urx="479" ury="678"/>
+ <char name="underscore" width="500" llx="0" lly="-125" urx="500" ury="-75"/>
+ <char name="v" width="500" llx="19" lly="-14" urx="477" ury="450"/>
+ <char name="w" width="722" llx="21" lly="-14" urx="694" ury="450"/>
+ <char name="x" width="500" llx="17" lly="0" urx="479" ury="450"/>
+ <char name="y" width="500" llx="14" lly="-218" urx="475" ury="450"/>
+ <char name="yacute" width="500" llx="14" lly="-218" urx="475" ury="678"/>
+ <char name="ydieresis" width="500" llx="14" lly="-218" urx="475" ury="623"/>
+ <char name="yen" width="500" llx="-53" lly="0" urx="512" ury="662"/>
+ <char name="z" width="444" llx="27" lly="0" urx="418" ury="450"/>
+ <char name="zcaron" width="444" llx="27" lly="0" urx="418" ury="674"/>
+ <char name="zero" width="500" llx="24" lly="-14" urx="476" ury="676"/>
+ </char-metrics>
<kerning kpx1="79">
<pair kern="-35" kpx2="65"/>
<pair kern="-35" kpx2="87"/>
diff --git a/src/codegen/fonts/ZapfDingbats.xml b/src/codegen/fonts/ZapfDingbats.xml
index f6ed76d25..bd72901dc 100644
--- a/src/codegen/fonts/ZapfDingbats.xml
+++ b/src/codegen/fonts/ZapfDingbats.xml
@@ -21,214 +21,216 @@
<family-name>ZapfDingbats</family-name>
<class-name>ZapfDingbats</class-name>
<encoding>ZapfDingbatsEncoding</encoding>
+ <underline-position>-100</underline-position>
+ <underline-thickness>50</underline-thickness>
<cap-height>820</cap-height>
<x-height>426</x-height>
<ascender>820</ascender>
<descender>-143</descender>
<first-char>32</first-char>
<last-char>255</last-char>
- <widths>
- <char name="space" width="278"/>
- <char name="a1" width="974"/>
- <char name="a2" width="961"/>
- <char name="a202" width="974"/>
- <char name="a3" width="980"/>
- <char name="a4" width="719"/>
- <char name="a5" width="789"/>
- <char name="a119" width="790"/>
- <char name="a118" width="791"/>
- <char name="a117" width="690"/>
- <char name="a11" width="960"/>
- <char name="a12" width="939"/>
- <char name="a13" width="549"/>
- <char name="a14" width="855"/>
- <char name="a15" width="911"/>
- <char name="a16" width="933"/>
- <char name="a105" width="911"/>
- <char name="a17" width="945"/>
- <char name="a18" width="974"/>
- <char name="a19" width="755"/>
- <char name="a20" width="846"/>
- <char name="a21" width="762"/>
- <char name="a22" width="761"/>
- <char name="a23" width="571"/>
- <char name="a24" width="677"/>
- <char name="a25" width="763"/>
- <char name="a26" width="760"/>
- <char name="a27" width="759"/>
- <char name="a28" width="754"/>
- <char name="a6" width="494"/>
- <char name="a7" width="552"/>
- <char name="a8" width="537"/>
- <char name="a9" width="577"/>
- <char name="a10" width="692"/>
- <char name="a29" width="786"/>
- <char name="a30" width="788"/>
- <char name="a31" width="788"/>
- <char name="a32" width="790"/>
- <char name="a33" width="793"/>
- <char name="a34" width="794"/>
- <char name="a35" width="816"/>
- <char name="a36" width="823"/>
- <char name="a37" width="789"/>
- <char name="a38" width="841"/>
- <char name="a39" width="823"/>
- <char name="a40" width="833"/>
- <char name="a41" width="816"/>
- <char name="a42" width="831"/>
- <char name="a43" width="923"/>
- <char name="a44" width="744"/>
- <char name="a45" width="723"/>
- <char name="a46" width="749"/>
- <char name="a47" width="790"/>
- <char name="a48" width="792"/>
- <char name="a49" width="695"/>
- <char name="a50" width="776"/>
- <char name="a51" width="768"/>
- <char name="a52" width="792"/>
- <char name="a53" width="759"/>
- <char name="a54" width="707"/>
- <char name="a55" width="708"/>
- <char name="a56" width="682"/>
- <char name="a57" width="701"/>
- <char name="a58" width="826"/>
- <char name="a59" width="815"/>
- <char name="a60" width="789"/>
- <char name="a61" width="789"/>
- <char name="a62" width="707"/>
- <char name="a63" width="687"/>
- <char name="a64" width="696"/>
- <char name="a65" width="689"/>
- <char name="a66" width="786"/>
- <char name="a67" width="787"/>
- <char name="a68" width="713"/>
- <char name="a69" width="791"/>
- <char name="a70" width="785"/>
- <char name="a71" width="791"/>
- <char name="a72" width="873"/>
- <char name="a73" width="761"/>
- <char name="a74" width="762"/>
- <char name="a203" width="762"/>
- <char name="a75" width="759"/>
- <char name="a204" width="759"/>
- <char name="a76" width="892"/>
- <char name="a77" width="892"/>
- <char name="a78" width="788"/>
- <char name="a79" width="784"/>
- <char name="a81" width="438"/>
- <char name="a82" width="138"/>
- <char name="a83" width="277"/>
- <char name="a84" width="415"/>
- <char name="a97" width="392"/>
- <char name="a98" width="392"/>
- <char name="a99" width="668"/>
- <char name="a100" width="668"/>
- <char name="a101" width="732"/>
- <char name="a102" width="544"/>
- <char name="a103" width="544"/>
- <char name="a104" width="910"/>
- <char name="a106" width="667"/>
- <char name="a107" width="760"/>
- <char name="a108" width="760"/>
- <char name="a112" width="776"/>
- <char name="a111" width="595"/>
- <char name="a110" width="694"/>
- <char name="a109" width="626"/>
- <char name="a120" width="788"/>
- <char name="a121" width="788"/>
- <char name="a122" width="788"/>
- <char name="a123" width="788"/>
- <char name="a124" width="788"/>
- <char name="a125" width="788"/>
- <char name="a126" width="788"/>
- <char name="a127" width="788"/>
- <char name="a128" width="788"/>
- <char name="a129" width="788"/>
- <char name="a130" width="788"/>
- <char name="a131" width="788"/>
- <char name="a132" width="788"/>
- <char name="a133" width="788"/>
- <char name="a134" width="788"/>
- <char name="a135" width="788"/>
- <char name="a136" width="788"/>
- <char name="a137" width="788"/>
- <char name="a138" width="788"/>
- <char name="a139" width="788"/>
- <char name="a140" width="788"/>
- <char name="a141" width="788"/>
- <char name="a142" width="788"/>
- <char name="a143" width="788"/>
- <char name="a144" width="788"/>
- <char name="a145" width="788"/>
- <char name="a146" width="788"/>
- <char name="a147" width="788"/>
- <char name="a148" width="788"/>
- <char name="a149" width="788"/>
- <char name="a150" width="788"/>
- <char name="a151" width="788"/>
- <char name="a152" width="788"/>
- <char name="a153" width="788"/>
- <char name="a154" width="788"/>
- <char name="a155" width="788"/>
- <char name="a156" width="788"/>
- <char name="a157" width="788"/>
- <char name="a158" width="788"/>
- <char name="a159" width="788"/>
- <char name="a160" width="894"/>
- <char name="a161" width="838"/>
- <char name="a163" width="1016"/>
- <char name="a164" width="458"/>
- <char name="a196" width="748"/>
- <char name="a165" width="924"/>
- <char name="a192" width="748"/>
- <char name="a166" width="918"/>
- <char name="a167" width="927"/>
- <char name="a168" width="928"/>
- <char name="a169" width="928"/>
- <char name="a170" width="834"/>
- <char name="a171" width="873"/>
- <char name="a172" width="828"/>
- <char name="a173" width="924"/>
- <char name="a162" width="924"/>
- <char name="a174" width="917"/>
- <char name="a175" width="930"/>
- <char name="a176" width="931"/>
- <char name="a177" width="463"/>
- <char name="a178" width="883"/>
- <char name="a179" width="836"/>
- <char name="a193" width="836"/>
- <char name="a180" width="867"/>
- <char name="a199" width="867"/>
- <char name="a181" width="696"/>
- <char name="a200" width="696"/>
- <char name="a182" width="874"/>
- <char name="a201" width="874"/>
- <char name="a183" width="760"/>
- <char name="a184" width="946"/>
- <char name="a197" width="771"/>
- <char name="a185" width="865"/>
- <char name="a194" width="771"/>
- <char name="a198" width="888"/>
- <char name="a186" width="967"/>
- <char name="a195" width="888"/>
- <char name="a187" width="831"/>
- <char name="a188" width="873"/>
- <char name="a189" width="927"/>
- <char name="a190" width="970"/>
- <char name="a191" width="918"/>
- <char name="a86" width="410"/>
- <char name="a85" width="509"/>
- <char name="a95" width="334"/>
- <char name="a205" width="509"/>
- <char name="a89" width="390"/>
- <char name="a87" width="234"/>
- <char name="a91" width="276"/>
- <char name="a90" width="390"/>
- <char name="a206" width="410"/>
- <char name="a94" width="317"/>
- <char name="a93" width="317"/>
- <char name="a92" width="276"/>
- <char name="a96" width="334"/>
- <char name="a88" width="234"/>
- </widths>
-</font-metrics> \ No newline at end of file
+ <char-metrics>
+ <char name="space" width="278" llx="0" lly="0" urx="0" ury="0"/>
+ <char name="a1" width="974" llx="35" lly="72" urx="939" ury="621"/>
+ <char name="a2" width="961" llx="35" lly="81" urx="927" ury="611"/>
+ <char name="a202" width="974" llx="35" lly="72" urx="939" ury="621"/>
+ <char name="a3" width="980" llx="35" lly="0" urx="945" ury="692"/>
+ <char name="a4" width="719" llx="34" lly="139" urx="685" ury="566"/>
+ <char name="a5" width="789" llx="35" lly="-14" urx="755" ury="705"/>
+ <char name="a119" width="790" llx="35" lly="-14" urx="755" ury="705"/>
+ <char name="a118" width="791" llx="35" lly="-13" urx="761" ury="705"/>
+ <char name="a117" width="690" llx="34" lly="138" urx="655" ury="553"/>
+ <char name="a11" width="960" llx="35" lly="123" urx="925" ury="568"/>
+ <char name="a12" width="939" llx="35" lly="134" urx="904" ury="559"/>
+ <char name="a13" width="549" llx="29" lly="-11" urx="516" ury="705"/>
+ <char name="a14" width="855" llx="34" lly="59" urx="820" ury="632"/>
+ <char name="a15" width="911" llx="35" lly="50" urx="876" ury="642"/>
+ <char name="a16" width="933" llx="35" lly="139" urx="899" ury="550"/>
+ <char name="a105" width="911" llx="35" lly="50" urx="876" ury="642"/>
+ <char name="a17" width="945" llx="35" lly="139" urx="909" ury="553"/>
+ <char name="a18" width="974" llx="35" lly="104" urx="938" ury="587"/>
+ <char name="a19" width="755" llx="34" lly="-13" urx="721" ury="705"/>
+ <char name="a20" width="846" llx="36" lly="-14" urx="811" ury="705"/>
+ <char name="a21" width="762" llx="35" lly="0" urx="727" ury="692"/>
+ <char name="a22" width="761" llx="35" lly="0" urx="727" ury="692"/>
+ <char name="a23" width="571" llx="-1" lly="-68" urx="571" ury="661"/>
+ <char name="a24" width="677" llx="36" lly="-13" urx="642" ury="705"/>
+ <char name="a25" width="763" llx="35" lly="0" urx="728" ury="692"/>
+ <char name="a26" width="760" llx="35" lly="0" urx="726" ury="692"/>
+ <char name="a27" width="759" llx="35" lly="0" urx="725" ury="692"/>
+ <char name="a28" width="754" llx="35" lly="0" urx="720" ury="692"/>
+ <char name="a6" width="494" llx="35" lly="0" urx="460" ury="692"/>
+ <char name="a7" width="552" llx="35" lly="0" urx="517" ury="692"/>
+ <char name="a8" width="537" llx="35" lly="0" urx="503" ury="692"/>
+ <char name="a9" width="577" llx="35" lly="96" urx="542" ury="596"/>
+ <char name="a10" width="692" llx="35" lly="-14" urx="657" ury="705"/>
+ <char name="a29" width="786" llx="35" lly="-14" urx="751" ury="705"/>
+ <char name="a30" width="788" llx="35" lly="-14" urx="752" ury="705"/>
+ <char name="a31" width="788" llx="35" lly="-14" urx="753" ury="705"/>
+ <char name="a32" width="790" llx="35" lly="-14" urx="756" ury="705"/>
+ <char name="a33" width="793" llx="35" lly="-13" urx="759" ury="705"/>
+ <char name="a34" width="794" llx="35" lly="-13" urx="759" ury="705"/>
+ <char name="a35" width="816" llx="35" lly="-14" urx="782" ury="705"/>
+ <char name="a36" width="823" llx="35" lly="-14" urx="787" ury="705"/>
+ <char name="a37" width="789" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a38" width="841" llx="35" lly="-14" urx="807" ury="705"/>
+ <char name="a39" width="823" llx="35" lly="-14" urx="789" ury="705"/>
+ <char name="a40" width="833" llx="35" lly="-14" urx="798" ury="705"/>
+ <char name="a41" width="816" llx="35" lly="-13" urx="782" ury="705"/>
+ <char name="a42" width="831" llx="35" lly="-14" urx="796" ury="705"/>
+ <char name="a43" width="923" llx="35" lly="-14" urx="888" ury="705"/>
+ <char name="a44" width="744" llx="35" lly="0" urx="710" ury="692"/>
+ <char name="a45" width="723" llx="35" lly="0" urx="688" ury="692"/>
+ <char name="a46" width="749" llx="35" lly="0" urx="714" ury="692"/>
+ <char name="a47" width="790" llx="34" lly="-14" urx="756" ury="705"/>
+ <char name="a48" width="792" llx="35" lly="-14" urx="758" ury="705"/>
+ <char name="a49" width="695" llx="35" lly="-14" urx="661" ury="706"/>
+ <char name="a50" width="776" llx="35" lly="-6" urx="741" ury="699"/>
+ <char name="a51" width="768" llx="35" lly="-7" urx="734" ury="699"/>
+ <char name="a52" width="792" llx="35" lly="-14" urx="757" ury="705"/>
+ <char name="a53" width="759" llx="35" lly="0" urx="725" ury="692"/>
+ <char name="a54" width="707" llx="35" lly="-13" urx="672" ury="704"/>
+ <char name="a55" width="708" llx="35" lly="-14" urx="672" ury="705"/>
+ <char name="a56" width="682" llx="35" lly="-14" urx="647" ury="705"/>
+ <char name="a57" width="701" llx="35" lly="-14" urx="666" ury="705"/>
+ <char name="a58" width="826" llx="35" lly="-14" urx="791" ury="705"/>
+ <char name="a59" width="815" llx="35" lly="-14" urx="780" ury="705"/>
+ <char name="a60" width="789" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a61" width="789" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a62" width="707" llx="34" lly="-14" urx="673" ury="705"/>
+ <char name="a63" width="687" llx="36" lly="0" urx="651" ury="692"/>
+ <char name="a64" width="696" llx="35" lly="0" urx="661" ury="691"/>
+ <char name="a65" width="689" llx="35" lly="0" urx="655" ury="692"/>
+ <char name="a66" width="786" llx="34" lly="-14" urx="751" ury="705"/>
+ <char name="a67" width="787" llx="35" lly="-14" urx="752" ury="705"/>
+ <char name="a68" width="713" llx="35" lly="-14" urx="678" ury="705"/>
+ <char name="a69" width="791" llx="35" lly="-14" urx="756" ury="705"/>
+ <char name="a70" width="785" llx="36" lly="-14" urx="751" ury="705"/>
+ <char name="a71" width="791" llx="35" lly="-14" urx="757" ury="705"/>
+ <char name="a72" width="873" llx="35" lly="-14" urx="838" ury="705"/>
+ <char name="a73" width="761" llx="35" lly="0" urx="726" ury="692"/>
+ <char name="a74" width="762" llx="35" lly="0" urx="727" ury="692"/>
+ <char name="a203" width="762" llx="35" lly="0" urx="727" ury="692"/>
+ <char name="a75" width="759" llx="35" lly="0" urx="725" ury="692"/>
+ <char name="a204" width="759" llx="35" lly="0" urx="725" ury="692"/>
+ <char name="a76" width="892" llx="35" lly="0" urx="858" ury="705"/>
+ <char name="a77" width="892" llx="35" lly="-14" urx="858" ury="692"/>
+ <char name="a78" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a79" width="784" llx="35" lly="-14" urx="749" ury="705"/>
+ <char name="a81" width="438" llx="35" lly="-14" urx="403" ury="705"/>
+ <char name="a82" width="138" llx="35" lly="0" urx="104" ury="692"/>
+ <char name="a83" width="277" llx="35" lly="0" urx="242" ury="692"/>
+ <char name="a84" width="415" llx="35" lly="0" urx="380" ury="692"/>
+ <char name="a97" width="392" llx="35" lly="263" urx="357" ury="705"/>
+ <char name="a98" width="392" llx="34" lly="263" urx="357" ury="705"/>
+ <char name="a99" width="668" llx="35" lly="263" urx="633" ury="705"/>
+ <char name="a100" width="668" llx="36" lly="263" urx="634" ury="705"/>
+ <char name="a101" width="732" llx="35" lly="-143" urx="697" ury="806"/>
+ <char name="a102" width="544" llx="56" lly="-14" urx="488" ury="706"/>
+ <char name="a103" width="544" llx="34" lly="-14" urx="508" ury="705"/>
+ <char name="a104" width="910" llx="35" lly="40" urx="875" ury="651"/>
+ <char name="a106" width="667" llx="35" lly="-14" urx="633" ury="705"/>
+ <char name="a107" width="760" llx="35" lly="-14" urx="726" ury="705"/>
+ <char name="a108" width="760" llx="0" lly="121" urx="758" ury="569"/>
+ <char name="a112" width="776" llx="35" lly="0" urx="741" ury="705"/>
+ <char name="a111" width="595" llx="34" lly="-14" urx="560" ury="705"/>
+ <char name="a110" width="694" llx="35" lly="-14" urx="659" ury="705"/>
+ <char name="a109" width="626" llx="34" lly="0" urx="591" ury="705"/>
+ <char name="a120" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a121" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a122" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a123" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a124" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a125" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a126" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a127" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a128" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a129" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a130" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a131" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a132" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a133" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a134" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a135" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a136" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a137" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a138" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a139" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a140" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a141" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a142" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a143" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a144" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a145" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a146" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a147" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a148" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a149" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a150" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a151" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a152" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a153" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a154" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a155" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a156" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a157" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a158" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a159" width="788" llx="35" lly="-14" urx="754" ury="705"/>
+ <char name="a160" width="894" llx="35" lly="58" urx="860" ury="634"/>
+ <char name="a161" width="838" llx="35" lly="152" urx="803" ury="540"/>
+ <char name="a163" width="1016" llx="34" lly="152" urx="981" ury="540"/>
+ <char name="a164" width="458" llx="35" lly="-127" urx="422" ury="820"/>
+ <char name="a196" width="748" llx="35" lly="94" urx="698" ury="597"/>
+ <char name="a165" width="924" llx="35" lly="140" urx="890" ury="552"/>
+ <char name="a192" width="748" llx="35" lly="94" urx="698" ury="597"/>
+ <char name="a166" width="918" llx="35" lly="166" urx="884" ury="526"/>
+ <char name="a167" width="927" llx="35" lly="32" urx="892" ury="660"/>
+ <char name="a168" width="928" llx="35" lly="129" urx="891" ury="562"/>
+ <char name="a169" width="928" llx="35" lly="128" urx="893" ury="563"/>
+ <char name="a170" width="834" llx="35" lly="155" urx="799" ury="537"/>
+ <char name="a171" width="873" llx="35" lly="93" urx="838" ury="599"/>
+ <char name="a172" width="828" llx="35" lly="104" urx="791" ury="588"/>
+ <char name="a173" width="924" llx="35" lly="98" urx="889" ury="594"/>
+ <char name="a162" width="924" llx="35" lly="98" urx="889" ury="594"/>
+ <char name="a174" width="917" llx="35" lly="0" urx="882" ury="692"/>
+ <char name="a175" width="930" llx="35" lly="84" urx="896" ury="608"/>
+ <char name="a176" width="931" llx="35" lly="84" urx="896" ury="608"/>
+ <char name="a177" width="463" llx="35" lly="-99" urx="429" ury="791"/>
+ <char name="a178" width="883" llx="35" lly="71" urx="848" ury="623"/>
+ <char name="a179" width="836" llx="35" lly="44" urx="802" ury="648"/>
+ <char name="a193" width="836" llx="35" lly="44" urx="802" ury="648"/>
+ <char name="a180" width="867" llx="35" lly="101" urx="832" ury="591"/>
+ <char name="a199" width="867" llx="35" lly="101" urx="832" ury="591"/>
+ <char name="a181" width="696" llx="35" lly="44" urx="661" ury="648"/>
+ <char name="a200" width="696" llx="35" lly="44" urx="661" ury="648"/>
+ <char name="a182" width="874" llx="35" lly="77" urx="840" ury="619"/>
+ <char name="a201" width="874" llx="35" lly="73" urx="840" ury="615"/>
+ <char name="a183" width="760" llx="35" lly="0" urx="725" ury="692"/>
+ <char name="a184" width="946" llx="35" lly="160" urx="911" ury="533"/>
+ <char name="a197" width="771" llx="34" lly="37" urx="736" ury="655"/>
+ <char name="a185" width="865" llx="35" lly="207" urx="830" ury="481"/>
+ <char name="a194" width="771" llx="34" lly="37" urx="736" ury="655"/>
+ <char name="a198" width="888" llx="34" lly="-19" urx="853" ury="712"/>
+ <char name="a186" width="967" llx="35" lly="124" urx="932" ury="568"/>
+ <char name="a195" width="888" llx="34" lly="-19" urx="853" ury="712"/>
+ <char name="a187" width="831" llx="35" lly="113" urx="796" ury="579"/>
+ <char name="a188" width="873" llx="36" lly="118" urx="838" ury="578"/>
+ <char name="a189" width="927" llx="35" lly="150" urx="891" ury="542"/>
+ <char name="a190" width="970" llx="35" lly="76" urx="931" ury="616"/>
+ <char name="a191" width="918" llx="34" lly="99" urx="884" ury="593"/>
+ <char name="a86" width="410" llx="35" lly="0" urx="375" ury="692"/>
+ <char name="a85" width="509" llx="35" lly="0" urx="475" ury="692"/>
+ <char name="a95" width="334" llx="35" lly="0" urx="299" ury="692"/>
+ <char name="a205" width="509" llx="35" lly="0" urx="475" ury="692"/>
+ <char name="a89" width="390" llx="35" lly="-14" urx="356" ury="705"/>
+ <char name="a87" width="234" llx="35" lly="-14" urx="199" ury="705"/>
+ <char name="a91" width="276" llx="35" lly="0" urx="242" ury="692"/>
+ <char name="a90" width="390" llx="35" lly="-14" urx="355" ury="705"/>
+ <char name="a206" width="410" llx="35" lly="0" urx="375" ury="692"/>
+ <char name="a94" width="317" llx="35" lly="0" urx="283" ury="692"/>
+ <char name="a93" width="317" llx="35" lly="0" urx="283" ury="692"/>
+ <char name="a92" width="276" llx="35" lly="0" urx="242" ury="692"/>
+ <char name="a96" width="334" llx="35" lly="0" urx="299" ury="692"/>
+ <char name="a88" width="234" llx="35" lly="-14" urx="199" ury="705"/>
+ </char-metrics>
+</font-metrics>
diff --git a/src/codegen/fonts/font-file.xsl b/src/codegen/fonts/font-file.xsl
index 85b968808..8723ed960 100644
--- a/src/codegen/fonts/font-file.xsl
+++ b/src/codegen/fonts/font-file.xsl
@@ -35,6 +35,7 @@
<xsl:template match="font-metrics">
package org.apache.fop.fonts.base14;
+import java.awt.Rectangle;
<xsl:if test="count(kerning) &gt; 0">
import java.util.Map;
</xsl:if>
@@ -42,12 +43,14 @@ import java.util.Set;
import org.apache.fop.fonts.FontType;
import org.apache.fop.fonts.Base14Font;
import org.apache.fop.fonts.CodePointMapping;
-import org.apache.fop.fonts.Typeface;;
+import org.apache.fop.fonts.Typeface;
public class <xsl:value-of select="class-name"/> extends Base14Font {
private final static String fontName = "<xsl:value-of select="font-name"/>";
private final static String fullName = "<xsl:value-of select="full-name"/>";
private final static Set familyNames;
+ private final static int underlinePosition = <xsl:value-of select="underline-position"/>;
+ private final static int underlineThickness = <xsl:value-of select="underline-thickness"/>;
private final static String encoding = "<xsl:value-of select="$encoding"/>";
private final static int capHeight = <xsl:value-of select="cap-height"/>;
private final static int xHeight = <xsl:value-of select="x-height"/>;
@@ -56,6 +59,7 @@ public class <xsl:value-of select="class-name"/> extends Base14Font {
private final static int firstChar = <xsl:value-of select="first-char"/>;
private final static int lastChar = <xsl:value-of select="last-char"/>;
private final static int[] width;
+ private final static Rectangle[] boundingBoxes;
private final CodePointMapping mapping =
CodePointMapping.getMapping("<xsl:value-of select="$encoding"/>");
<xsl:if test="count(kerning) &gt; 0">
@@ -66,7 +70,8 @@ public class <xsl:value-of select="class-name"/> extends Base14Font {
static {
width = new int[256];
- <xsl:apply-templates select="widths"/>
+ boundingBoxes = new Rectangle[256];
+ <xsl:apply-templates select="char-metrics"/>
<xsl:if test="count(kerning) &gt; 0">
kerning = new java.util.HashMap();
Integer first, second;
@@ -125,6 +130,14 @@ public class <xsl:value-of select="class-name"/> extends Base14Font {
return size * xHeight;
}
+ public int getUnderlinePosition(int size) {
+ return size * underlinePosition;
+ }
+
+ public int getUnderlineThickness(int size) {
+ return size * underlineThickness;
+ }
+
public int getFirstChar() {
return firstChar;
}
@@ -137,6 +150,11 @@ public class <xsl:value-of select="class-name"/> extends Base14Font {
return size * width[i];
}
+ public Rectangle getBoundingBox(int glyphIndex, int size) {
+ Rectangle bbox = boundingBoxes[glyphIndex];
+ return new Rectangle(bbox.x * size, bbox.y * size, bbox.width * size, bbox.height * size);
+ }
+
public int[] getWidths() {
int[] arr = new int[getLastChar() - getFirstChar() + 1];
System.arraycopy(width, getFirstChar(), arr, 0, getLastChar() - getFirstChar() + 1);
@@ -182,7 +200,9 @@ public class <xsl:value-of select="class-name"/> extends Base14Font {
}
</xsl:template>
- <xsl:template match="widths/char"><xsl:variable name="char-name" select="@name"/><xsl:variable name="char-num" select="$glyphs[@name = $char-name]/@codepoint"/><xsl:if test="$char-num!=''"> width[0x<xsl:value-of select="$char-num"/>] = <xsl:value-of select="@width"/>;</xsl:if></xsl:template>
+ <xsl:template match="char-metrics/char">
+ <xsl:variable name="char-name" select="@name"/><xsl:variable name="char-num" select="$glyphs[@name = $char-name]/@codepoint"/><xsl:if test="$char-num!=''"> width[0x<xsl:value-of select="$char-num"/>] = <xsl:value-of select="@width"/>;
+ boundingBoxes[0x<xsl:value-of select="$char-num"/>] = new Rectangle(<xsl:value-of select="@llx"/>,<xsl:value-of select="@lly"/>,<xsl:value-of select="@urx - @llx"/>,<xsl:value-of select="@ury - @lly"/>);</xsl:if></xsl:template>
<xsl:template match="kerning">
first = new Integer(<xsl:value-of select="@kpx1"/>);
diff --git a/src/java/org/apache/fop/afp/AFPEventProducer.java b/src/java/org/apache/fop/afp/AFPEventProducer.java
index 01d5c4ad7..1b43400c5 100644
--- a/src/java/org/apache/fop/afp/AFPEventProducer.java
+++ b/src/java/org/apache/fop/afp/AFPEventProducer.java
@@ -122,4 +122,12 @@ public interface AFPEventProducer extends EventProducer {
* @event.severity WARN
*/
void charactersetMissingMetrics(Object source, char character, String charSet);
+
+ /**
+ * Double-byte fonts are not currently supported in SVG.
+ * @param source the event source
+ * @param fontFamily name of DB font
+ * @event.severity WARN
+ */
+ void invalidDBFontInSVG(Object source, String fontFamily);
}
diff --git a/src/java/org/apache/fop/afp/fonts/AFPFont.java b/src/java/org/apache/fop/afp/fonts/AFPFont.java
index 06f484f37..99e15a46b 100644
--- a/src/java/org/apache/fop/afp/fonts/AFPFont.java
+++ b/src/java/org/apache/fop/afp/fonts/AFPFont.java
@@ -19,6 +19,7 @@
package org.apache.fop.afp.fonts;
+import java.awt.Rectangle;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
@@ -34,6 +35,8 @@ import org.apache.fop.fonts.Typeface;
*/
public abstract class AFPFont extends Typeface {
+ private static final double STRIKEOUT_POSITION_FACTOR = 0.45;
+
/** The font name */
protected final String name;
@@ -117,7 +120,34 @@ public abstract class AFPFont extends Typeface {
*/
protected static final char toUnicodeCodepoint(int character) {
//AFP fonts use Unicode directly as their mapped code points, so we can simply cast to char
- return (char)character;
+ return (char) character;
+ }
+
+ /** {@inheritDoc} */
+ public int getUnderlineThickness(int size) {
+ // This is the FOCA recommendation in the absence of the Underline Thickness parameter
+ return getBoundingBox('-', size).height;
+ }
+
+ /** {@inheritDoc} */
+ public int getStrikeoutPosition(int size) {
+ //TODO This conflicts with the FOCA recommendation of 0 in the absence of the Throughscore Position
+ // parameter
+ return (int) (STRIKEOUT_POSITION_FACTOR * getCapHeight(size));
+ }
+
+ /** {@inheritDoc} */
+ public int getStrikeoutThickness(int size) {
+ // This is the FOCA recommendation in the absence of the Throughscore Thickness parameter
+ return getBoundingBox('-', size).height;
+ }
+
+ /** {@inheritDoc} */
+ public abstract Rectangle getBoundingBox(int glyphIndex, int size);
+
+ /** {@inheritDoc} */
+ public int[] getWidths() {
+ throw new UnsupportedOperationException();
}
/** {@inheritDoc} */
diff --git a/src/java/org/apache/fop/afp/fonts/AbstractOutlineFont.java b/src/java/org/apache/fop/afp/fonts/AbstractOutlineFont.java
index 7b57a2b8c..edbdf5e95 100644
--- a/src/java/org/apache/fop/afp/fonts/AbstractOutlineFont.java
+++ b/src/java/org/apache/fop/afp/fonts/AbstractOutlineFont.java
@@ -71,22 +71,6 @@ public abstract class AbstractOutlineFont extends AFPFont {
}
/**
- * Get the first character in this font.
- * @return the first character in this font
- */
- public int getFirstChar() {
- return charSet.getFirstChar();
- }
-
- /**
- * Get the last character in this font.
- * @return the last character in this font
- */
- public int getLastChar() {
- return charSet.getLastChar();
- }
-
- /**
* The ascender is the part of a lowercase letter that extends above the
* "x-height" (the height of the letter "x"), such as "d", "t", or "h". Also
* used to denote the part of the letter extending above the x-height.
@@ -98,6 +82,17 @@ public abstract class AbstractOutlineFont extends AFPFont {
return charSet.getAscender() * size;
}
+ /** {@inheritDoc} */
+ public int getUnderlinePosition(int size) {
+ return charSet.getUnderscorePosition() * size;
+ }
+
+ @Override
+ public int getUnderlineThickness(int size) {
+ int underscoreWidth = charSet.getUnderscoreWidth();
+ return underscoreWidth == 0 ? super.getUnderlineThickness(size) : underscoreWidth * size;
+ }
+
/**
* Obtains the height of capital letters for the specified point size.
*
@@ -130,40 +125,7 @@ public abstract class AbstractOutlineFont extends AFPFont {
return charSet.getXHeight() * size;
}
- /**
- * Obtain the width of the character for the specified point size.
- * @param character the character
- * @param size the font size (in mpt)
- * @return the width of the character for the specified point size
- */
- public int getWidth(int character, int size) {
- return charSet.getWidth(toUnicodeCodepoint(character)) * size;
- }
- /**
- * Get the getWidth (in 1/1000ths of a point size) of all characters in this
- * character set.
- *
- * @param size the font size (in mpt)
- * @return the widths of all characters
- */
- public int[] getWidths(int size) {
- int[] widths = charSet.getWidths();
- for (int i = 0; i < widths.length; i++) {
- widths[i] = widths[i] * size;
- }
- return widths;
- }
-
- /**
- * Get the getWidth (in 1/1000ths of a point size) of all characters in this
- * character set.
- *
- * @return the widths of all characters
- */
- public int[] getWidths() {
- return getWidths(1000);
- }
/** {@inheritDoc} */
public boolean hasChar(char c) {
diff --git a/src/java/org/apache/fop/afp/fonts/CharacterSet.java b/src/java/org/apache/fop/afp/fonts/CharacterSet.java
index e0c3b9c9a..3df8ba4c4 100644
--- a/src/java/org/apache/fop/afp/fonts/CharacterSet.java
+++ b/src/java/org/apache/fop/afp/fonts/CharacterSet.java
@@ -19,10 +19,9 @@
package org.apache.fop.afp.fonts;
+import java.awt.Rectangle;
import java.io.UnsupportedEncodingException;
import java.nio.charset.CharacterCodingException;
-import java.util.HashMap;
-import java.util.Map;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
@@ -63,6 +62,8 @@ public class CharacterSet {
private static final int MAX_NAME_LEN = 8;
+ /** The current orientation (currently only 0 is supported by FOP) */
+ public static final int SUPPORTED_ORIENTATION = 0;
/** The code page to which the character set relates */
protected final String codePage;
@@ -79,11 +80,8 @@ public class CharacterSet {
/** The path to the installed fonts */
private final AFPResourceAccessor accessor;
- /** The current orientation (currently only 0 is supported by FOP) */
- private final String currentOrientation = "0";
-
/** The collection of objects for each orientation */
- private final Map<String, CharacterSetOrientation> characterSetOrientations;
+ private CharacterSetOrientation characterSetOrientation;
/** The nominal vertical size (in millipoints) for bitmap fonts. 0 for outline fonts. */
private int nominalVerticalSize;
@@ -116,8 +114,6 @@ public class CharacterSet {
this.encoding = encoding;
this.encoder = charsetType.getEncoder(encoding);
this.accessor = accessor;
-
- this.characterSetOrientations = new HashMap<String, CharacterSetOrientation>(4);
}
// right pad short names with space
@@ -131,7 +127,9 @@ public class CharacterSet {
* @param cso the metrics for the orientation
*/
public void addCharacterSetOrientation(CharacterSetOrientation cso) {
- characterSetOrientations.put(String.valueOf(cso.getOrientation()), cso);
+ if (cso.getOrientation() == SUPPORTED_ORIENTATION) {
+ characterSetOrientation = cso;
+ }
}
/**
@@ -165,11 +163,24 @@ public class CharacterSet {
* @return the ascender value in millipoints
*/
public int getAscender() {
-
return getCharacterSetOrientation().getAscender();
}
/**
+ * TODO
+ */
+ public int getUnderscoreWidth() {
+ return getCharacterSetOrientation().getUnderscoreWidth();
+ }
+
+ /**
+ * TODO
+ */
+ public int getUnderscorePosition() {
+ return getCharacterSetOrientation().getUnderscorePosition();
+ }
+
+ /**
* Cap height is the average height of the uppercase characters in
* a font. This value is specified by the designer of a font and is
* usually the height of the uppercase M.
@@ -177,7 +188,6 @@ public class CharacterSet {
* @return the cap height value in millipoints
*/
public int getCapHeight() {
-
return getCharacterSetOrientation().getCapHeight();
}
@@ -194,24 +204,6 @@ public class CharacterSet {
}
/**
- * Returns the first character in the character set
- *
- * @return the first character in the character set (Unicode codepoint)
- */
- public char getFirstChar() {
- return getCharacterSetOrientation().getFirstChar();
- }
-
- /**
- * Returns the last character in the character set
- *
- * @return the last character in the character set (Unicode codepoint)
- */
- public char getLastChar() {
- return getCharacterSetOrientation().getLastChar();
- }
-
- /**
* Returns the resource accessor to load the font resources with.
* @return the resource accessor to load the font resources with
*/
@@ -220,16 +212,6 @@ public class CharacterSet {
}
/**
- * Get the width (in 1/1000ths of a point size) of all characters
- *
- * @return the widths of all characters
- */
- public int[] getWidths() {
-
- return getCharacterSetOrientation().getWidths();
- }
-
- /**
* XHeight refers to the height of the lower case letters above the baseline.
*
* @return the typical height of characters
@@ -246,11 +228,13 @@ public class CharacterSet {
* @param character the Unicode character from which the width will be calculated
* @return the width of the character
*/
- public int getWidth(char character) {
- return getCharacterSetOrientation().getWidth(character);
+ public int getWidth(char character, int size) {
+ return getCharacterSetOrientation().getWidth(character, size);
}
-
+ public Rectangle getCharacterBox(char character, int size) {
+ return getCharacterSetOrientation().getCharacterBox(character, size);
+ }
/**
* Returns the AFP character set identifier
@@ -309,9 +293,7 @@ public class CharacterSet {
* @return characterSetOrentation The current orientation metrics.
*/
private CharacterSetOrientation getCharacterSetOrientation() {
- CharacterSetOrientation c
- = characterSetOrientations.get(currentOrientation);
- return c;
+ return characterSetOrientation;
}
/**
diff --git a/src/java/org/apache/fop/afp/fonts/CharacterSetBuilder.java b/src/java/org/apache/fop/afp/fonts/CharacterSetBuilder.java
index 2565942b5..a3b2ab8ec 100644
--- a/src/java/org/apache/fop/afp/fonts/CharacterSetBuilder.java
+++ b/src/java/org/apache/fop/afp/fonts/CharacterSetBuilder.java
@@ -19,6 +19,7 @@
package org.apache.fop.afp.fonts;
+import java.awt.Rectangle;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
@@ -292,16 +293,14 @@ public abstract class CharacterSetBuilder {
metricNormalizationFactor = 1000.0d * 72000.0d
/ fontDescriptor.getNominalFontSizeInMillipoints() / dpi;
}
-
+ ValueNormalizer normalizer = new ValueNormalizer(metricNormalizationFactor);
//process D3AC89 Font Position
- processFontPosition(structuredFieldReader, characterSetOrientations,
- metricNormalizationFactor);
-
+ processFontPosition(structuredFieldReader, characterSetOrientations, normalizer);
//process D38C89 Font Index (per orientation)
for (int i = 0; i < characterSetOrientations.length; i++) {
- processFontIndex(structuredFieldReader,
- characterSetOrientations[i], codePage, metricNormalizationFactor);
- characterSet.addCharacterSetOrientation(characterSetOrientations[i]);
+ CharacterSetOrientation characterSetOrientation = characterSetOrientations[i];
+ processFontIndex(structuredFieldReader, characterSetOrientation, codePage, normalizer);
+ characterSet.addCharacterSetOrientation(characterSetOrientation);
}
} else {
throw new IOException("Missing D3AE89 Font Control structured field.");
@@ -314,6 +313,19 @@ public abstract class CharacterSetBuilder {
return characterSet;
}
+ private static class ValueNormalizer {
+
+ private final double factor;
+
+ public ValueNormalizer(double factor) {
+ this.factor = factor;
+ }
+
+ public int normalize(int value) {
+ return (int) Math.round(value * factor);
+ }
+ }
+
/**
* Load the code page information from the appropriate file. The file name
* to load is determined by the code page name and the file extension 'CDP'.
@@ -475,7 +487,7 @@ public abstract class CharacterSetBuilder {
* @throws IOException if an I/O exception of some sort has occurred.
*/
private void processFontPosition(StructuredFieldReader structuredFieldReader,
- CharacterSetOrientation[] characterSetOrientations, double metricNormalizationFactor)
+ CharacterSetOrientation[] characterSetOrientations, ValueNormalizer normalizer)
throws IOException {
byte[] data = structuredFieldReader.getNext(FONT_POSITION_SF);
@@ -493,48 +505,34 @@ public abstract class CharacterSetBuilder {
if (position == 9) {
CharacterSetOrientation characterSetOrientation
= characterSetOrientations[characterSetOrientationIndex];
-
int xHeight = getSBIN(fpData, 2);
int capHeight = getSBIN(fpData, 4);
int ascHeight = getSBIN(fpData, 6);
int dscHeight = getSBIN(fpData, 8);
-
dscHeight = dscHeight * -1;
-
- characterSetOrientation.setXHeight(
- (int)Math.round(xHeight * metricNormalizationFactor));
- characterSetOrientation.setCapHeight(
- (int)Math.round(capHeight * metricNormalizationFactor));
- characterSetOrientation.setAscender(
- (int)Math.round(ascHeight * metricNormalizationFactor));
- characterSetOrientation.setDescender(
- (int)Math.round(dscHeight * metricNormalizationFactor));
+ int underscoreWidth = getUBIN(fpData, 17);
+ int underscorePosition = getSBIN(fpData, 20);
+ characterSetOrientation.setXHeight(normalizer.normalize(xHeight));
+ characterSetOrientation.setCapHeight(normalizer.normalize(capHeight));
+ characterSetOrientation.setAscender(normalizer.normalize(ascHeight));
+ characterSetOrientation.setDescender(normalizer.normalize(dscHeight));
+ characterSetOrientation.setUnderscoreWidth(normalizer.normalize(underscoreWidth));
+ characterSetOrientation.setUnderscorePosition(normalizer.normalize(underscorePosition));
}
} else if (position == 22) {
position = 0;
characterSetOrientationIndex++;
fpData[position] = data[index];
}
-
position++;
}
}
- /**
- * Process the font index details for the character set orientation.
- *
- * @param structuredFieldReader the structured field reader
- * @param cso the CharacterSetOrientation object to populate
- * @param codepage the map of code pages
- * @param metricNormalizationFactor factor to apply to the metrics to get normalized
- * font metric values
- * @throws IOException if an I/O exception of some sort has occurred.
- */
- private void processFontIndex(StructuredFieldReader structuredFieldReader,
- CharacterSetOrientation cso, Map<String, String> codepage,
- double metricNormalizationFactor)
- throws IOException {
+
+ private void processFontIndex(StructuredFieldReader structuredFieldReader, CharacterSetOrientation cso,
+ Map<String, String> codepage, ValueNormalizer normalizer)
+ throws IOException {
byte[] data = structuredFieldReader.getNext(FONT_INDEX_SF);
@@ -543,8 +541,6 @@ public abstract class CharacterSetBuilder {
byte[] gcgid = new byte[8];
byte[] fiData = new byte[20];
- char lowest = 255;
- char highest = 0;
String firstABCMismatch = null;
// Read data, ignoring bytes 0 - 2
@@ -569,13 +565,15 @@ public abstract class CharacterSetBuilder {
char cidx = idx.charAt(0);
int width = getUBIN(fiData, 0);
+ int ascendHt = getSBIN(fiData, 2);
+ int descendDp = getSBIN(fiData, 4);
int a = getSBIN(fiData, 10);
int b = getUBIN(fiData, 12);
int c = getSBIN(fiData, 14);
int abc = a + b + c;
int diff = Math.abs(abc - width);
if (diff != 0 && width != 0) {
- double diffPercent = 100 * diff / (double)width;
+ double diffPercent = 100 * diff / (double) width;
if (diffPercent > 2) {
if (LOG.isTraceEnabled()) {
LOG.trace(gcgiString + ": "
@@ -587,27 +585,16 @@ public abstract class CharacterSetBuilder {
}
}
}
-
- if (cidx < lowest) {
- lowest = cidx;
- }
-
- if (cidx > highest) {
- highest = cidx;
- }
-
- int normalizedWidth = (int)Math.round(width * metricNormalizationFactor);
-
- cso.setWidth(cidx, normalizedWidth);
-
+ int normalizedWidth = normalizer.normalize(width);
+ int x0 = normalizer.normalize(a);
+ int y0 = normalizer.normalize(-descendDp);
+ int dx = normalizer.normalize(b);
+ int dy = normalizer.normalize(ascendHt + descendDp);
+ cso.setCharacterMetrics(cidx, normalizedWidth, new Rectangle(x0, y0, dx, dy));
}
-
}
}
- cso.setFirstChar(lowest);
- cso.setLastChar(highest);
-
if (LOG.isDebugEnabled() && firstABCMismatch != null) {
//Debug level because it usually is no problem.
LOG.debug("Font has metrics inconsitencies where A+B+C doesn't equal the"
diff --git a/src/java/org/apache/fop/afp/fonts/CharacterSetOrientation.java b/src/java/org/apache/fop/afp/fonts/CharacterSetOrientation.java
index a730525d2..5fe524536 100644
--- a/src/java/org/apache/fop/afp/fonts/CharacterSetOrientation.java
+++ b/src/java/org/apache/fop/afp/fonts/CharacterSetOrientation.java
@@ -19,7 +19,7 @@
package org.apache.fop.afp.fonts;
-import java.util.Arrays;
+import java.awt.Rectangle;
/**
* The IBM Font Object Content Architecture (FOCA) supports presentation
@@ -60,23 +60,13 @@ public class CharacterSetOrientation {
/**
* The character widths in the character set (indexed using Unicode codepoints)
*/
- private int[] charsWidths;
+ private IntegerKeyStore<CharacterMetrics> characterMetrics;
/**
* The height of lowercase letters
*/
private int xHeight;
- /**
- * The first character (Unicode codepoint)
- */
- private char firstChar;
-
- /**
- * The last character (Unicode codepoint)
- */
- private char lastChar;
-
/** The character set orientation */
private final int orientation;
/** space increment */
@@ -86,6 +76,10 @@ public class CharacterSetOrientation {
/** Nominal Character Increment */
private final int nomCharIncrement;
+ private int underscoreWidth;
+
+ private int underscorePosition;
+
/**
* Constructor for the CharacterSetOrientation, the orientation is
* expressed as the degrees rotation (i.e 0, 90, 180, 270)
@@ -97,8 +91,7 @@ public class CharacterSetOrientation {
this.spaceIncrement = spaceIncrement;
this.emSpaceIncrement = emSpaceIncrement;
this.nomCharIncrement = nomCharIncrement;
- charsWidths = new int[256];
- Arrays.fill(charsWidths, -1);
+ this.characterMetrics = new IntegerKeyStore<CharacterMetrics>();
}
/**
@@ -138,19 +131,17 @@ public class CharacterSetOrientation {
}
/**
- * The first character in the character set
- * @return the first character (Unicode codepoint)
+ * TODO
*/
- public char getFirstChar() {
- return firstChar;
+ public int getUnderscoreWidth() {
+ return underscoreWidth;
}
/**
- * The last character in the character set
- * @return the last character (Unicode codepoint)
+ * TODO
*/
- public char getLastChar() {
- return lastChar;
+ public int getUnderscorePosition() {
+ return underscorePosition;
}
/**
@@ -162,17 +153,6 @@ public class CharacterSetOrientation {
}
/**
- * Get the width (in 1/1000ths of a point size) of all characters
- * in this character set.
- * @return the widths of all characters
- */
- public int[] getWidths() {
- int[] arr = new int[(getLastChar() - getFirstChar()) + 1];
- System.arraycopy(charsWidths, getFirstChar(), arr, 0, (getLastChar() - getFirstChar()) + 1);
- return arr;
- }
-
- /**
* XHeight refers to the height of the lower case letters above
* the baseline.
* @return heightX the typical height of characters
@@ -187,13 +167,38 @@ public class CharacterSetOrientation {
* @param character the Unicode character to evaluate
* @return the widths of the character
*/
- public int getWidth(char character) {
- if (character >= charsWidths.length) {
- throw new IllegalArgumentException("Invalid character: "
- + character + " (" + Integer.toString(character)
- + "), maximum is " + (charsWidths.length - 1));
+ public int getWidth(char character, int size) {
+ CharacterMetrics cm = getCharacterMetrics(character);
+ return cm == null ? -1 : size * cm.width;
+ }
+
+ private CharacterMetrics getCharacterMetrics(char character) {
+ return characterMetrics.get((int) character);
+ }
+
+ /**
+ * Get the character box (rectangle with dimensions in 1/1000ths of a point size) of the character
+ * identified by the parameter passed.
+ * @param character the Unicode character to evaluate
+ * @return the character box
+ */
+ public Rectangle getCharacterBox(char character, int size) {
+ CharacterMetrics cm = getCharacterMetrics(character);
+ return scale(cm == null ? getFallbackCharacterBox() : cm.characterBox, size);
+ }
+
+ private static Rectangle scale(Rectangle rectangle, int size) {
+ if (rectangle == null) {
+ return null;
+ } else {
+ return new Rectangle((int) (size * rectangle.getX()), (int) (size * rectangle.getY()),
+ (int) (size * rectangle.getWidth()), (int) (size * rectangle.getHeight()));
}
- return charsWidths[character];
+ }
+
+ private Rectangle getFallbackCharacterBox() {
+ // TODO replace with something sensible
+ return new Rectangle(0, 0, 0, 0);
}
/**
@@ -233,19 +238,19 @@ public class CharacterSetOrientation {
}
/**
- * The first character in the character set
- * @param firstChar the first character
+ * TODO
+ * @param underscoreWidth the underscore width value in millipoints
*/
- public void setFirstChar(char firstChar) {
- this.firstChar = firstChar;
+ public void setUnderscoreWidth(int underscoreWidth) {
+ this.underscoreWidth = underscoreWidth;
}
/**
- * The last character in the character set
- * @param lastChar the last character
+ * TODO
+ * @param underscorePosition the underscore position value in millipoints
*/
- public void setLastChar(char lastChar) {
- this.lastChar = lastChar;
+ public void setUnderscorePosition(int underscorePosition) {
+ this.underscorePosition = underscorePosition;
}
/**
@@ -254,17 +259,8 @@ public class CharacterSetOrientation {
* @param character the Unicode character for which the width is being set
* @param width the widths of the character
*/
- public void setWidth(char character, int width) {
- if (character >= charsWidths.length) {
- // Increase the size of the array if necessary
- // TODO Can we remove firstChar? surely firstChar==0 at this stage?
- int[] arr = new int[(character - firstChar) + 1];
- System.arraycopy(charsWidths, 0, arr, 0, charsWidths.length);
- Arrays.fill(arr, charsWidths.length, character - firstChar, -1);
- charsWidths = arr;
- }
- charsWidths[character] = width;
-
+ public void setCharacterMetrics(char character, int width, Rectangle characterBox) {
+ characterMetrics.put((int) character, new CharacterMetrics(width, characterBox));
}
/**
@@ -299,4 +295,16 @@ public class CharacterSetOrientation {
public int getNominalCharIncrement() {
return this.nomCharIncrement;
}
+
+ private static class CharacterMetrics {
+
+ public final int width;
+
+ public final Rectangle characterBox;
+
+ public CharacterMetrics(int width, Rectangle characterBox) {
+ this.width = width;
+ this.characterBox = characterBox;
+ }
+ }
}
diff --git a/src/java/org/apache/fop/afp/fonts/DoubleByteFont.java b/src/java/org/apache/fop/afp/fonts/DoubleByteFont.java
index 78da6ea82..5b9bf6101 100644
--- a/src/java/org/apache/fop/afp/fonts/DoubleByteFont.java
+++ b/src/java/org/apache/fop/afp/fonts/DoubleByteFont.java
@@ -19,6 +19,7 @@
package org.apache.fop.afp.fonts;
+import java.awt.Rectangle;
import java.lang.Character.UnicodeBlock;
import java.util.HashSet;
import java.util.Set;
@@ -68,7 +69,7 @@ public class DoubleByteFont extends AbstractOutlineFont {
public int getWidth(int character, int size) {
int charWidth;
try {
- charWidth = charSet.getWidth(toUnicodeCodepoint(character));
+ charWidth = charSet.getWidth(toUnicodeCodepoint(character), size);
} catch (IllegalArgumentException e) {
if (!charsProcessed.contains(character)) {
charsProcessed.add(character);
@@ -80,9 +81,9 @@ public class DoubleByteFont extends AbstractOutlineFont {
}
if (charWidth == -1) {
- charWidth = getDefaultCharacterWidth(character);
+ charWidth = getDefaultCharacterWidth(character) * size;
}
- return charWidth * size;
+ return charWidth;
}
private int getDefaultCharacterWidth(int character) {
@@ -94,6 +95,33 @@ public class DoubleByteFont extends AbstractOutlineFont {
}
}
+ @Override
+ public Rectangle getBoundingBox(int character, int size) {
+ Rectangle characterBox = getBoundingBoxOrNull(character, size);
+ if (characterBox == null) {
+ characterBox = getDefaultCharacterBox(character, size);
+ }
+ return characterBox;
+ }
+
+ private Rectangle getBoundingBoxOrNull(int character, int size) {
+ Rectangle characterBox = null;
+ try {
+ characterBox = charSet.getCharacterBox(toUnicodeCodepoint(character), size);
+ } catch (IllegalArgumentException e) {
+ if (!charsProcessed.contains(character)) {
+ charsProcessed.add(character);
+ getAFPEventProducer().charactersetMissingMetrics(this, (char) character,
+ charSet.getName().trim());
+ }
+ }
+ return characterBox;
+ }
+
+ private Rectangle getDefaultCharacterBox(int character, int size) {
+ return getBoundingBoxOrNull('-', size);
+ }
+
private int inferCharWidth(int character) {
//Is this character an ideograph?
diff --git a/src/java/org/apache/fop/afp/fonts/FopCharacterSet.java b/src/java/org/apache/fop/afp/fonts/FopCharacterSet.java
index 7c2b68506..b729a8995 100644
--- a/src/java/org/apache/fop/afp/fonts/FopCharacterSet.java
+++ b/src/java/org/apache/fop/afp/fonts/FopCharacterSet.java
@@ -19,6 +19,8 @@
package org.apache.fop.afp.fonts;
+import java.awt.Rectangle;
+
import org.apache.fop.afp.AFPEventProducer;
import org.apache.fop.afp.util.AFPResourceAccessor;
import org.apache.fop.fonts.Typeface;
@@ -84,45 +86,31 @@ public class FopCharacterSet extends CharacterSet {
}
/**
- * The first character in the character set
- * @return the first character
+ * XHeight refers to the height of the lower case letters above the baseline.
+ * @return the typical height of characters
*/
- public char getFirstChar() {
- return 0;
+ public int getXHeight() {
+ return charSet.getXHeight(1);
}
- /**
- * The last character in the character set
- * @return the last character
- */
- public char getLastChar() {
- return 0;
+ @Override
+ public int getWidth(char character, int size) {
+ return charSet.getWidth(character, size);
}
- /**
- * Get the width (in 1/1000ths of a point size) of all characters
- * @return the widths of all characters
- */
- public int[] getWidths() {
- return charSet.getWidths();
- }
+ @Override
+ public Rectangle getCharacterBox(char character, int size) {
+ return charSet.getBoundingBox(character, size);
+ };
- /**
- * XHeight refers to the height of the lower case letters above the baseline.
- * @return the typical height of characters
- */
- public int getXHeight() {
- return charSet.getXHeight(1);
+ @Override
+ public int getUnderscoreWidth() {
+ return charSet.getUnderlineThickness(1);
}
- /**
- * Get the width (in 1/1000ths of a point size) of the character
- * identified by the parameter passed.
- * @param character the character from which the width will be calculated
- * @return the width of the character
- */
- public int getWidth(char character) {
- return charSet.getWidth(character, 1);
+ @Override
+ public int getUnderscorePosition() {
+ return charSet.getUnderlinePosition(1);
}
/**
diff --git a/src/java/org/apache/fop/afp/fonts/IntegerKeyStore.java b/src/java/org/apache/fop/afp/fonts/IntegerKeyStore.java
new file mode 100644
index 000000000..7e73b5b9f
--- /dev/null
+++ b/src/java/org/apache/fop/afp/fonts/IntegerKeyStore.java
@@ -0,0 +1,71 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.afp.fonts;
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.Map;
+
+/**
+ * A simple compact data structure to model a sparse array
+ */
+class IntegerKeyStore<T> {
+
+ private static final int RANGE_BIT_SIZE = 8;
+
+ private static final int RANGE_SIZE = 1 << RANGE_BIT_SIZE;
+
+ private final Map<Integer, ArrayList<T>> arrays = new HashMap<Integer, ArrayList<T>>();
+
+ /**
+ *
+ * @param index a positive integer
+ * @param value value to store
+ */
+ public void put(Integer index, T value) {
+ if (index < 0) {
+ throw new IndexOutOfBoundsException();
+ }
+ int rangeKey = index >> RANGE_BIT_SIZE;
+ int rangeIndex = index % RANGE_SIZE;
+ ArrayList<T> range = arrays.get(rangeKey);
+ if (range == null) {
+ range = new ArrayList<T>(Collections.<T>nCopies(RANGE_SIZE, null));
+ arrays.put(rangeKey, range);
+ }
+ range.set(rangeIndex, value);
+ }
+
+ /**
+ *
+ * @param index a positive integer
+ * @return value the value associated with the index or null
+ */
+ public T get(Integer index) {
+ if (index < 0) {
+ throw new IndexOutOfBoundsException();
+ }
+ int rangeKey = index >> RANGE_BIT_SIZE;
+ int rangeIndex = index % RANGE_SIZE;
+ ArrayList<T> range = arrays.get(rangeKey);
+ return range == null ? null : range.get(rangeIndex);
+ }
+}
diff --git a/src/java/org/apache/fop/afp/fonts/OutlineFont.java b/src/java/org/apache/fop/afp/fonts/OutlineFont.java
index e9cdf5ba4..fc2332ce9 100644
--- a/src/java/org/apache/fop/afp/fonts/OutlineFont.java
+++ b/src/java/org/apache/fop/afp/fonts/OutlineFont.java
@@ -19,6 +19,8 @@
package org.apache.fop.afp.fonts;
+import java.awt.Rectangle;
+
import org.apache.fop.afp.AFPEventProducer;
/**
@@ -38,4 +40,18 @@ public class OutlineFont extends AbstractOutlineFont {
super(name, embeddable, charSet, eventProducer);
}
+ /**
+ * Obtain the width of the character for the specified point size.
+ * @param character the character
+ * @param size the font size (in mpt)
+ * @return the width of the character for the specified point size
+ */
+ public int getWidth(int character, int size) {
+ return charSet.getWidth(toUnicodeCodepoint(character), size);
+ }
+
+ @Override
+ public Rectangle getBoundingBox(int character, int size) {
+ return charSet.getCharacterBox(toUnicodeCodepoint(character), size);
+ }
}
diff --git a/src/java/org/apache/fop/afp/fonts/RasterFont.java b/src/java/org/apache/fop/afp/fonts/RasterFont.java
index 5c4c38dc5..1fd30e0ba 100644
--- a/src/java/org/apache/fop/afp/fonts/RasterFont.java
+++ b/src/java/org/apache/fop/afp/fonts/RasterFont.java
@@ -19,8 +19,8 @@
package org.apache.fop.afp.fonts;
+import java.awt.Rectangle;
import java.util.HashMap;
-import java.util.Iterator;
import java.util.Map;
import java.util.SortedMap;
import java.util.TreeMap;
@@ -135,46 +135,21 @@ public class RasterFont extends AFPFont {
}
- /**
- * Get the first character in this font.
- * @return the first character in this font.
- */
- public int getFirstChar() {
- Iterator<CharacterSet> it = charSets.values().iterator();
- if (it.hasNext()) {
- CharacterSet csm = it.next();
- return csm.getFirstChar();
- } else {
- String msg = "getFirstChar() - No character set found for font:" + getFontName();
- LOG.error(msg);
- throw new FontRuntimeException(msg);
- }
- }
-
- /**
- * Get the last character in this font.
- * @return the last character in this font.
- */
- public int getLastChar() {
-
- Iterator<CharacterSet> it = charSets.values().iterator();
- if (it.hasNext()) {
- CharacterSet csm = it.next();
- return csm.getLastChar();
+ private int metricsToAbsoluteSize(CharacterSet cs, int value, int givenSize) {
+ int nominalVerticalSize = cs.getNominalVerticalSize();
+ if (nominalVerticalSize != 0) {
+ return value * nominalVerticalSize;
} else {
- String msg = "getLastChar() - No character set found for font:" + getFontName();
- LOG.error(msg);
- throw new FontRuntimeException(msg);
+ return value * givenSize;
}
-
}
- private int metricsToAbsoluteSize(CharacterSet cs, int value, int givenSize) {
+ private int metricsToAbsoluteSize(CharacterSet cs, double value, int givenSize) {
int nominalVerticalSize = cs.getNominalVerticalSize();
if (nominalVerticalSize != 0) {
- return value * nominalVerticalSize;
+ return (int) (value * nominalVerticalSize);
} else {
- return value * givenSize;
+ return (int) (value * givenSize);
}
}
@@ -191,6 +166,20 @@ public class RasterFont extends AFPFont {
return metricsToAbsoluteSize(cs, cs.getAscender(), size);
}
+ /** {@inheritDoc} */
+ public int getUnderlinePosition(int size) {
+ CharacterSet cs = getCharacterSet(size);
+ return metricsToAbsoluteSize(cs, cs.getUnderscorePosition(), size);
+ }
+
+ @Override
+ public int getUnderlineThickness(int size) {
+ CharacterSet cs = getCharacterSet(size);
+ int underscoreWidth = cs.getUnderscoreWidth();
+ return underscoreWidth == 0 ? super.getUnderlineThickness(size)
+ : metricsToAbsoluteSize(cs, underscoreWidth, size);
+ }
+
/**
* Obtains the height of capital letters for the specified point size.
*
@@ -234,33 +223,20 @@ public class RasterFont extends AFPFont {
*/
public int getWidth(int character, int size) {
CharacterSet cs = getCharacterSet(size);
- return metricsToAbsoluteSize(cs, cs.getWidth(toUnicodeCodepoint(character)), size);
+ return metricsToAbsoluteSize(cs, cs.getWidth(toUnicodeCodepoint(character), 1), size);
}
/**
- * Get the getWidth (in 1/1000ths of a point size) of all characters in this
- * character set.
- *
- * @param size the font size (in mpt)
- * @return the widths of all characters
+ * TODO
*/
- public int[] getWidths(int size) {
+ public Rectangle getBoundingBox(int character, int size) {
CharacterSet cs = getCharacterSet(size);
- int[] widths = cs.getWidths();
- for (int i = 0, c = widths.length; i < c; i++) {
- widths[i] = metricsToAbsoluteSize(cs, widths[i], size);
- }
- return widths;
- }
-
- /**
- * Get the getWidth (in 1/1000ths of a point size) of all characters in this
- * character set.
- *
- * @return the widths of all characters
- */
- public int[] getWidths() {
- return getWidths(1000);
+ Rectangle characterBox = cs.getCharacterBox(toUnicodeCodepoint(character), 1);
+ int x = metricsToAbsoluteSize(cs, characterBox.getX(), size);
+ int y = metricsToAbsoluteSize(cs, characterBox.getY(), size);
+ int w = metricsToAbsoluteSize(cs, characterBox.getWidth(), size);
+ int h = metricsToAbsoluteSize(cs, characterBox.getHeight(), size);
+ return new Rectangle(x, y, w, h);
}
/** {@inheritDoc} */
diff --git a/src/java/org/apache/fop/afp/goca/GraphicsCharacterString.java b/src/java/org/apache/fop/afp/goca/GraphicsCharacterString.java
index 77ad7e806..d41adf867 100644
--- a/src/java/org/apache/fop/afp/goca/GraphicsCharacterString.java
+++ b/src/java/org/apache/fop/afp/goca/GraphicsCharacterString.java
@@ -67,7 +67,11 @@ public class GraphicsCharacterString extends AbstractGraphicsCoord {
/** {@inheritDoc} */
public int getDataLength() {
- return super.getDataLength() + str.length();
+ try {
+ return super.getDataLength() + getStringAsBytes().length;
+ } catch (IOException ioe) {
+ throw new RuntimeException(ioe);
+ }
}
/** {@inheritDoc} */
diff --git a/src/java/org/apache/fop/afp/svg/AFPBridgeContext.java b/src/java/org/apache/fop/afp/svg/AFPBridgeContext.java
index 2c6668454..5c13b9d92 100644
--- a/src/java/org/apache/fop/afp/svg/AFPBridgeContext.java
+++ b/src/java/org/apache/fop/afp/svg/AFPBridgeContext.java
@@ -25,13 +25,17 @@ import org.apache.batik.bridge.BridgeContext;
import org.apache.batik.bridge.DocumentLoader;
import org.apache.batik.bridge.UserAgent;
import org.apache.batik.gvt.TextPainter;
+import org.apache.batik.gvt.font.DefaultFontFamilyResolver;
+import org.apache.batik.gvt.font.FontFamilyResolver;
import org.apache.xmlgraphics.image.loader.ImageManager;
import org.apache.xmlgraphics.image.loader.ImageSessionContext;
import org.apache.fop.afp.AFPGraphics2D;
+import org.apache.fop.events.EventBroadcaster;
import org.apache.fop.fonts.FontInfo;
import org.apache.fop.svg.AbstractFOPBridgeContext;
+import org.apache.fop.svg.font.AggregatingFontFamilyResolver;
/**
* An AFP specific implementation of a Batik BridgeContext
@@ -40,6 +44,8 @@ public class AFPBridgeContext extends AbstractFOPBridgeContext {
private final AFPGraphics2D g2d;
+ private final EventBroadcaster eventBroadCaster;
+
/**
* Constructs a new bridge context.
*
@@ -54,47 +60,35 @@ public class AFPBridgeContext extends AbstractFOPBridgeContext {
*/
public AFPBridgeContext(UserAgent userAgent, FontInfo fontInfo,
ImageManager imageManager, ImageSessionContext imageSessionContext,
- AffineTransform linkTransform, AFPGraphics2D g2d) {
+ AffineTransform linkTransform, AFPGraphics2D g2d, EventBroadcaster eventBroadCaster) {
super(userAgent, fontInfo, imageManager, imageSessionContext, linkTransform);
this.g2d = g2d;
+ this.eventBroadCaster = eventBroadCaster;
}
- /**
- * Constructs a new bridge context.
- * @param userAgent the user agent
- * @param documentLoader the Document Loader to use for referenced documents.
- * @param fontInfo the font list for the text painter, may be null
- * in which case text is painted as shapes
- * @param imageManager an image manager
- * @param imageSessionContext an image session context
- * @param linkTransform AffineTransform to properly place links,
- * may be null
- * @param g2d an AFPGraphics 2D implementation
- */
- public AFPBridgeContext(UserAgent userAgent, DocumentLoader documentLoader,
+ private AFPBridgeContext(UserAgent userAgent, DocumentLoader documentLoader,
FontInfo fontInfo, ImageManager imageManager,
ImageSessionContext imageSessionContext,
- AffineTransform linkTransform, AFPGraphics2D g2d) {
- super(userAgent, documentLoader, fontInfo, imageManager,
- imageSessionContext, linkTransform);
+ AffineTransform linkTransform, AFPGraphics2D g2d, EventBroadcaster eventBroadCaster) {
+ super(userAgent, documentLoader, fontInfo, imageManager, imageSessionContext, linkTransform);
this.g2d = g2d;
+ this.eventBroadCaster = eventBroadCaster;
}
/** {@inheritDoc} */
@Override
public void registerSVGBridges() {
super.registerSVGBridges();
-
if (fontInfo != null) {
AFPTextHandler textHandler = new AFPTextHandler(fontInfo, g2d.getResourceManager());
g2d.setCustomTextHandler(textHandler);
-
- TextPainter textPainter = new AFPTextPainter(textHandler);
- setTextPainter(textPainter);
-
+ //TODO
+ FontFamilyResolver fontFamilyResolver = new AggregatingFontFamilyResolver(
+ new AFPFontFamilyResolver(fontInfo, eventBroadCaster), DefaultFontFamilyResolver.SINGLETON);
+ TextPainter textPainter = new AFPTextPainter(textHandler, fontFamilyResolver);
+ setTextPainter(new AFPTextPainter(textHandler, fontFamilyResolver));
putBridge(new AFPTextElementBridge(textPainter));
}
-
putBridge(new AFPImageElementBridge());
}
@@ -105,7 +99,7 @@ public class AFPBridgeContext extends AbstractFOPBridgeContext {
fontInfo,
getImageManager(),
getImageSessionContext(),
- linkTransform, g2d);
+ linkTransform, g2d, eventBroadCaster);
}
}
diff --git a/src/java/org/apache/fop/afp/svg/AFPFontFamilyResolver.java b/src/java/org/apache/fop/afp/svg/AFPFontFamilyResolver.java
new file mode 100644
index 000000000..27026f4f3
--- /dev/null
+++ b/src/java/org/apache/fop/afp/svg/AFPFontFamilyResolver.java
@@ -0,0 +1,84 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.afp.svg;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.apache.fop.afp.AFPEventProducer;
+import org.apache.fop.afp.fonts.DoubleByteFont;
+import org.apache.fop.events.EventBroadcaster;
+import org.apache.fop.fonts.Font;
+import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.fonts.FontTriplet;
+import org.apache.fop.fonts.Typeface;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
+import org.apache.fop.svg.font.FOPGVTFontFamily;
+import org.apache.fop.svg.font.FilteringFontFamilyResolver;
+
+public class AFPFontFamilyResolver extends FilteringFontFamilyResolver {
+
+ private final FontInfo fontInfo;
+
+ private final AFPEventProducer eventProducer;
+
+
+ public AFPFontFamilyResolver(FontInfo fontInfo, EventBroadcaster eventBroadCaster) {
+ super(new FOPFontFamilyResolverImpl(fontInfo));
+ this.fontInfo = fontInfo;
+ this.eventProducer = AFPEventProducer.Provider.get(eventBroadCaster);
+ }
+
+ @Override
+ public FOPGVTFontFamily resolve(String familyName) {
+ FOPGVTFontFamily fopGVTFontFamily = super.resolve(familyName);
+ // TODO why don't DB fonts work with GOCA?!?
+ if (fopGVTFontFamily != null && fopGVTFontFamily.deriveFont(1, new HashMap())
+ .getFont().getFontMetrics() instanceof DoubleByteFont) {
+ notifyDBFontRejection(fopGVTFontFamily.getFamilyName());
+ fopGVTFontFamily = null;
+ }
+ return fopGVTFontFamily;
+ }
+
+ @Override
+ public FOPGVTFontFamily getFamilyThatCanDisplay(char c) {
+ Map<String, Typeface> fonts = fontInfo.getFonts();
+ for (Typeface font : fonts.values()) {
+ // TODO why don't DB fonts work with GOCA?!?
+ if (font.hasChar(c) && !(font instanceof DoubleByteFont)) {
+ String fontFamily = font.getFamilyNames().iterator().next();
+ if (font instanceof DoubleByteFont) {
+ notifyDBFontRejection(font.getFontName());
+ } else {
+ return new FOPGVTFontFamily(fontInfo, fontFamily,
+ new FontTriplet(fontFamily, Font.STYLE_NORMAL, Font.WEIGHT_NORMAL));
+ }
+
+ }
+ }
+ return null;
+ }
+
+ private void notifyDBFontRejection(String fontFamily) {
+ eventProducer.invalidDBFontInSVG(this, fontFamily);
+ }
+
+}
diff --git a/src/java/org/apache/fop/afp/svg/AFPTextHandler.java b/src/java/org/apache/fop/afp/svg/AFPTextHandler.java
index 2bb4cb60e..3e987648e 100644
--- a/src/java/org/apache/fop/afp/svg/AFPTextHandler.java
+++ b/src/java/org/apache/fop/afp/svg/AFPTextHandler.java
@@ -135,27 +135,18 @@ public class AFPTextHandler extends FOPTextHandlerAdapter {
if (log.isDebugEnabled()) {
log.debug(" with overriding font: " + internalFontName + ", " + fontSize);
}
- } else {
- java.awt.Font awtFont = g2d.getFont();
- Font fopFont = fontInfo.getFontInstanceForAWTFont(awtFont);
- if (log.isDebugEnabled()) {
- log.debug(" with font: " + fopFont);
- }
- internalFontName = fopFont.getFontName();
- fontSize = fopFont.getFontSize();
+ fontSize = (int) Math.round(g2d.convertToAbsoluteLength(fontSize));
+ fontReference = registerPageFont(pageFonts, internalFontName, fontSize);
+ // TODO: re-think above registerPageFont code...
+ AFPFont afpFont = (AFPFont) fontInfo.getFonts().get(internalFontName);
+ final CharacterSet charSet = afpFont.getCharacterSet(fontSize);
+ // Work-around for InfoPrint's AFP which loses character set state
+ // over Graphics Data
+ // boundaries.
+ graphicsObj.setCharacterSet(fontReference);
+ // add the character string
+ graphicsObj.addString(str, Math.round(x), Math.round(y), charSet);
}
- fontSize = (int)Math.round(
- g2d.convertToAbsoluteLength(fontSize));
- fontReference = registerPageFont(pageFonts, internalFontName, fontSize);
- // TODO: re-think above registerPageFont code...
- AFPFont afpFont = (AFPFont) fontInfo.getFonts().get(internalFontName);
- final CharacterSet charSet = afpFont.getCharacterSet(fontSize);
- // Work-around for InfoPrint's AFP which loses character set state
- // over Graphics Data
- // boundaries.
- graphicsObj.setCharacterSet(fontReference);
- // add the character string
- graphicsObj.addString(str, Math.round(x), Math.round(y), charSet);
} else {
//Inside Batik's SVG filter operations, you won't get an AFPGraphics2D
g.drawString(str, x, y);
diff --git a/src/java/org/apache/fop/afp/svg/AFPTextPainter.java b/src/java/org/apache/fop/afp/svg/AFPTextPainter.java
index 3815c9eae..996ae8691 100644
--- a/src/java/org/apache/fop/afp/svg/AFPTextPainter.java
+++ b/src/java/org/apache/fop/afp/svg/AFPTextPainter.java
@@ -21,6 +21,9 @@ package org.apache.fop.afp.svg;
import java.awt.Graphics2D;
+import org.apache.batik.gvt.font.FontFamilyResolver;
+import org.apache.batik.gvt.renderer.StrokingTextPainter;
+
import org.apache.fop.afp.AFPGraphics2D;
import org.apache.fop.svg.AbstractFOPTextPainter;
import org.apache.fop.svg.FOPTextHandler;
@@ -39,8 +42,8 @@ public class AFPTextPainter extends AbstractFOPTextPainter {
* Create a new text painter with the given font information.
* @param nativeTextHandler the NativeTextHandler instance used for text painting
*/
- public AFPTextPainter(FOPTextHandler nativeTextHandler) {
- super(nativeTextHandler);
+ public AFPTextPainter(FOPTextHandler nativeTextHandler, FontFamilyResolver fopFontFamilyResolver) {
+ super(nativeTextHandler, new FOPStrokingTextPainter(fopFontFamilyResolver));
}
/** {@inheritDoc} */
@@ -48,4 +51,18 @@ public class AFPTextPainter extends AbstractFOPTextPainter {
return g2d instanceof AFPGraphics2D;
}
+ private static class FOPStrokingTextPainter extends StrokingTextPainter {
+
+ private final FontFamilyResolver fopFontFontFamily;
+
+ FOPStrokingTextPainter(FontFamilyResolver fopFontFontFamily) {
+ this.fopFontFontFamily = fopFontFontFamily;
+ }
+
+ @Override
+ protected FontFamilyResolver getFontFamilyResolver() {
+ return fopFontFontFamily;
+ }
+ }
+
}
diff --git a/src/java/org/apache/fop/fo/FOText.java b/src/java/org/apache/fop/fo/FOText.java
index 2fc998c63..0b7dde212 100644
--- a/src/java/org/apache/fop/fo/FOText.java
+++ b/src/java/org/apache/fop/fo/FOText.java
@@ -21,7 +21,6 @@ package org.apache.fop.fo;
import java.awt.Color;
import java.nio.CharBuffer;
-import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Stack;
@@ -38,12 +37,13 @@ import org.apache.fop.fo.properties.CommonTextDecoration;
import org.apache.fop.fo.properties.KeepProperty;
import org.apache.fop.fo.properties.Property;
import org.apache.fop.fo.properties.SpaceProperty;
+import org.apache.fop.fonts.TextFragment;
import org.apache.fop.util.CharUtilities;
/**
* A text node (PCDATA) in the formatting object tree.
*/
-public class FOText extends FONode implements CharSequence {
+public class FOText extends FONode implements CharSequence, TextFragment {
/** the <code>CharBuffer</code> containing the text */
private CharBuffer charBuffer;
@@ -93,9 +93,6 @@ public class FOText extends FONode implements CharSequence {
/* bidi levels */
private int[] bidiLevels;
- /* advanced script processing state */
- private Map/*<MapRange,String>*/ mappings;
-
private static final int IS_WORD_CHAR_FALSE = 0;
private static final int IS_WORD_CHAR_TRUE = 1;
private static final int IS_WORD_CHAR_MAYBE = 2;
@@ -804,93 +801,6 @@ public class FOText extends FONode implements CharSequence {
}
}
- /**
- * Add characters mapped by script substitution processing.
- * @param start index in character buffer
- * @param end index in character buffer
- * @param mappedChars sequence of character codes denoting substituted characters
- */
- public void addMapping(int start, int end, CharSequence mappedChars) {
- if (mappings == null) {
- mappings = new java.util.HashMap();
- }
- mappings.put(new MapRange(start, end), mappedChars.toString());
- }
-
- /**
- * Determine if characters over specific interval have a mapping.
- * @param start index in character buffer
- * @param end index in character buffer
- * @return true if a mapping exist such that the mapping's interval is coincident to
- * [start,end)
- */
- public boolean hasMapping(int start, int end) {
- return (mappings != null) && (mappings.containsKey(new MapRange(start, end)));
- }
-
- /**
- * Obtain mapping of characters over specific interval.
- * @param start index in character buffer
- * @param end index in character buffer
- * @return a string of characters representing the mapping over the interval
- * [start,end)
- */
- public String getMapping(int start, int end) {
- if (mappings != null) {
- return (String) mappings.get(new MapRange(start, end));
- } else {
- return null;
- }
- }
-
- /**
- * Obtain length of mapping of characters over specific interval.
- * @param start index in character buffer
- * @param end index in character buffer
- * @return the length of the mapping (if present) or zero
- */
- public int getMappingLength(int start, int end) {
- if (mappings != null) {
- return ((String) mappings.get(new MapRange(start, end))) .length();
- } else {
- return 0;
- }
- }
-
- /**
- * Obtain bidirectional levels of mapping of characters over specific interval.
- * @param start index in character buffer
- * @param end index in character buffer
- * @return a (possibly empty) array of bidi levels or null
- * in case no bidi levels have been assigned
- */
- public int[] getMappingBidiLevels(int start, int end) {
- if (hasMapping(start, end)) {
- int nc = end - start;
- int nm = getMappingLength(start, end);
- int[] la = getBidiLevels(start, end);
- if (la == null) {
- return null;
- } else if (nm == nc) { // mapping is same length as mapped range
- return la;
- } else if (nm > nc) { // mapping is longer than mapped range
- int[] ma = new int [ nm ];
- System.arraycopy(la, 0, ma, 0, la.length);
- for (int i = la.length,
- n = ma.length, l = (i > 0) ? la [ i - 1 ] : 0; i < n; i++) {
- ma [ i ] = l;
- }
- return ma;
- } else { // mapping is shorter than mapped range
- int[] ma = new int [ nm ];
- System.arraycopy(la, 0, ma, 0, ma.length);
- return ma;
- }
- } else {
- return getBidiLevels(start, end);
- }
- }
-
@Override
protected Stack collectDelimitedTextRanges(Stack ranges, DelimitedTextRange currentRange) {
if (currentRange != null) {
diff --git a/src/java/org/apache/fop/fonts/Base14Font.java b/src/java/org/apache/fop/fonts/Base14Font.java
index 9b2e95bc7..fdefd0cdd 100644
--- a/src/java/org/apache/fop/fonts/Base14Font.java
+++ b/src/java/org/apache/fop/fonts/Base14Font.java
@@ -25,4 +25,15 @@ package org.apache.fop.fonts;
*/
public abstract class Base14Font extends Typeface {
+ /** Thickness for underline and strikeout. */
+ private static final int LINE_THICKNESS = 50;
+
+ public int getStrikeoutPosition(int size) {
+ return getXHeight(size) / 2;
+ }
+
+ public int getStrikeoutThickness(int size) {
+ return size * LINE_THICKNESS;
+ }
+
}
diff --git a/src/java/org/apache/fop/fonts/CIDFont.java b/src/java/org/apache/fop/fonts/CIDFont.java
index dc398263e..26212ea56 100644
--- a/src/java/org/apache/fop/fonts/CIDFont.java
+++ b/src/java/org/apache/fop/fonts/CIDFont.java
@@ -19,6 +19,7 @@
package org.apache.fop.fonts;
+
import org.apache.fop.apps.io.InternalResourceResolver;
//Java
diff --git a/src/java/org/apache/fop/fonts/CustomFont.java b/src/java/org/apache/fop/fonts/CustomFont.java
index 70961a55c..96b70feea 100644
--- a/src/java/org/apache/fop/fonts/CustomFont.java
+++ b/src/java/org/apache/fop/fonts/CustomFont.java
@@ -37,6 +37,9 @@ import org.apache.fop.apps.io.InternalResourceResolver;
public abstract class CustomFont extends Typeface
implements FontDescriptor, MutableFont {
+ /** Fallback thickness for underline and strikeout when not provided by the font. */
+ private static final int DEFAULT_LINE_THICKNESS = 50;
+
private String fontName;
private String fullName;
private Set<String> familyNames;
@@ -60,6 +63,14 @@ public abstract class CustomFont extends Typeface
private int firstChar;
private int lastChar = 255;
+ private int underlinePosition;
+
+ private int underlineThickness;
+
+ private int strikeoutPosition;
+
+ private int strikeoutThickness;
+
private Map<Integer, Map<Integer, Integer>> kerning;
private boolean useKerning = true;
@@ -507,4 +518,40 @@ public abstract class CustomFont extends Typeface
return copy;
}
+ public int getUnderlinePosition(int size) {
+ return (underlinePosition == 0)
+ ? getDescender(size) / 2
+ : size * underlinePosition;
+ }
+
+ public void setUnderlinePosition(int underlinePosition) {
+ this.underlinePosition = underlinePosition;
+ }
+
+ public int getUnderlineThickness(int size) {
+ return size * ((underlineThickness == 0) ? DEFAULT_LINE_THICKNESS : underlineThickness);
+ }
+
+ public void setUnderlineThickness(int underlineThickness) {
+ this.underlineThickness = underlineThickness;
+ }
+
+ public int getStrikeoutPosition(int size) {
+ return (strikeoutPosition == 0)
+ ? getXHeight(size) / 2
+ : size * strikeoutPosition;
+ }
+
+ public void setStrikeoutPosition(int strikeoutPosition) {
+ this.strikeoutPosition = strikeoutPosition;
+ }
+
+ public int getStrikeoutThickness(int size) {
+ return (strikeoutThickness == 0) ? getUnderlineThickness(size) : size * strikeoutThickness;
+ }
+
+ public void setStrikeoutThickness(int strikeoutThickness) {
+ this.strikeoutThickness = strikeoutThickness;
+ }
+
}
diff --git a/src/java/org/apache/fop/fonts/FontInfo.java b/src/java/org/apache/fop/fonts/FontInfo.java
index 472567eec..720531205 100644
--- a/src/java/org/apache/fop/fonts/FontInfo.java
+++ b/src/java/org/apache/fop/fonts/FontInfo.java
@@ -30,6 +30,7 @@ import java.util.TreeSet;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
+
/**
* The FontInfo holds font information for the layout and rendering of a fo document.
* This stores the list of available fonts that are setup by
@@ -135,12 +136,10 @@ public class FontInfo {
if (oldName != null) {
int oldPriority = tripletPriorities.get(triplet).intValue();
if (oldPriority < newPriority) {
- logDuplicateFont(triplet, false, oldName, oldPriority,
- internalFontKey, newPriority);
+ logDuplicateFont(triplet, false, oldName, oldPriority, internalFontKey, newPriority);
return;
} else {
- logDuplicateFont(triplet, true, oldName, oldPriority,
- internalFontKey, newPriority);
+ logDuplicateFont(triplet, true, oldName, oldPriority, internalFontKey, newPriority);
}
}
this.triplets.put(triplet, internalFontKey);
@@ -157,9 +156,8 @@ public class FontInfo {
* @param newKey the new internal font name
* @param newPriority the priority of the duplicate font mapping
*/
- private void logDuplicateFont(FontTriplet triplet, boolean replacing,
- String oldKey, int oldPriority,
- String newKey, int newPriority) {
+ private void logDuplicateFont(FontTriplet triplet, boolean replacing, String oldKey, int oldPriority,
+ String newKey, int newPriority) {
if (log.isDebugEnabled()) {
log.debug(triplet
+ (replacing ? ": Replacing " : ": Not replacing ")
@@ -198,8 +196,7 @@ public class FontInfo {
* default font if not found
* @return internal font triplet key
*/
- private FontTriplet fontLookup(String family, String style,
- int weight, boolean substitutable) {
+ private FontTriplet fontLookup(String family, String style, int weight, boolean substitutable) {
if (log.isTraceEnabled()) {
log.trace("Font lookup: " + family + " " + style + " " + weight
+ (substitutable ? " substitutable" : ""));
@@ -302,8 +299,7 @@ public class FontInfo {
* @return the requested Font instance
*/
public Font getFontInstance(FontTriplet triplet, int fontSize) {
- Map<Integer, Font> sizes
- = getFontInstanceCache().get(triplet);
+ Map<Integer, Font> sizes = getFontInstanceCache().get(triplet);
if (sizes == null) {
sizes = new HashMap<Integer, Font>();
getFontInstanceCache().put(triplet, sizes);
@@ -379,13 +375,11 @@ public class FontInfo {
* @param weight font weight
* @return the font triplet of the font chosen
*/
- public FontTriplet fontLookup(String family, String style,
- int weight) {
+ public FontTriplet fontLookup(String family, String style, int weight) {
return fontLookup(family, style, weight, true);
}
- private List<FontTriplet> fontLookup(String[] families, String style,
- int weight, boolean substitutable) {
+ private List<FontTriplet> fontLookup(String[] families, String style, int weight, boolean substitutable) {
List<FontTriplet> matchingTriplets = new ArrayList<FontTriplet>();
FontTriplet triplet = null;
for (int i = 0; i < families.length; i++) {
@@ -410,8 +404,7 @@ public class FontInfo {
* @return the set of font triplets of all supported and chosen font-families
* in the specified style and weight.
*/
- public FontTriplet[] fontLookup(String[] families, String style,
- int weight) {
+ public FontTriplet[] fontLookup(String[] families, String style, int weight) {
if (families.length == 0) {
throw new IllegalArgumentException("Specify at least one font family");
}
@@ -434,8 +427,8 @@ public class FontInfo {
sb.append(families[i]);
}
throw new IllegalStateException(
- "fontLookup must return an array with at least one "
- + "FontTriplet on the last call. Lookup: " + sb.toString());
+ "fontLookup must return an array with at least one "
+ + "FontTriplet on the last call. Lookup: " + sb.toString());
}
FontTriplet[] fontTriplets = new FontTriplet[matchedTriplets.size()];
@@ -469,8 +462,7 @@ public class FontInfo {
* @param weight font weight
* @return internal key
*/
- public FontTriplet findAdjustWeight(String family, String style,
- int weight) {
+ public FontTriplet findAdjustWeight(String family, String style, int weight) {
FontTriplet key = null;
String f = null;
int newWeight = weight;
@@ -542,8 +534,7 @@ public class FontInfo {
* @param weight font weight
* @return internal key
*/
- public static FontTriplet createFontKey(String family, String style,
- int weight) {
+ public static FontTriplet createFontKey(String family, String style, int weight) {
return new FontTriplet(family, style, weight);
}
diff --git a/src/java/org/apache/fop/fonts/FontMetrics.java b/src/java/org/apache/fop/fonts/FontMetrics.java
index ff32d7305..159d321f7 100644
--- a/src/java/org/apache/fop/fonts/FontMetrics.java
+++ b/src/java/org/apache/fop/fonts/FontMetrics.java
@@ -19,6 +19,7 @@
package org.apache.fop.fonts;
+import java.awt.Rectangle;
import java.util.Map;
import java.util.Set;
@@ -120,6 +121,15 @@ public interface FontMetrics {
int[] getWidths();
/**
+ * Returns the bounding box of the glyph at the given index, for the given font size.
+ *
+ * @param glyphIndex glyph index
+ * @param size font size
+ * @return the scaled bounding box scaled in 1/1000ths of the given size
+ */
+ Rectangle getBoundingBox(int glyphIndex, int size);
+
+ /**
* Indicates if the font has kering information.
* @return True, if kerning is available.
*/
@@ -131,4 +141,38 @@ public interface FontMetrics {
*/
Map<Integer, Map<Integer, Integer>> getKerningInfo();
+ /**
+ * Returns the distance from the baseline to the center of the underline (negative
+ * value indicates below baseline).
+ *
+ * @param size font size
+ * @return the position in 1/1000ths of the font size
+ */
+ int getUnderlinePosition(int size);
+
+ /**
+ * Returns the thickness of the underline.
+ *
+ * @param size font size
+ * @return the thickness in 1/1000ths of the font size
+ */
+ int getUnderlineThickness(int size);
+
+ /**
+ * Returns the distance from the baseline to the center of the strikeout line
+ * (negative value indicates below baseline).
+ *
+ * @param size font size
+ * @return the position in 1/1000ths of the font size
+ */
+ int getStrikeoutPosition(int size);
+
+ /**
+ * Returns the thickness of the strikeout line.
+ *
+ * @param size font size
+ * @return the thickness in 1/1000ths of the font size
+ */
+ int getStrikeoutThickness(int size);
+
}
diff --git a/src/java/org/apache/fop/fonts/GlyphMapping.java b/src/java/org/apache/fop/fonts/GlyphMapping.java
new file mode 100644
index 000000000..a8a82085e
--- /dev/null
+++ b/src/java/org/apache/fop/fonts/GlyphMapping.java
@@ -0,0 +1,327 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.fonts;
+
+import org.apache.commons.logging.Log;
+import org.apache.commons.logging.LogFactory;
+
+import org.apache.fop.complexscripts.fonts.GlyphPositioningTable;
+import org.apache.fop.complexscripts.util.CharScript;
+import org.apache.fop.traits.MinOptMax;
+import org.apache.fop.util.CharUtilities;
+
+/**
+ * Stores the mapping of a text fragment to glyphs, along with various information.
+ */
+public class GlyphMapping {
+
+ private static final Log LOG = LogFactory.getLog(GlyphMapping.class);
+ /** Inclusive. */
+ public final int startIndex;
+ /** Exclusive. */
+ public final int endIndex;
+ private int wordCharLength;
+ public final int wordSpaceCount;
+ public int letterSpaceCount;
+ public MinOptMax areaIPD;
+ public final boolean isHyphenated;
+ public final boolean isSpace;
+ public boolean breakOppAfter;
+ public final Font font;
+ public final int level;
+ public final int[][] gposAdjustments;
+ public final String mapping;
+
+ public GlyphMapping(int startIndex, int endIndex, int wordSpaceCount, int letterSpaceCount,
+ MinOptMax areaIPD, boolean isHyphenated, boolean isSpace, boolean breakOppAfter,
+ Font font, int level, int[][] gposAdjustments) {
+ this(startIndex, endIndex, wordSpaceCount, letterSpaceCount, areaIPD, isHyphenated,
+ isSpace, breakOppAfter, font, level, gposAdjustments, null);
+ }
+
+ public GlyphMapping(int startIndex, int endIndex, int wordSpaceCount, int letterSpaceCount,
+ MinOptMax areaIPD, boolean isHyphenated, boolean isSpace, boolean breakOppAfter,
+ Font font, int level, int[][] gposAdjustments, String mapping) {
+ assert startIndex <= endIndex;
+ this.startIndex = startIndex;
+ this.endIndex = endIndex;
+ this.wordCharLength = -1;
+ this.wordSpaceCount = wordSpaceCount;
+ this.letterSpaceCount = letterSpaceCount;
+ this.areaIPD = areaIPD;
+ this.isHyphenated = isHyphenated;
+ this.isSpace = isSpace;
+ this.breakOppAfter = breakOppAfter;
+ this.font = font;
+ this.level = level;
+ this.gposAdjustments = gposAdjustments;
+ this.mapping = mapping;
+ }
+
+ public static GlyphMapping doGlyphMapping(TextFragment text, int startIndex, int endIndex,
+ Font font, MinOptMax letterSpaceIPD, MinOptMax[] letterSpaceAdjustArray,
+ char precedingChar, char breakOpportunityChar, final boolean endsWithHyphen, int level) {
+ GlyphMapping mapping;
+ if (font.performsSubstitution() || font.performsPositioning()) {
+ mapping = processWordMapping(text, startIndex, endIndex, font,
+ breakOpportunityChar, endsWithHyphen, level);
+ } else {
+ mapping = processWordNoMapping(text, startIndex, endIndex, font,
+ letterSpaceIPD, letterSpaceAdjustArray, precedingChar, breakOpportunityChar, endsWithHyphen,
+ level);
+ }
+ return mapping;
+ }
+
+ private static GlyphMapping processWordMapping(TextFragment text, int startIndex,
+ int endIndex, final Font font, final char breakOpportunityChar,
+ final boolean endsWithHyphen, int level) {
+ int e = endIndex; // end index of word in FOText character buffer
+ int nLS = 0; // # of letter spaces
+ String script = text.getScript();
+ String language = text.getLanguage();
+
+ if (LOG.isDebugEnabled()) {
+ LOG.debug("PW: [" + startIndex + "," + endIndex + "]: {"
+ + " +M"
+ + ", level = " + level
+ + " }");
+ }
+
+ // 1. extract unmapped character sequence
+ CharSequence ics = text.subSequence(startIndex, e);
+
+ // 2. if script is not specified (by FO property) or it is specified as 'auto',
+ // then compute dominant script
+ if ((script == null) || "auto".equals(script)) {
+ script = CharScript.scriptTagFromCode(CharScript.dominantScript(ics));
+ }
+ if ((language == null) || "none".equals(language)) {
+ language = "dflt";
+ }
+
+ // 3. perform mapping of chars to glyphs ... to glyphs ... to chars
+ CharSequence mcs = font.performSubstitution(ics, script, language);
+
+ // 4. compute glyph position adjustments on (substituted) characters
+ int[][] gpa;
+ if (font.performsPositioning()) {
+ // handle GPOS adjustments
+ gpa = font.performPositioning(mcs, script, language);
+ } else if (font.hasKerning()) {
+ // handle standard (non-GPOS) kerning adjustments
+ gpa = getKerningAdjustments(mcs, font);
+ } else {
+ gpa = null;
+ }
+
+ // 5. reorder combining marks so that they precede (within the mapped char sequence) the
+ // base to which they are applied; N.B. position adjustments (gpa) are reordered in place
+ mcs = font.reorderCombiningMarks(mcs, gpa, script, language);
+
+ // 6. compute word ipd based on final position adjustments
+ MinOptMax ipd = MinOptMax.ZERO;
+ for (int i = 0, n = mcs.length(); i < n; i++) {
+ int c = mcs.charAt(i);
+ // TODO !BMP
+ int w = font.getCharWidth(c);
+ if (w < 0) {
+ w = 0;
+ }
+ if (gpa != null) {
+ w += gpa[i][GlyphPositioningTable.Value.IDX_X_ADVANCE];
+ }
+ ipd = ipd.plus(w);
+ }
+
+ // [TBD] - handle letter spacing
+
+ return new GlyphMapping(startIndex, e, 0, nLS, ipd, endsWithHyphen, false,
+ breakOpportunityChar != 0, font, level, gpa,
+ CharUtilities.isSameSequence(mcs, ics) ? null : mcs.toString());
+ }
+
+ /**
+ * Given a mapped character sequence MCS, obtain glyph position adjustments from the
+ * font's kerning data.
+ *
+ * @param mcs mapped character sequence
+ * @param font applicable font
+ * @return glyph position adjustments (or null if no kerning)
+ */
+ private static int[][] getKerningAdjustments(CharSequence mcs, final Font font) {
+ int nc = mcs.length();
+ // extract kerning array
+ int[] ka = new int[nc]; // kerning array
+ for (int i = 0, n = nc, cPrev = -1; i < n; i++) {
+ int c = mcs.charAt(i);
+ // TODO !BMP
+ if (cPrev >= 0) {
+ ka[i] = font.getKernValue(cPrev, c);
+ }
+ cPrev = c;
+ }
+ // was there a non-zero kerning?
+ boolean hasKerning = false;
+ for (int i = 0, n = nc; i < n; i++) {
+ if (ka[i] != 0) {
+ hasKerning = true;
+ break;
+ }
+ }
+ // if non-zero kerning, then create and return glyph position adjustment array
+ if (hasKerning) {
+ int[][] gpa = new int[nc][4];
+ for (int i = 0, n = nc; i < n; i++) {
+ if (i > 0) {
+ gpa [i - 1][GlyphPositioningTable.Value.IDX_X_ADVANCE] = ka[i];
+ }
+ }
+ return gpa;
+ } else {
+ return null;
+ }
+ }
+
+ private static GlyphMapping processWordNoMapping(TextFragment text, int startIndex, int endIndex,
+ final Font font, MinOptMax letterSpaceIPD, MinOptMax[] letterSpaceAdjustArray,
+ char precedingChar, final char breakOpportunityChar, final boolean endsWithHyphen, int level) {
+ boolean kerning = font.hasKerning();
+ MinOptMax wordIPD = MinOptMax.ZERO;
+
+ if (LOG.isDebugEnabled()) {
+ LOG.debug("PW: [" + startIndex + "," + endIndex + "]: {"
+ + " -M"
+ + ", level = " + level
+ + " }");
+ }
+
+ for (int i = startIndex; i < endIndex; i++) {
+ char currentChar = text.charAt(i);
+
+ // character width
+ int charWidth = font.getCharWidth(currentChar);
+ wordIPD = wordIPD.plus(charWidth);
+
+ // kerning
+ if (kerning) {
+ int kern = 0;
+ if (i > startIndex) {
+ char previousChar = text.charAt(i - 1);
+ kern = font.getKernValue(previousChar, currentChar);
+ } else if (precedingChar != 0) {
+ kern = font.getKernValue(precedingChar, currentChar);
+ }
+ if (kern != 0) {
+ addToLetterAdjust(letterSpaceAdjustArray, i, kern);
+ wordIPD = wordIPD.plus(kern);
+ }
+ }
+ }
+ if (kerning
+ && (breakOpportunityChar != 0)
+ && !isSpace(breakOpportunityChar)
+ && endIndex > 0
+ && endsWithHyphen) {
+ int kern = font.getKernValue(text.charAt(endIndex - 1), breakOpportunityChar);
+ if (kern != 0) {
+ addToLetterAdjust(letterSpaceAdjustArray, endIndex, kern);
+ // TODO: add kern to wordIPD?
+ }
+ }
+ // shy+chars at start of word: wordLength == 0 && breakOpportunity
+ // shy only characters in word: wordLength == 0 && !breakOpportunity
+ int wordLength = endIndex - startIndex;
+ int letterSpaces = 0;
+ if (wordLength != 0) {
+ letterSpaces = wordLength - 1;
+ // if there is a break opportunity and the next one (break character)
+ // is not a space, it could be used as a line end;
+ // add one more letter space, in case other text follows
+ if ((breakOpportunityChar != 0) && !isSpace(breakOpportunityChar)) {
+ letterSpaces++;
+ }
+ }
+ assert letterSpaces >= 0;
+ wordIPD = wordIPD.plus(letterSpaceIPD.mult(letterSpaces));
+
+ // create and return the AreaInfo object
+ return new GlyphMapping(startIndex, endIndex, 0,
+ letterSpaces, wordIPD,
+ endsWithHyphen,
+ false, breakOpportunityChar != 0, font, level, null);
+ }
+
+ private static void addToLetterAdjust(MinOptMax[] letterSpaceAdjustArray, int index, int width) {
+ if (letterSpaceAdjustArray[index] == null) {
+ letterSpaceAdjustArray[index] = MinOptMax.getInstance(width);
+ } else {
+ letterSpaceAdjustArray[index] = letterSpaceAdjustArray[index].plus(width);
+ }
+ }
+
+ /**
+ * Indicates whether a character is a space in terms of this layout manager.
+ *
+ * @param ch the character
+ * @return true if it's a space
+ */
+ public static boolean isSpace(final char ch) {
+ return ch == CharUtilities.SPACE
+ || CharUtilities.isNonBreakableSpace(ch)
+ || CharUtilities.isFixedWidthSpace(ch);
+ }
+
+ /**
+ * Obtain number of 'characters' contained in word. If word is mapped, then this
+ * number may be less than or greater than the original length (breakIndex -
+ * startIndex). We compute and memoize thius length upon first invocation of this
+ * method.
+ */
+ public int getWordLength() {
+ if (wordCharLength == -1) {
+ if (mapping != null) {
+ wordCharLength = mapping.length();
+ } else {
+ assert endIndex >= startIndex;
+ wordCharLength = endIndex - startIndex;
+ }
+ }
+ return wordCharLength;
+ }
+
+ public void addToAreaIPD(MinOptMax idp) {
+ areaIPD = areaIPD.plus(idp);
+ }
+
+ public String toString() {
+ return super.toString() + "{"
+ + "interval = [" + startIndex + "," + endIndex + "]"
+ + ", isSpace = " + isSpace
+ + ", level = " + level
+ + ", areaIPD = " + areaIPD
+ + ", letterSpaceCount = " + letterSpaceCount
+ + ", wordSpaceCount = " + wordSpaceCount
+ + ", isHyphenated = " + isHyphenated
+ + ", font = " + font
+ + "}";
+ }
+
+}
diff --git a/src/java/org/apache/fop/fonts/LazyFont.java b/src/java/org/apache/fop/fonts/LazyFont.java
index 49b2e0457..1877a895e 100644
--- a/src/java/org/apache/fop/fonts/LazyFont.java
+++ b/src/java/org/apache/fop/fonts/LazyFont.java
@@ -18,6 +18,7 @@
/* $Id$ */
package org.apache.fop.fonts;
+import java.awt.Rectangle;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
@@ -250,6 +251,26 @@ public class LazyFont extends Typeface implements FontDescriptor, Substitutable,
return realFont.getXHeight(size);
}
+ public int getUnderlinePosition(int size) {
+ load(true);
+ return realFont.getUnderlinePosition(size);
+ }
+
+ public int getUnderlineThickness(int size) {
+ load(true);
+ return realFont.getUnderlineThickness(size);
+ }
+
+ public int getStrikeoutPosition(int size) {
+ load(true);
+ return realFont.getStrikeoutPosition(size);
+ }
+
+ public int getStrikeoutThickness(int size) {
+ load(true);
+ return realFont.getStrikeoutThickness(size);
+ }
+
/**
* {@inheritDoc}
*/
@@ -268,6 +289,11 @@ public class LazyFont extends Typeface implements FontDescriptor, Substitutable,
return realFont.getWidths();
}
+ public Rectangle getBoundingBox(int glyphIndex, int size) {
+ load(true);
+ return realFont.getBoundingBox(glyphIndex, size);
+ }
+
/**
* {@inheritDoc}
*/
diff --git a/src/java/org/apache/fop/fonts/MultiByteFont.java b/src/java/org/apache/fop/fonts/MultiByteFont.java
index 83f6491f3..1d1186dcb 100644
--- a/src/java/org/apache/fop/fonts/MultiByteFont.java
+++ b/src/java/org/apache/fop/fonts/MultiByteFont.java
@@ -19,6 +19,7 @@
package org.apache.fop.fonts;
+import java.awt.Rectangle;
import java.nio.CharBuffer;
import java.nio.IntBuffer;
import java.util.BitSet;
@@ -67,6 +68,9 @@ public class MultiByteFont extends CIDFont implements Substitutable, Positionabl
private int firstUnmapped;
private int lastUnmapped;
+ /** Contains the character bounding boxes for all characters in the font */
+ protected Rectangle[] boundingBoxes;
+
/**
* Default constructor
*/
@@ -170,6 +174,12 @@ public class MultiByteFont extends CIDFont implements Substitutable, Positionabl
return arr;
}
+ public Rectangle getBoundingBox(int glyphIndex, int size) {
+ int index = isEmbeddable() ? cidSet.getOriginalGlyphIndex(glyphIndex) : glyphIndex;
+ Rectangle bbox = boundingBoxes[index];
+ return new Rectangle(bbox.x * size, bbox.y * size, bbox.width * size, bbox.height * size);
+ }
+
/**
* Returns the glyph index for a Unicode character. The method returns 0 if there's no
* such glyph in the character map.
@@ -366,6 +376,14 @@ public class MultiByteFont extends CIDFont implements Substitutable, Positionabl
}
/**
+ * Sets the bounding boxes array.
+ * @param boundingBoxes array of bounding boxes.
+ */
+ public void setBBoxArray(Rectangle[] boundingBoxes) {
+ this.boundingBoxes = boundingBoxes;
+ }
+
+ /**
* Returns a Map of used Glyphs.
* @return Map Map of used Glyphs
*/
diff --git a/src/java/org/apache/fop/fonts/SingleByteFont.java b/src/java/org/apache/fop/fonts/SingleByteFont.java
index cd11c7849..d412609d5 100644
--- a/src/java/org/apache/fop/fonts/SingleByteFont.java
+++ b/src/java/org/apache/fop/fonts/SingleByteFont.java
@@ -19,6 +19,7 @@
package org.apache.fop.fonts;
+import java.awt.Rectangle;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
@@ -47,6 +48,8 @@ public class SingleByteFont extends CustomFont {
private int[] width = null;
+ private Rectangle[] boundingBoxes;
+
private Map<Character, UnencodedCharacter> unencodedCharacters;
private List<SimpleSingleByteEncoding> additionalEncodings;
private Map<Character, Character> alternativeCodes;
@@ -111,6 +114,24 @@ public class SingleByteFont extends CustomFont {
return arr;
}
+ public Rectangle getBoundingBox(int glyphIndex, int size) {
+ Rectangle bbox = null;
+ if (glyphIndex < 256) {
+ int idx = glyphIndex - getFirstChar();
+ if (idx >= 0 && idx < boundingBoxes.length) {
+ bbox = boundingBoxes[idx];
+ }
+ } else if (this.additionalEncodings != null) {
+ int encodingIndex = (glyphIndex / 256) - 1;
+ SimpleSingleByteEncoding encoding = getAdditionalEncoding(encodingIndex);
+ int codePoint = glyphIndex % 256;
+ NamedCharacter nc = encoding.getCharacterForIndex(codePoint);
+ UnencodedCharacter uc = this.unencodedCharacters.get(Character.valueOf(nc.getSingleUnicodeValue()));
+ bbox = uc.getBBox();
+ }
+ return bbox == null ? null : new Rectangle(bbox.x * size, bbox.y * size, bbox.width * size, bbox.height * size);
+ }
+
/**
* Lookup a character using its alternative names. If found, cache it so we
* can speed up lookups.
@@ -292,17 +313,24 @@ public class SingleByteFont extends CustomFont {
this.width[index - getFirstChar()] = w;
}
+ public void setBoundingBox(int index, Rectangle bbox) {
+ if (this.boundingBoxes == null) {
+ this.boundingBoxes = new Rectangle[getLastChar() - getFirstChar() + 1];
+ }
+ this.boundingBoxes[index - getFirstChar()] = bbox;
+ }
+
/**
* Adds an unencoded character (one that is not supported by the primary encoding).
* @param ch the named character
* @param width the width of the character
*/
- public void addUnencodedCharacter(NamedCharacter ch, int width) {
+ public void addUnencodedCharacter(NamedCharacter ch, int width, Rectangle bbox) {
if (this.unencodedCharacters == null) {
this.unencodedCharacters = new HashMap<Character, UnencodedCharacter>();
}
if (ch.hasSingleUnicodeValue()) {
- UnencodedCharacter uc = new UnencodedCharacter(ch, width);
+ UnencodedCharacter uc = new UnencodedCharacter(ch, width, bbox);
this.unencodedCharacters.put(Character.valueOf(ch.getSingleUnicodeValue()), uc);
} else {
//Cannot deal with unicode sequences, so ignore this character
@@ -381,10 +409,12 @@ public class SingleByteFont extends CustomFont {
private final NamedCharacter character;
private final int width;
+ private final Rectangle bbox;
- public UnencodedCharacter(NamedCharacter character, int width) {
+ public UnencodedCharacter(NamedCharacter character, int width, Rectangle bbox) {
this.character = character;
this.width = width;
+ this.bbox = bbox;
}
public NamedCharacter getCharacter() {
@@ -395,6 +425,10 @@ public class SingleByteFont extends CustomFont {
return this.width;
}
+ public Rectangle getBBox() {
+ return bbox;
+ }
+
/** {@inheritDoc} */
@Override
public String toString() {
diff --git a/src/java/org/apache/fop/fonts/TextFragment.java b/src/java/org/apache/fop/fonts/TextFragment.java
new file mode 100644
index 000000000..ad72db8e0
--- /dev/null
+++ b/src/java/org/apache/fop/fonts/TextFragment.java
@@ -0,0 +1,31 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.fonts;
+
+public interface TextFragment {
+
+ String getScript();
+
+ String getLanguage();
+
+ char charAt(int index);
+
+ CharSequence subSequence(int startIndex, int endIndex);
+}
diff --git a/src/java/org/apache/fop/fonts/truetype/TTFFile.java b/src/java/org/apache/fop/fonts/truetype/TTFFile.java
index 2e78ef7f0..62686bbd4 100644
--- a/src/java/org/apache/fop/fonts/truetype/TTFFile.java
+++ b/src/java/org/apache/fop/fonts/truetype/TTFFile.java
@@ -19,6 +19,7 @@
package org.apache.fop.fonts.truetype;
+import java.awt.Rectangle;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
@@ -199,6 +200,8 @@ public class TTFFile {
private int os2CapHeight = 0;
private int underlinePosition = 0;
private int underlineThickness = 0;
+ private int strikeoutPosition;
+ private int strikeoutThickness;
private int xHeight = 0;
private int os2xHeight = 0;
//Effective ascender/descender
@@ -976,10 +979,22 @@ public class TTFFile {
for (int i = 0; i < wx.length; i++) {
wx[i] = convertTTFUnit2PDFUnit(mtxTab[i].getWx());
}
-
return wx;
}
+ public Rectangle[] getBoundingBoxes() {
+ Rectangle[] boundingBoxes = new Rectangle[mtxTab.length];
+ for (int i = 0; i < boundingBoxes.length; i++) {
+ int[] boundingBox = mtxTab[i].getBoundingBox();
+ boundingBoxes[i] = new Rectangle(
+ convertTTFUnit2PDFUnit(boundingBox[0]),
+ convertTTFUnit2PDFUnit(boundingBox[1]),
+ convertTTFUnit2PDFUnit(boundingBox[2] - boundingBox[0]),
+ convertTTFUnit2PDFUnit(boundingBox[3] - boundingBox[1]));
+ }
+ return boundingBoxes;
+ }
+
/**
* Returns an array (xMin, yMin, xMax, yMax) for a glyph.
*
@@ -1020,6 +1035,22 @@ public class TTFFile {
return ansiKerningTab;
}
+ public int getUnderlinePosition() {
+ return convertTTFUnit2PDFUnit(underlinePosition);
+ }
+
+ public int getUnderlineThickness() {
+ return convertTTFUnit2PDFUnit(underlineThickness);
+ }
+
+ public int getStrikeoutPosition() {
+ return convertTTFUnit2PDFUnit(strikeoutPosition);
+ }
+
+ public int getStrikeoutThickness() {
+ return convertTTFUnit2PDFUnit(strikeoutThickness);
+ }
+
/**
* Indicates if the font may be embedded.
* @return boolean True if it may be embedded
@@ -1301,7 +1332,10 @@ public class TTFFile {
} else {
isEmbeddable = true;
}
- fontFile.skip(11 * 2);
+ fontFile.skip(8 * 2);
+ strikeoutThickness = fontFile.readTTFShort();
+ strikeoutPosition = fontFile.readTTFShort();
+ fontFile.skip(2);
fontFile.skip(10); //panose array
fontFile.skip(4 * 4); //unicode ranges
fontFile.skip(4);
diff --git a/src/java/org/apache/fop/fonts/truetype/TTFFontLoader.java b/src/java/org/apache/fop/fonts/truetype/TTFFontLoader.java
index b7adbd4c9..98f81ab3e 100644
--- a/src/java/org/apache/fop/fonts/truetype/TTFFontLoader.java
+++ b/src/java/org/apache/fop/fonts/truetype/TTFFontLoader.java
@@ -19,6 +19,7 @@
package org.apache.fop.fonts.truetype;
+import java.awt.Rectangle;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
@@ -144,6 +145,10 @@ public class TTFFontLoader extends FontLoader {
returnFont.setAscender(ttf.getLowerCaseAscent());
returnFont.setDescender(ttf.getLowerCaseDescent());
returnFont.setFontBBox(ttf.getFontBBox());
+ returnFont.setUnderlinePosition(ttf.getUnderlinePosition() - ttf.getUnderlineThickness() / 2);
+ returnFont.setUnderlineThickness(ttf.getUnderlineThickness());
+ returnFont.setStrikeoutPosition(ttf.getStrikeoutPosition() - ttf.getStrikeoutThickness() / 2);
+ returnFont.setStrikeoutThickness(ttf.getStrikeoutThickness());
returnFont.setFlags(ttf.getFlags());
returnFont.setStemV(Integer.parseInt(ttf.getStemV())); //not used for TTF
returnFont.setItalicAngle(Integer.parseInt(ttf.getItalicAngle()));
@@ -152,15 +157,15 @@ public class TTFFontLoader extends FontLoader {
returnFont.setEmbeddingMode(this.embeddingMode);
if (isCid) {
multiFont.setCIDType(CIDFontType.CIDTYPE2);
- int[] wx = ttf.getWidths();
- multiFont.setWidthArray(wx);
+ multiFont.setWidthArray(ttf.getWidths());
+ multiFont.setBBoxArray(ttf.getBoundingBoxes());
} else {
singleFont.setFontType(FontType.TRUETYPE);
singleFont.setEncoding(ttf.getCharSetName());
returnFont.setFirstChar(ttf.getFirstChar());
returnFont.setLastChar(ttf.getLastChar());
singleFont.setTrueTypePostScriptVersion(ttf.getPostScriptVersion());
- copyWidthsSingleByte(ttf);
+ copyGlyphMetricsSingleByte(ttf);
}
returnFont.setCMap(getCMap(ttf));
@@ -186,12 +191,15 @@ public class TTFFontLoader extends FontLoader {
return ttf.getCMaps().toArray(array);
}
- private void copyWidthsSingleByte(TTFFile ttf) {
+ private void copyGlyphMetricsSingleByte(TTFFile ttf) {
int[] wx = ttf.getWidths();
+ Rectangle[] bboxes = ttf.getBoundingBoxes();
for (int i = singleFont.getFirstChar(); i <= singleFont.getLastChar(); i++) {
singleFont.setWidth(i, ttf.getCharWidth(i));
+ int[] bbox = ttf.getBBox(i);
+ singleFont.setBoundingBox(i,
+ new Rectangle(bbox[0], bbox[1], bbox[2] - bbox[0], bbox[3] - bbox[1]));
}
-
for (CMapSegment segment : ttf.getCMaps()) {
if (segment.getUnicodeStart() < 0xFFFE) {
for (char u = (char)segment.getUnicodeStart(); u <= segment.getUnicodeEnd(); u++) {
@@ -205,7 +213,7 @@ public class TTFFontLoader extends FontLoader {
if (glyphName.length() > 0) {
String unicode = Character.toString(u);
NamedCharacter nc = new NamedCharacter(glyphName, unicode);
- singleFont.addUnencodedCharacter(nc, wx[glyphIndex]);
+ singleFont.addUnencodedCharacter(nc, wx[glyphIndex], bboxes[glyphIndex]);
}
}
}
diff --git a/src/java/org/apache/fop/fonts/type1/AFMCharMetrics.java b/src/java/org/apache/fop/fonts/type1/AFMCharMetrics.java
index 4906f2c31..8d9f4238b 100644
--- a/src/java/org/apache/fop/fonts/type1/AFMCharMetrics.java
+++ b/src/java/org/apache/fop/fonts/type1/AFMCharMetrics.java
@@ -19,7 +19,7 @@
package org.apache.fop.fonts.type1;
-import java.awt.geom.RectangularShape;
+import java.awt.Rectangle;
import org.apache.fop.fonts.NamedCharacter;
@@ -33,7 +33,7 @@ public class AFMCharMetrics {
private NamedCharacter character;
private double widthX;
private double widthY;
- private RectangularShape bBox;
+ private Rectangle bBox;
/**
* Returns the character code.
@@ -137,7 +137,7 @@ public class AFMCharMetrics {
* Returns the character's bounding box.
* @return the bounding box (or null if it isn't available)
*/
- public RectangularShape getBBox() {
+ public Rectangle getBBox() {
return bBox;
}
@@ -145,7 +145,7 @@ public class AFMCharMetrics {
* Sets the character's bounding box.
* @param box the bounding box
*/
- public void setBBox(RectangularShape box) {
+ public void setBBox(Rectangle box) {
bBox = box;
}
diff --git a/src/java/org/apache/fop/fonts/type1/Type1FontLoader.java b/src/java/org/apache/fop/fonts/type1/Type1FontLoader.java
index 853e23eb5..7922ff6fd 100644
--- a/src/java/org/apache/fop/fonts/type1/Type1FontLoader.java
+++ b/src/java/org/apache/fop/fonts/type1/Type1FontLoader.java
@@ -203,12 +203,16 @@ public class Type1FontLoader extends FontLoader {
for (AFMCharMetrics metrics : charMetrics) {
String charName = metrics.getCharName();
if (charName != null && !glyphNames.contains(charName)) {
- singleFont.addUnencodedCharacter(metrics.getCharacter(),
- (int)Math.round(metrics.getWidthX()));
+ addUnencodedCharacter(singleFont, metrics);
}
}
}
+ private static void addUnencodedCharacter(SingleByteFont font, AFMCharMetrics metrics) {
+ font.addUnencodedCharacter(metrics.getCharacter(),
+ (int) Math.round(metrics.getWidthX()), metrics.getBBox());
+ }
+
/**
* Adds characters not encoded in the font's primary encoding. This method is used when
* the primary encoding is built based on the character codes in the AFM rather than
@@ -220,8 +224,7 @@ public class Type1FontLoader extends FontLoader {
for (int i = 0, c = afm.getCharCount(); i < c; i++) {
AFMCharMetrics metrics = (AFMCharMetrics)charMetrics.get(i);
if (!metrics.hasCharCode() && metrics.getCharacter() != null) {
- singleFont.addUnencodedCharacter(metrics.getCharacter(),
- (int)Math.round(metrics.getWidthX()));
+ addUnencodedCharacter(singleFont, metrics);
}
}
}
@@ -267,7 +270,10 @@ public class Type1FontLoader extends FontLoader {
} else {
returnFont.setStemV(80); // Arbitrary value
}
- returnFont.setItalicAngle((int) afm.getWritingDirectionMetrics(0).getItalicAngle());
+ AFMWritingDirectionMetrics metrics = afm.getWritingDirectionMetrics(0);
+ returnFont.setItalicAngle((int) metrics.getItalicAngle());
+ returnFont.setUnderlinePosition(metrics.getUnderlinePosition().intValue());
+ returnFont.setUnderlineThickness(metrics.getUnderlineThickness().intValue());
} else {
returnFont.setFontBBox(pfm.getFontBBox());
returnFont.setStemV(pfm.getStemV());
@@ -369,6 +375,7 @@ public class Type1FontLoader extends FontLoader {
for (AFMCharMetrics chm : afm.getCharMetrics()) {
if (chm.hasCharCode()) {
singleFont.setWidth(chm.getCharCode(), (int) Math.round(chm.getWidthX()));
+ singleFont.setBoundingBox(chm.getCharCode(), chm.getBBox());
}
}
if (useKerning) {
diff --git a/src/java/org/apache/fop/image/loader/batik/ImageConverterSVG2G2D.java b/src/java/org/apache/fop/image/loader/batik/ImageConverterSVG2G2D.java
index 79592efd1..ec187b8d0 100644
--- a/src/java/org/apache/fop/image/loader/batik/ImageConverterSVG2G2D.java
+++ b/src/java/org/apache/fop/image/loader/batik/ImageConverterSVG2G2D.java
@@ -33,6 +33,7 @@ import org.apache.batik.bridge.GVTBuilder;
import org.apache.batik.bridge.UserAgent;
import org.apache.batik.dom.svg.SVGDOMImplementation;
import org.apache.batik.gvt.GraphicsNode;
+import org.apache.batik.gvt.font.DefaultFontFamilyResolver;
import org.apache.xmlgraphics.image.GraphicsConstants;
import org.apache.xmlgraphics.image.loader.Image;
@@ -123,10 +124,8 @@ public class ImageConverterSVG2G2D extends AbstractImageConverter {
* @return the newly created user agent
*/
protected SimpleSVGUserAgent createBatikUserAgent(float pxToMillimeter) {
- return new SimpleSVGUserAgent(
- pxToMillimeter,
- new AffineTransform()) {
-
+ return new SimpleSVGUserAgent(pxToMillimeter, new AffineTransform(),
+ DefaultFontFamilyResolver.SINGLETON) {
/** {@inheritDoc} */
public void displayMessage(String message) {
//TODO Refine and pipe through to caller
diff --git a/src/java/org/apache/fop/image/loader/batik/PreloaderSVG.java b/src/java/org/apache/fop/image/loader/batik/PreloaderSVG.java
index 3aa340a4a..58daadc52 100644
--- a/src/java/org/apache/fop/image/loader/batik/PreloaderSVG.java
+++ b/src/java/org/apache/fop/image/loader/batik/PreloaderSVG.java
@@ -38,6 +38,7 @@ import org.apache.batik.bridge.UnitProcessor;
import org.apache.batik.bridge.UserAgent;
import org.apache.batik.dom.svg.SAXSVGDocumentFactory;
import org.apache.batik.dom.svg.SVGOMDocument;
+import org.apache.batik.gvt.font.DefaultFontFamilyResolver;
import org.apache.xmlgraphics.image.loader.ImageContext;
import org.apache.xmlgraphics.image.loader.ImageInfo;
@@ -162,7 +163,7 @@ public class PreloaderSVG extends AbstractImagePreloader {
Element e = doc.getRootElement();
float pxUnitToMillimeter = UnitConv.IN2MM / context.getSourceResolution();
UserAgent userAg = new SimpleSVGUserAgent(pxUnitToMillimeter,
- new AffineTransform()) {
+ new AffineTransform(), DefaultFontFamilyResolver.SINGLETON) {
/** {@inheritDoc} */
public void displayMessage(String message) {
diff --git a/src/java/org/apache/fop/layoutmgr/inline/TextLayoutManager.java b/src/java/org/apache/fop/layoutmgr/inline/TextLayoutManager.java
index 53f51cd32..625b43ee5 100644
--- a/src/java/org/apache/fop/layoutmgr/inline/TextLayoutManager.java
+++ b/src/java/org/apache/fop/layoutmgr/inline/TextLayoutManager.java
@@ -30,12 +30,11 @@ import org.apache.commons.logging.LogFactory;
import org.apache.fop.area.Trait;
import org.apache.fop.area.inline.TextArea;
-import org.apache.fop.complexscripts.fonts.GlyphPositioningTable;
-import org.apache.fop.complexscripts.util.CharScript;
import org.apache.fop.fo.Constants;
import org.apache.fop.fo.FOText;
import org.apache.fop.fonts.Font;
import org.apache.fop.fonts.FontSelector;
+import org.apache.fop.fonts.GlyphMapping;
import org.apache.fop.layoutmgr.InlineKnuthSequence;
import org.apache.fop.layoutmgr.KnuthBox;
import org.apache.fop.layoutmgr.KnuthElement;
@@ -65,91 +64,15 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
private static final int SOFT_HYPHEN_PENALTY = 1;
/**
- * Store information about each potential text area.
- * Index of character which ends the area, IPD of area, including
- * any word-space and letter-space.
- * Number of word-spaces?
- */
- private class AreaInfo {
-
- private final int startIndex;
- private final int breakIndex;
- private int wordCharLength;
- private final int wordSpaceCount;
- private int letterSpaceCount;
- private MinOptMax areaIPD;
- private final boolean isHyphenated;
- private final boolean isSpace;
- private boolean breakOppAfter;
- private final Font font;
- private final int level;
- private final int[][] gposAdjustments;
-
- AreaInfo(
- int startIndex, int breakIndex, int wordSpaceCount, int letterSpaceCount,
- MinOptMax areaIPD, boolean isHyphenated, boolean isSpace, boolean breakOppAfter,
- Font font, int level, int[][] gposAdjustments) {
- assert startIndex <= breakIndex;
- this.startIndex = startIndex;
- this.breakIndex = breakIndex;
- this.wordCharLength = -1;
- this.wordSpaceCount = wordSpaceCount;
- this.letterSpaceCount = letterSpaceCount;
- this.areaIPD = areaIPD;
- this.isHyphenated = isHyphenated;
- this.isSpace = isSpace;
- this.breakOppAfter = breakOppAfter;
- this.font = font;
- this.level = level;
- this.gposAdjustments = gposAdjustments;
- }
-
- /**
- * Obtain number of 'characters' contained in word. If word
- * is mapped, then this number may be less than or greater than the
- * original length (breakIndex - startIndex). We compute and
- * memoize thius length upon first invocation of this method.
- */
- private int getWordLength() {
- if (wordCharLength == -1) {
- if (foText.hasMapping(startIndex, breakIndex)) {
- wordCharLength = foText.getMapping(startIndex, breakIndex).length();
- } else {
- assert breakIndex >= startIndex;
- wordCharLength = breakIndex - startIndex;
- }
- }
- return wordCharLength;
- }
-
- private void addToAreaIPD(MinOptMax idp) {
- areaIPD = areaIPD.plus(idp);
- }
-
- public String toString() {
- return super.toString() + "{"
- + "interval = [" + startIndex + "," + breakIndex + "]"
- + ", isSpace = " + isSpace
- + ", level = " + level
- + ", areaIPD = " + areaIPD
- + ", letterSpaceCount = " + letterSpaceCount
- + ", wordSpaceCount = " + wordSpaceCount
- + ", isHyphenated = " + isHyphenated
- + ", font = " + font
- + "}";
- }
- }
-
- /**
* this class stores information about changes in vecAreaInfo which are not yet applied
*/
private final class PendingChange {
- private final AreaInfo areaInfo;
+ private final GlyphMapping mapping;
private final int index;
- private PendingChange(final AreaInfo areaInfo, final int index) {
- this.areaInfo = areaInfo;
+ private PendingChange(final GlyphMapping mapping, final int index) {
+ this.mapping = mapping;
this.index = index;
}
}
@@ -160,7 +83,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
private static final Log LOG = LogFactory.getLog(TextLayoutManager.class);
// Hold all possible breaks for the text in this LM's FO.
- private final List areaInfos;
+ private final List<GlyphMapping> mappings;
/** Non-space characters on which we can end a line. */
private static final String BREAK_CHARS = "-/";
@@ -216,7 +139,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
public TextLayoutManager(FOText node) {
foText = node;
letterSpaceAdjustArray = new MinOptMax[node.length() + 1];
- areaInfos = new ArrayList();
+ mappings = new ArrayList<GlyphMapping>();
}
private KnuthPenalty makeZeroWidthPenalty(int penaltyValue) {
@@ -274,61 +197,61 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
public void addAreas(final PositionIterator posIter, final LayoutContext context) {
// Add word areas
- AreaInfo areaInfo;
+ GlyphMapping mapping;
int wordSpaceCount = 0;
int letterSpaceCount = 0;
- int firstAreaInfoIndex = -1;
- int lastAreaInfoIndex = 0;
+ int firstMappingIndex = -1;
+ int lastMappingIndex = 0;
MinOptMax realWidth = MinOptMax.ZERO;
/* On first area created, add any leading space.
* Calculate word-space stretch value.
*/
- AreaInfo lastAreaInfo = null;
+ GlyphMapping lastMapping = null;
while (posIter.hasNext()) {
final LeafPosition tbpNext = (LeafPosition) posIter.next();
if (tbpNext == null) {
continue; //Ignore elements without Positions
}
if (tbpNext.getLeafPos() != -1) {
- areaInfo = (AreaInfo) areaInfos.get(tbpNext.getLeafPos());
- if (lastAreaInfo == null
- || (areaInfo.font != lastAreaInfo.font)
- || (areaInfo.level != lastAreaInfo.level)) {
- if (lastAreaInfo != null) {
- addAreaInfoAreas(lastAreaInfo, wordSpaceCount,
- letterSpaceCount, firstAreaInfoIndex,
- lastAreaInfoIndex, realWidth, context);
+ mapping = mappings.get(tbpNext.getLeafPos());
+ if (lastMapping == null
+ || (mapping.font != lastMapping.font)
+ || (mapping.level != lastMapping.level)) {
+ if (lastMapping != null) {
+ addMappingAreas(lastMapping, wordSpaceCount,
+ letterSpaceCount, firstMappingIndex,
+ lastMappingIndex, realWidth, context);
}
- firstAreaInfoIndex = tbpNext.getLeafPos();
+ firstMappingIndex = tbpNext.getLeafPos();
wordSpaceCount = 0;
letterSpaceCount = 0;
realWidth = MinOptMax.ZERO;
}
- wordSpaceCount += areaInfo.wordSpaceCount;
- letterSpaceCount += areaInfo.letterSpaceCount;
- realWidth = realWidth.plus(areaInfo.areaIPD);
- lastAreaInfoIndex = tbpNext.getLeafPos();
- lastAreaInfo = areaInfo;
+ wordSpaceCount += mapping.wordSpaceCount;
+ letterSpaceCount += mapping.letterSpaceCount;
+ realWidth = realWidth.plus(mapping.areaIPD);
+ lastMappingIndex = tbpNext.getLeafPos();
+ lastMapping = mapping;
}
}
- if (lastAreaInfo != null) {
- addAreaInfoAreas(lastAreaInfo, wordSpaceCount, letterSpaceCount, firstAreaInfoIndex,
- lastAreaInfoIndex, realWidth, context);
+ if (lastMapping != null) {
+ addMappingAreas(lastMapping, wordSpaceCount, letterSpaceCount, firstMappingIndex,
+ lastMappingIndex, realWidth, context);
}
}
- private void addAreaInfoAreas(AreaInfo areaInfo, int wordSpaceCount, int letterSpaceCount,
- int firstAreaInfoIndex, int lastAreaInfoIndex,
+ private void addMappingAreas(GlyphMapping mapping, int wordSpaceCount, int letterSpaceCount,
+ int firstMappingIndex, int lastMappingIndex,
MinOptMax realWidth, LayoutContext context) {
// TODO: These two statements (if, for) were like this before my recent
- // changes. However, it seems as if they should use the AreaInfo from
- // firstAreaInfoIndex.. lastAreaInfoIndex rather than just the last areaInfo.
+ // changes. However, it seems as if they should use the GlyphMapping from
+ // firstMappingIndex.. lastMappingIndex rather than just the last mapping.
// This needs to be checked.
- int textLength = areaInfo.getWordLength();
- if (areaInfo.letterSpaceCount == textLength && !areaInfo.isHyphenated
+ int textLength = mapping.getWordLength();
+ if (mapping.letterSpaceCount == textLength && !mapping.isHyphenated
&& context.isLastArea()) {
// the line ends at a character like "/" or "-";
// remove the letter space after the last character
@@ -336,7 +259,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
letterSpaceCount--;
}
- for (int i = areaInfo.startIndex; i < areaInfo.breakIndex; i++) {
+ for (int i = mapping.startIndex; i < mapping.endIndex; i++) {
MinOptMax letterSpaceAdjustment = letterSpaceAdjustArray[i + 1];
if (letterSpaceAdjustment != null && letterSpaceAdjustment.isElastic()) {
letterSpaceCount++;
@@ -344,7 +267,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
}
// add hyphenation character if the last word is hyphenated
- if (context.isLastArea() && areaInfo.isHyphenated) {
+ if (context.isLastArea() && mapping.isHyphenated) {
realWidth = realWidth.plus(hyphIPD);
}
@@ -385,8 +308,8 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
totalAdjust = difference;
}
- TextArea textArea = new TextAreaBuilder(realWidth, totalAdjust, context, firstAreaInfoIndex,
- lastAreaInfoIndex, context.isLastArea(), areaInfo.font).build();
+ TextArea textArea = new TextAreaBuilder(realWidth, totalAdjust, context, firstMappingIndex,
+ lastMappingIndex, context.isLastArea(), mapping.font).build();
// wordSpaceDim is computed in relation to wordSpaceIPD.opt
// but the renderer needs to know the adjustment in relation
@@ -417,15 +340,15 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
private final MinOptMax width; // content ipd
private final int adjust; // content ipd adjustment
private final LayoutContext context; // layout context
- private final int firstIndex; // index of first AreaInfo
- private final int lastIndex; // index of last AreaInfo
+ private final int firstIndex; // index of first GlyphMapping
+ private final int lastIndex; // index of last GlyphMapping
private final boolean isLastArea; // true if last inline area in line area
private final Font font; // applicable font
// other, non-constructor state
private TextArea textArea; // text area being constructed
private int blockProgressionDimension; // calculated bpd
- private AreaInfo areaInfo; // current area info when iterating over words
+ private GlyphMapping mapping; // current mapping when iterating over words
private StringBuffer wordChars; // current word's character buffer
private int[] letterSpaceAdjust; // current word's letter space adjustments
private int letterSpaceAdjustIndex; // last written letter space adjustment index
@@ -442,8 +365,8 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
* @param width the MinOptMax width of the content
* @param adjust the total ipd adjustment with respect to the optimal width
* @param context the layout context
- * @param firstIndex the index of the first AreaInfo used for the TextArea
- * @param lastIndex the index of the last AreaInfo used for the TextArea
+ * @param firstIndex the index of the first GlyphMapping used for the TextArea
+ * @param lastIndex the index of the last GlyphMapping used for the TextArea
* @param isLastArea is this TextArea the last in a line?
* @param font Font to be used in this particular TextArea
*/
@@ -516,30 +439,30 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
* Sets the text of the TextArea, split into words and spaces.
*/
private void setText() {
- int areaInfoIndex = -1;
+ int mappingIndex = -1;
int wordCharLength = 0;
for (int wordIndex = firstIndex; wordIndex <= lastIndex; wordIndex++) {
- areaInfo = getAreaInfo(wordIndex);
- if (areaInfo.isSpace) {
+ mapping = getGlyphMapping(wordIndex);
+ if (mapping.isSpace) {
addSpaces();
} else {
- // areaInfo stores information about a word fragment
- if (areaInfoIndex == -1) {
+ // mapping stores information about a word fragment
+ if (mappingIndex == -1) {
// here starts a new word
- areaInfoIndex = wordIndex;
+ mappingIndex = wordIndex;
wordCharLength = 0;
}
- wordCharLength += areaInfo.getWordLength();
+ wordCharLength += mapping.getWordLength();
if (isWordEnd(wordIndex)) {
- addWord(areaInfoIndex, wordIndex, wordCharLength);
- areaInfoIndex = -1;
+ addWord(mappingIndex, wordIndex, wordCharLength);
+ mappingIndex = -1;
}
}
}
}
- private boolean isWordEnd(int areaInfoIndex) {
- return areaInfoIndex == lastIndex || getAreaInfo(areaInfoIndex + 1).isSpace;
+ private boolean isWordEnd(int mappingIndex) {
+ return mappingIndex == lastIndex || getGlyphMapping(mappingIndex + 1).isSpace;
}
/**
@@ -564,10 +487,10 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
// iterate over word's fragments, adding word chars (with bidi
// levels), letter space adjustments, and glyph position adjustments
for (int i = startIndex; i <= endIndex; i++) {
- AreaInfo wordAreaInfo = getAreaInfo(i);
- addWordChars(wordAreaInfo);
- addLetterAdjust(wordAreaInfo);
- if (addGlyphPositionAdjustments(wordAreaInfo)) {
+ GlyphMapping wordMapping = getGlyphMapping(i);
+ addWordChars(wordMapping);
+ addLetterAdjust(wordMapping);
+ if (addGlyphPositionAdjustments(wordMapping)) {
gposAdjusted = true;
}
}
@@ -617,7 +540,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
}
private boolean isHyphenated(int endIndex) {
- return isLastArea && endIndex == lastIndex && areaInfo.isHyphenated;
+ return isLastArea && endIndex == lastIndex && mapping.isHyphenated;
}
private void addHyphenationChar() {
@@ -632,21 +555,54 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
* (1) concatenate (possibly mapped) word characters to word character buffer;
* (2) concatenante (possibly mapped) word bidi levels to levels buffer;
* (3) update word's IPD with optimal IPD of fragment.
- * @param wordAreaInfo fragment info
+ * @param wordMapping fragment info
*/
- private void addWordChars(AreaInfo wordAreaInfo) {
- int s = wordAreaInfo.startIndex;
- int e = wordAreaInfo.breakIndex;
- if (foText.hasMapping(s, e)) {
- wordChars.append(foText.getMapping(s, e));
- addWordLevels(foText.getMappingBidiLevels(s, e));
+ private void addWordChars(GlyphMapping wordMapping) {
+ int s = wordMapping.startIndex;
+ int e = wordMapping.endIndex;
+ if (wordMapping.mapping != null) {
+ wordChars.append(wordMapping.mapping);
+ addWordLevels(getMappingBidiLevels(wordMapping));
} else {
for (int i = s; i < e; i++) {
wordChars.append(foText.charAt(i));
}
addWordLevels(foText.getBidiLevels(s, e));
}
- wordIPD += wordAreaInfo.areaIPD.getOpt();
+ wordIPD += wordMapping.areaIPD.getOpt();
+ }
+
+ /**
+ * Obtain bidirectional levels of mapping of characters over specific interval.
+ * @param start index in character buffer
+ * @param end index in character buffer
+ * @return a (possibly empty) array of bidi levels or null
+ * in case no bidi levels have been assigned
+ */
+ private int[] getMappingBidiLevels(GlyphMapping mapping) {
+ if (mapping.mapping != null) {
+ int nc = mapping.endIndex - mapping.startIndex;
+ int nm = mapping.mapping.length();
+ int[] la = foText.getBidiLevels(mapping.startIndex, mapping.endIndex);
+ if (la == null) {
+ return null;
+ } else if (nm == nc) { // mapping is same length as mapped range
+ return la;
+ } else if (nm > nc) { // mapping is longer than mapped range
+ int[] ma = new int[nm];
+ System.arraycopy(la, 0, ma, 0, la.length);
+ for (int i = la.length, n = ma.length, l = (i > 0) ? la[i - 1] : 0; i < n; i++) {
+ ma[i] = l;
+ }
+ return ma;
+ } else { // mapping is shorter than mapped range
+ int[] ma = new int[nm];
+ System.arraycopy(la, 0, ma, 0, ma.length);
+ return ma;
+ }
+ } else {
+ return foText.getBidiLevels(mapping.startIndex, mapping.endIndex);
+ }
}
/**
@@ -672,16 +628,16 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
/**
* Given a word area info associated with a word fragment,
* concatenate letter space adjustments for each (possibly mapped) character.
- * @param wordAreaInfo fragment info
+ * @param wordMapping fragment info
*/
- private void addLetterAdjust(AreaInfo wordAreaInfo) {
- int letterSpaceCount = wordAreaInfo.letterSpaceCount;
- int wordLength = wordAreaInfo.getWordLength();
+ private void addLetterAdjust(GlyphMapping wordMapping) {
+ int letterSpaceCount = wordMapping.letterSpaceCount;
+ int wordLength = wordMapping.getWordLength();
int taAdjust = textArea.getTextLetterSpaceAdjust();
for (int i = 0, n = wordLength; i < n; i++) {
int j = letterSpaceAdjustIndex + i;
if (j > 0) {
- int k = wordAreaInfo.startIndex + i;
+ int k = wordMapping.startIndex + i;
MinOptMax adj = (k < letterSpaceAdjustArray.length)
? letterSpaceAdjustArray [ k ] : null;
letterSpaceAdjust [ j ] = (adj == null) ? 0 : adj.getOpt();
@@ -697,14 +653,14 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
/**
* Given a word area info associated with a word fragment,
* concatenate glyph position adjustments for each (possibly mapped) character.
- * @param wordAreaInfo fragment info
+ * @param wordMapping fragment info
* @return true if an adjustment was non-zero
*/
- private boolean addGlyphPositionAdjustments(AreaInfo wordAreaInfo) {
+ private boolean addGlyphPositionAdjustments(GlyphMapping wordMapping) {
boolean adjusted = false;
- int[][] gpa = wordAreaInfo.gposAdjustments;
+ int[][] gpa = wordMapping.gposAdjustments;
int numAdjusts = (gpa != null) ? gpa.length : 0;
- int wordLength = wordAreaInfo.getWordLength();
+ int wordLength = wordMapping.getWordLength();
if (numAdjusts > 0) {
int need = gposAdjustmentsIndex + numAdjusts;
if (need <= gposAdjustments.length) {
@@ -733,7 +689,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
}
/**
- * The <code>AreaInfo</code> stores information about spaces.
+ * The <code>GlyphMapping</code> stores information about spaces.
* <p/>
* Add the spaces - except zero-width spaces - to the TextArea.
*/
@@ -743,16 +699,16 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
// divide the area info's allocated IPD evenly among the
// non-zero-width space characters
int numZeroWidthSpaces = 0;
- for (int i = areaInfo.startIndex; i < areaInfo.breakIndex; i++) {
+ for (int i = mapping.startIndex; i < mapping.endIndex; i++) {
char spaceChar = foText.charAt(i);
if (CharUtilities.isZeroWidthSpace(spaceChar)) {
numZeroWidthSpaces++;
}
}
- int numSpaces = areaInfo.breakIndex - areaInfo.startIndex - numZeroWidthSpaces;
- int spaceIPD = areaInfo.areaIPD.getOpt() / ((numSpaces > 0) ? numSpaces : 1);
+ int numSpaces = mapping.endIndex - mapping.startIndex - numZeroWidthSpaces;
+ int spaceIPD = mapping.areaIPD.getOpt() / ((numSpaces > 0) ? numSpaces : 1);
// add space area children, one for each non-zero-width space character
- for (int i = areaInfo.startIndex; i < areaInfo.breakIndex; i++) {
+ for (int i = mapping.startIndex; i < mapping.endIndex; i++) {
char spaceChar = foText.charAt(i);
int level = foText.bidiLevelAt(i);
if (!CharUtilities.isZeroWidthSpace(spaceChar)) {
@@ -766,39 +722,20 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
}
- private void addAreaInfo(AreaInfo ai) {
- addAreaInfo(areaInfos.size(), ai);
+ private void addGlyphMapping(GlyphMapping mapping) {
+ addGlyphMapping(mappings.size(), mapping);
}
- private void addAreaInfo(int index, AreaInfo ai) {
- areaInfos.add(index, ai);
+ private void addGlyphMapping(int index, GlyphMapping mapping) {
+ mappings.add(index, mapping);
}
- private void removeAreaInfo(int index) {
- areaInfos.remove(index);
+ private void removeGlyphMapping(int index) {
+ mappings.remove(index);
}
- private AreaInfo getAreaInfo(int index) {
- return (AreaInfo) areaInfos.get(index);
- }
-
- private void addToLetterAdjust(int index, int width) {
- if (letterSpaceAdjustArray[index] == null) {
- letterSpaceAdjustArray[index] = MinOptMax.getInstance(width);
- } else {
- letterSpaceAdjustArray[index] = letterSpaceAdjustArray[index].plus(width);
- }
- }
-
- /**
- * Indicates whether a character is a space in terms of this layout manager.
- * @param ch the character
- * @return true if it's a space
- */
- private static boolean isSpace(final char ch) {
- return ch == CharUtilities.SPACE
- || CharUtilities.isNonBreakableSpace(ch)
- || CharUtilities.isFixedWidthSpace(ch);
+ private GlyphMapping getGlyphMapping(int index) {
+ return mappings.get(index);
}
/** {@inheritDoc} */
@@ -810,8 +747,8 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
final List returnList = new LinkedList();
KnuthSequence sequence = new InlineKnuthSequence();
- AreaInfo areaInfo = null;
- AreaInfo prevAreaInfo = null;
+ GlyphMapping mapping = null;
+ GlyphMapping prevMapping = null;
returnList.add(sequence);
if (LOG.isDebugEnabled()) {
@@ -857,24 +794,24 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
}
if (inWord) {
if (breakOpportunity
- || TextLayoutManager.isSpace(ch)
+ || GlyphMapping.isSpace(ch)
|| CharUtilities.isExplicitBreak(ch)
|| ((prevLevel != -1) && (level != prevLevel))) {
// this.foText.charAt(lastIndex) == CharUtilities.SOFT_HYPHEN
- prevAreaInfo = processWord(alignment, sequence, prevAreaInfo, ch,
+ prevMapping = processWord(alignment, sequence, prevMapping, ch,
breakOpportunity, true, prevLevel);
}
} else if (inWhitespace) {
if (ch != CharUtilities.SPACE || breakOpportunity) {
- prevAreaInfo = processWhitespace(alignment, sequence,
+ prevMapping = processWhitespace(alignment, sequence,
breakOpportunity, prevLevel);
}
} else {
- if (areaInfo != null) {
- prevAreaInfo = areaInfo;
- processLeftoverAreaInfo(alignment, sequence, areaInfo,
+ if (mapping != null) {
+ prevMapping = mapping;
+ processLeftoverGlyphMapping(alignment, sequence, mapping,
ch == CharUtilities.SPACE || breakOpportunity);
- areaInfo = null;
+ mapping = null;
}
if (breakAction == LineBreakStatus.EXPLICIT_BREAK) {
sequence = processLinebreak(returnList, sequence);
@@ -888,15 +825,15 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
this.foText, this);
font.mapChar(ch);
// preserved space or non-breaking space:
- // create the AreaInfo object
- areaInfo = new AreaInfo(nextStart, nextStart + 1, 1, 0, wordSpaceIPD, false, true,
+ // create the GlyphMapping object
+ mapping = new GlyphMapping(nextStart, nextStart + 1, 1, 0, wordSpaceIPD, false, true,
breakOpportunity, spaceFont, level, null);
thisStart = nextStart + 1;
} else if (CharUtilities.isFixedWidthSpace(ch) || CharUtilities.isZeroWidthSpace(ch)) {
- // create the AreaInfo object
+ // create the GlyphMapping object
Font font = FontSelector.selectFontForCharacterInText(ch, foText, this);
MinOptMax ipd = MinOptMax.getInstance(font.getCharWidth(ch));
- areaInfo = new AreaInfo(nextStart, nextStart + 1, 0, 0, ipd, false, true,
+ mapping = new GlyphMapping(nextStart, nextStart + 1, 0, 0, ipd, false, true,
breakOpportunity, font, level, null);
thisStart = nextStart + 1;
} else if (CharUtilities.isExplicitBreak(ch)) {
@@ -904,7 +841,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
thisStart = nextStart + 1;
}
- inWord = !TextLayoutManager.isSpace(ch) && !CharUtilities.isExplicitBreak(ch);
+ inWord = !GlyphMapping.isSpace(ch) && !CharUtilities.isExplicitBreak(ch);
inWhitespace = ch == CharUtilities.SPACE
&& foText.getWhitespaceTreatment() != Constants.EN_PRESERVE;
prevLevel = level;
@@ -913,11 +850,11 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
// Process any last elements
if (inWord) {
- processWord(alignment, sequence, prevAreaInfo, ch, false, false, prevLevel);
+ processWord(alignment, sequence, prevMapping, ch, false, false, prevLevel);
} else if (inWhitespace) {
processWhitespace(alignment, sequence, !keepTogether, prevLevel);
- } else if (areaInfo != null) {
- processLeftoverAreaInfo(alignment, sequence, areaInfo,
+ } else if (mapping != null) {
+ processLeftoverGlyphMapping(alignment, sequence, mapping,
ch == CharUtilities.ZERO_WIDTH_SPACE);
} else if (CharUtilities.isExplicitBreak(ch)) {
this.processLinebreak(returnList, sequence);
@@ -948,15 +885,14 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
return sequence;
}
- private void processLeftoverAreaInfo(int alignment,
- KnuthSequence sequence, AreaInfo areaInfo,
- boolean breakOpportunityAfter) {
- addAreaInfo(areaInfo);
- areaInfo.breakOppAfter = breakOpportunityAfter;
- addElementsForASpace(sequence, alignment, areaInfo, areaInfos.size() - 1);
+ private void processLeftoverGlyphMapping(int alignment, KnuthSequence sequence,
+ GlyphMapping mapping, boolean breakOpportunityAfter) {
+ addGlyphMapping(mapping);
+ mapping.breakOppAfter = breakOpportunityAfter;
+ addElementsForASpace(sequence, alignment, mapping, mappings.size() - 1);
}
- private AreaInfo processWhitespace(final int alignment,
+ private GlyphMapping processWhitespace(final int alignment,
final KnuthSequence sequence, final boolean breakOpportunity, int level) {
if (LOG.isDebugEnabled()) {
@@ -964,209 +900,24 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
}
// End of whitespace
- // create the AreaInfo object
+ // create the GlyphMapping object
assert nextStart >= thisStart;
- AreaInfo areaInfo = new AreaInfo(
- thisStart, nextStart, nextStart - thisStart, 0,
+ GlyphMapping mapping = new GlyphMapping(
+ thisStart, nextStart, nextStart - thisStart, 0,
wordSpaceIPD.mult(nextStart - thisStart),
false, true, breakOpportunity, spaceFont, level, null);
- addAreaInfo(areaInfo);
+ addGlyphMapping(mapping);
// create the elements
- addElementsForASpace(sequence, alignment, areaInfo, areaInfos.size() - 1);
+ addElementsForASpace(sequence, alignment, mapping, mappings.size() - 1);
thisStart = nextStart;
- return areaInfo;
- }
-
- private AreaInfo processWordMapping(
- int lastIndex, final Font font, AreaInfo prevAreaInfo, final char breakOpportunityChar,
- final boolean endsWithHyphen, int level) {
- int s = this.thisStart; // start index of word in FOText character buffer
- int e = lastIndex; // end index of word in FOText character buffer
- int nLS = 0; // # of letter spaces
- String script = foText.getScript();
- String language = foText.getLanguage();
-
- if (LOG.isDebugEnabled()) {
- LOG.debug("PW: [" + thisStart + "," + lastIndex + "]: {"
- + " +M"
- + ", level = " + level
- + " }");
- }
-
- // 1. extract unmapped character sequence
- CharSequence ics = foText.subSequence(s, e);
-
- // 2. if script is not specified (by FO property) or it is specified as 'auto',
- // then compute dominant script
- if ((script == null) || "auto".equals(script)) {
- script = CharScript.scriptTagFromCode(CharScript.dominantScript(ics));
- }
- if ((language == null) || "none".equals(language)) {
- language = "dflt";
- }
-
- // 3. perform mapping of chars to glyphs ... to glyphs ... to chars
- CharSequence mcs = font.performSubstitution(ics, script, language);
-
- // 4. compute glyph position adjustments on (substituted) characters
- int[][] gpa;
- if (font.performsPositioning()) {
- // handle GPOS adjustments
- gpa = font.performPositioning(mcs, script, language);
- } else if (font.hasKerning()) {
- // handle standard (non-GPOS) kerning adjustments
- gpa = getKerningAdjustments(mcs, font);
- } else {
- gpa = null;
- }
-
- // 5. reorder combining marks so that they precede (within the mapped char sequence) the
- // base to which they are applied; N.B. position adjustments (gpa) are reordered in place
- mcs = font.reorderCombiningMarks(mcs, gpa, script, language);
-
- // 6. if mapped sequence differs from input sequence, then memoize mapped sequence
- if (!CharUtilities.isSameSequence(mcs, ics)) {
- foText.addMapping(s, e, mcs);
- }
-
- // 7. compute word ipd based on final position adjustments
- MinOptMax ipd = MinOptMax.ZERO;
- for (int i = 0, n = mcs.length(); i < n; i++) {
- int c = mcs.charAt(i);
- // TODO !BMP
- int w = font.getCharWidth(c);
- if (w < 0) {
- w = 0;
- }
- if (gpa != null) {
- w += gpa [ i ] [ GlyphPositioningTable.Value.IDX_X_ADVANCE ];
- }
- ipd = ipd.plus(w);
- }
-
- // [TBD] - handle letter spacing
-
- return new AreaInfo(
- s, e, 0, nLS, ipd, endsWithHyphen, false,
- breakOpportunityChar != 0, font, level, gpa);
- }
-
- /**
- * Given a mapped character sequence MCS, obtain glyph position adjustments
- * from the font's kerning data.
- * @param mcs mapped character sequence
- * @param font applicable font
- * @return glyph position adjustments (or null if no kerning)
- */
- private int[][] getKerningAdjustments(CharSequence mcs, final Font font) {
- int nc = mcs.length();
- // extract kerning array
- int[] ka = new int [ nc ]; // kerning array
- for (int i = 0, n = nc, cPrev = -1; i < n; i++) {
- int c = mcs.charAt(i);
- // TODO !BMP
- if (cPrev >= 0) {
- ka[i] = font.getKernValue(cPrev, c);
- }
- cPrev = c;
- }
- // was there a non-zero kerning?
- boolean hasKerning = false;
- for (int i = 0, n = nc; i < n; i++) {
- if (ka[i] != 0) {
- hasKerning = true;
- break;
- }
- }
- // if non-zero kerning, then create and return glyph position adjustment array
- if (hasKerning) {
- int[][] gpa = new int [ nc ] [ 4 ];
- for (int i = 0, n = nc; i < n; i++) {
- if (i > 0) {
- gpa [ i - 1 ] [ GlyphPositioningTable.Value.IDX_X_ADVANCE ] = ka [ i ];
- }
- }
- return gpa;
- } else {
- return null;
- }
+ return mapping;
}
- private AreaInfo processWordNoMapping(int lastIndex, final Font font, AreaInfo prevAreaInfo,
- final char breakOpportunityChar, final boolean endsWithHyphen, int level) {
- boolean kerning = font.hasKerning();
- MinOptMax wordIPD = MinOptMax.ZERO;
-
- if (LOG.isDebugEnabled()) {
- LOG.debug("PW: [" + thisStart + "," + lastIndex + "]: {"
- + " -M"
- + ", level = " + level
- + " }");
- }
-
- for (int i = thisStart; i < lastIndex; i++) {
- char currentChar = foText.charAt(i);
-
- //character width
- int charWidth = font.getCharWidth(currentChar);
- wordIPD = wordIPD.plus(charWidth);
-
- //kerning
- if (kerning) {
- int kern = 0;
- if (i > thisStart) {
- char previousChar = foText.charAt(i - 1);
- kern = font.getKernValue(previousChar, currentChar);
- } else if (prevAreaInfo != null
- && !prevAreaInfo.isSpace && prevAreaInfo.breakIndex > 0) {
- char previousChar = foText.charAt(prevAreaInfo.breakIndex - 1);
- kern = font.getKernValue(previousChar, currentChar);
- }
- if (kern != 0) {
- addToLetterAdjust(i, kern);
- wordIPD = wordIPD.plus(kern);
- }
- }
- }
- if (kerning
- && (breakOpportunityChar != 0)
- && !TextLayoutManager.isSpace(breakOpportunityChar)
- && lastIndex > 0
- && endsWithHyphen) {
- int kern = font.getKernValue(foText.charAt(lastIndex - 1), breakOpportunityChar);
- if (kern != 0) {
- addToLetterAdjust(lastIndex, kern);
- //TODO: add kern to wordIPD?
- }
- }
- // shy+chars at start of word: wordLength == 0 && breakOpportunity
- // shy only characters in word: wordLength == 0 && !breakOpportunity
- int wordLength = lastIndex - thisStart;
- int letterSpaces = 0;
- if (wordLength != 0) {
- letterSpaces = wordLength - 1;
- // if there is a break opportunity and the next one (break character)
- // is not a space, it could be used as a line end;
- // add one more letter space, in case other text follows
- if ((breakOpportunityChar != 0) && !TextLayoutManager.isSpace(breakOpportunityChar)) {
- letterSpaces++;
- }
- }
- assert letterSpaces >= 0;
- wordIPD = wordIPD.plus(letterSpaceIPD.mult(letterSpaces));
-
- // create and return the AreaInfo object
- return new AreaInfo(thisStart, lastIndex, 0,
- letterSpaces, wordIPD,
- endsWithHyphen,
- false, breakOpportunityChar != 0, font, level, null);
- }
-
- private AreaInfo processWord(final int alignment, final KnuthSequence sequence,
- AreaInfo prevAreaInfo, final char ch, final boolean breakOpportunity,
+ private GlyphMapping processWord(final int alignment, final KnuthSequence sequence,
+ GlyphMapping prevMapping, final char ch, final boolean breakOpportunity,
final boolean checkEndsWithHyphen, int level) {
//Word boundary found, process widths and kerning
@@ -1178,23 +929,20 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
&& foText.charAt(lastIndex) == CharUtilities.SOFT_HYPHEN;
Font font = FontSelector.selectFontForCharactersInText(
foText, thisStart, lastIndex, foText, this);
- AreaInfo areaInfo;
- if (font.performsSubstitution() || font.performsPositioning()) {
- areaInfo = processWordMapping(
- lastIndex, font, prevAreaInfo, breakOpportunity ? ch : 0, endsWithHyphen, level);
- } else {
- areaInfo = processWordNoMapping(
- lastIndex, font, prevAreaInfo, breakOpportunity ? ch : 0, endsWithHyphen, level);
- }
- prevAreaInfo = areaInfo;
- addAreaInfo(areaInfo);
+ char breakOpportunityChar = breakOpportunity ? ch : 0;
+ char precedingChar = prevMapping != null && !prevMapping.isSpace
+ && prevMapping.endIndex > 0 ? foText.charAt(prevMapping.endIndex - 1) : 0;
+ GlyphMapping mapping = GlyphMapping.doGlyphMapping(foText, thisStart, lastIndex, font,
+ letterSpaceIPD, letterSpaceAdjustArray, precedingChar, breakOpportunityChar, endsWithHyphen, level);
+ prevMapping = mapping;
+ addGlyphMapping(mapping);
tempStart = nextStart;
//add the elements
- addElementsForAWordFragment(sequence, alignment, areaInfo, areaInfos.size() - 1);
+ addElementsForAWordFragment(sequence, alignment, mapping, mappings.size() - 1);
thisStart = nextStart;
- return prevAreaInfo;
+ return prevMapping;
}
/** {@inheritDoc} */
@@ -1214,9 +962,9 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
int index = leafPos.getLeafPos();
//element could refer to '-1' position, for non-collapsed spaces (?)
if (index > -1) {
- AreaInfo areaInfo = getAreaInfo(index);
- areaInfo.letterSpaceCount++;
- areaInfo.addToAreaIPD(letterSpaceIPD);
+ GlyphMapping mapping = getGlyphMapping(index);
+ mapping.letterSpaceCount++;
+ mapping.addToAreaIPD(letterSpaceIPD);
if (TextLayoutManager.BREAK_CHARS.indexOf(foText.charAt(tempStart - 1)) >= 0) {
// the last character could be used as a line break
// append new elements to oldList
@@ -1227,13 +975,13 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
} else if (letterSpaceIPD.isStiff()) {
// constant letter space: replace the box
// give it the unwrapped position of the replaced element
- oldListIterator.set(new KnuthInlineBox(areaInfo.areaIPD.getOpt(),
+ oldListIterator.set(new KnuthInlineBox(mapping.areaIPD.getOpt(),
alignmentContext, pos, false));
} else {
// adjustable letter space: replace the glue
oldListIterator.next(); // this would return the penalty element
oldListIterator.next(); // this would return the glue element
- oldListIterator.set(new KnuthGlue(letterSpaceIPD.mult(areaInfo.letterSpaceCount),
+ oldListIterator.set(new KnuthGlue(letterSpaceIPD.mult(mapping.letterSpaceCount),
auxiliaryPosition, true));
}
}
@@ -1242,26 +990,26 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
/** {@inheritDoc} */
public void hyphenate(Position pos, HyphContext hyphContext) {
- AreaInfo areaInfo = getAreaInfo(((LeafPosition) pos).getLeafPos() + changeOffset);
- int startIndex = areaInfo.startIndex;
+ GlyphMapping mapping = getGlyphMapping(((LeafPosition) pos).getLeafPos() + changeOffset);
+ int startIndex = mapping.startIndex;
int stopIndex;
boolean nothingChanged = true;
- Font font = areaInfo.font;
+ Font font = mapping.font;
- while (startIndex < areaInfo.breakIndex) {
+ while (startIndex < mapping.endIndex) {
MinOptMax newIPD = MinOptMax.ZERO;
boolean hyphenFollows;
stopIndex = startIndex + hyphContext.getNextHyphPoint();
- if (hyphContext.hasMoreHyphPoints() && stopIndex <= areaInfo.breakIndex) {
+ if (hyphContext.hasMoreHyphPoints() && stopIndex <= mapping.endIndex) {
// stopIndex is the index of the first character
// after a hyphenation point
hyphenFollows = true;
} else {
// there are no more hyphenation points,
- // or the next one is after areaInfo.breakIndex
+ // or the next one is after mapping.breakIndex
hyphenFollows = false;
- stopIndex = areaInfo.breakIndex;
+ stopIndex = mapping.endIndex;
}
hyphContext.updateOffset(stopIndex - startIndex);
@@ -1286,18 +1034,18 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
// add letter spaces
boolean isWordEnd
- = (stopIndex == areaInfo.breakIndex)
- && (areaInfo.letterSpaceCount < areaInfo.getWordLength());
+ = (stopIndex == mapping.endIndex)
+ && (mapping.letterSpaceCount < mapping.getWordLength());
int letterSpaceCount = isWordEnd ? stopIndex - startIndex - 1 : stopIndex - startIndex;
assert letterSpaceCount >= 0;
newIPD = newIPD.plus(letterSpaceIPD.mult(letterSpaceCount));
- if (!(nothingChanged && stopIndex == areaInfo.breakIndex && !hyphenFollows)) {
- // the new AreaInfo object is not equal to the old one
+ if (!(nothingChanged && stopIndex == mapping.endIndex && !hyphenFollows)) {
+ // the new GlyphMapping object is not equal to the old one
changeList.add(
new PendingChange(
- new AreaInfo(startIndex, stopIndex, 0,
+ new GlyphMapping(startIndex, stopIndex, 0,
letterSpaceCount, newIPD, hyphenFollows,
false, false, font, -1, null),
((LeafPosition) pos).getLeafPos() + changeOffset));
@@ -1324,7 +1072,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
return false;
}
- // Find the first and last positions in oldList that point to an AreaInfo
+ // Find the first and last positions in oldList that point to a GlyphMapping
// (i.e. getLeafPos() != -1)
LeafPosition startPos = null;
LeafPosition endPos = null;
@@ -1349,8 +1097,8 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
returnedIndices[0] = (startPos != null ? startPos.getLeafPos() : -1) + changeOffset;
returnedIndices[1] = (endPos != null ? endPos.getLeafPos() : -1) + changeOffset;
- int areaInfosAdded = 0;
- int areaInfosRemoved = 0;
+ int mappingsAdded = 0;
+ int mappingsRemoved = 0;
if (!changeList.isEmpty()) {
int oldIndex = -1;
@@ -1360,24 +1108,24 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
while (changeListIterator.hasNext()) {
currChange = (PendingChange) changeListIterator.next();
if (currChange.index == oldIndex) {
- areaInfosAdded++;
- changeIndex = currChange.index + areaInfosAdded - areaInfosRemoved;
+ mappingsAdded++;
+ changeIndex = currChange.index + mappingsAdded - mappingsRemoved;
} else {
- areaInfosRemoved++;
- areaInfosAdded++;
+ mappingsRemoved++;
+ mappingsAdded++;
oldIndex = currChange.index;
- changeIndex = currChange.index + areaInfosAdded - areaInfosRemoved;
- removeAreaInfo(changeIndex);
+ changeIndex = currChange.index + mappingsAdded - mappingsRemoved;
+ removeGlyphMapping(changeIndex);
}
- addAreaInfo(changeIndex, currChange.areaInfo);
+ addGlyphMapping(changeIndex, currChange.mapping);
}
changeList.clear();
}
// increase the end index for getChangedKnuthElements()
- returnedIndices[1] += (areaInfosAdded - areaInfosRemoved);
+ returnedIndices[1] += (mappingsAdded - mappingsRemoved);
// increase offset to use for subsequent paragraphs
- changeOffset += (areaInfosAdded - areaInfosRemoved);
+ changeOffset += (mappingsAdded - mappingsRemoved);
return hasChanged;
}
@@ -1391,16 +1139,16 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
final LinkedList returnList = new LinkedList();
for (; returnedIndices[0] <= returnedIndices[1]; returnedIndices[0]++) {
- AreaInfo areaInfo = getAreaInfo(returnedIndices[0]);
- if (areaInfo.wordSpaceCount == 0) {
- // areaInfo refers either to a word or a word fragment
- addElementsForAWordFragment(returnList, alignment, areaInfo, returnedIndices[0]);
+ GlyphMapping mapping = getGlyphMapping(returnedIndices[0]);
+ if (mapping.wordSpaceCount == 0) {
+ // mapping refers either to a word or a word fragment
+ addElementsForAWordFragment(returnList, alignment, mapping, returnedIndices[0]);
} else {
- // areaInfo refers to a space
- addElementsForASpace(returnList, alignment, areaInfo, returnedIndices[0]);
+ // mapping refers to a space
+ addElementsForASpace(returnList, alignment, mapping, returnedIndices[0]);
}
}
- setFinished(returnedIndices[0] == areaInfos.size() - 1);
+ setFinished(returnedIndices[0] == mappings.size() - 1);
//ElementListObserver.observe(returnList, "text-changed", null);
return returnList;
}
@@ -1409,9 +1157,9 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
public String getWordChars(Position pos) {
int leafValue = ((LeafPosition) pos).getLeafPos() + changeOffset;
if (leafValue != -1) {
- AreaInfo areaInfo = getAreaInfo(leafValue);
- StringBuffer buffer = new StringBuffer(areaInfo.getWordLength());
- for (int i = areaInfo.startIndex; i < areaInfo.breakIndex; i++) {
+ GlyphMapping mapping = getGlyphMapping(leafValue);
+ StringBuffer buffer = new StringBuffer(mapping.getWordLength());
+ for (int i = mapping.startIndex; i < mapping.endIndex; i++) {
buffer.append(foText.charAt(i));
}
return buffer.toString();
@@ -1420,41 +1168,39 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
}
}
- private void addElementsForASpace(List baseList, int alignment, AreaInfo areaInfo,
+ private void addElementsForASpace(List baseList, int alignment, GlyphMapping mapping,
int leafValue) {
LeafPosition mainPosition = new LeafPosition(this, leafValue);
- if (!areaInfo.breakOppAfter) {
+ if (!mapping.breakOppAfter) {
// a non-breaking space
if (alignment == Constants.EN_JUSTIFY) {
// the space can stretch and shrink, and must be preserved
// when starting a line
baseList.add(makeAuxiliaryZeroWidthBox());
baseList.add(makeZeroWidthPenalty(KnuthElement.INFINITE));
- baseList.add(new KnuthGlue(areaInfo.areaIPD, mainPosition, false));
+ baseList.add(new KnuthGlue(mapping.areaIPD, mainPosition, false));
} else {
// the space does not need to stretch or shrink, and must be
// preserved when starting a line
- baseList.add(new KnuthInlineBox(areaInfo.areaIPD.getOpt(), null, mainPosition,
+ baseList.add(new KnuthInlineBox(mapping.areaIPD.getOpt(), null, mainPosition,
true));
}
} else {
- if (foText.charAt(areaInfo.startIndex) != CharUtilities.SPACE
+ if (foText.charAt(mapping.startIndex) != CharUtilities.SPACE
|| foText.getWhitespaceTreatment() == Constants.EN_PRESERVE) {
// a breaking space that needs to be preserved
- baseList
- .addAll(getElementsForBreakingSpace(alignment, areaInfo, auxiliaryPosition, 0,
- mainPosition, areaInfo.areaIPD.getOpt(), true));
+ baseList.addAll(getElementsForBreakingSpace(alignment, mapping, auxiliaryPosition, 0,
+ mainPosition, mapping.areaIPD.getOpt(), true));
} else {
// a (possible block) of breaking spaces
- baseList
- .addAll(getElementsForBreakingSpace(alignment, areaInfo, mainPosition,
- areaInfo.areaIPD.getOpt(), auxiliaryPosition, 0, false));
+ baseList.addAll(getElementsForBreakingSpace(alignment, mapping, mainPosition,
+ mapping.areaIPD.getOpt(), auxiliaryPosition, 0, false));
}
}
}
- private List getElementsForBreakingSpace(int alignment, AreaInfo areaInfo, Position pos2,
+ private List getElementsForBreakingSpace(int alignment, GlyphMapping mapping, Position pos2,
int p2WidthOffset, Position pos3,
int p3WidthOffset, boolean skipZeroCheck) {
List elements = new ArrayList();
@@ -1504,7 +1250,7 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
elements.add(g);
elements.add(makeZeroWidthPenalty(0));
g = new KnuthGlue(
- areaInfo.areaIPD.getOpt(),
+ mapping.areaIPD.getOpt(),
-3 * LineLayoutManager.DEFAULT_SPACE_WIDTH, 0, pos2, false);
elements.add(g);
}
@@ -1513,25 +1259,24 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
case EN_JUSTIFY:
// justified text:
// the stretch and shrink depends on the space width
- elements.addAll(getElementsForJustifiedText(areaInfo, pos2, p2WidthOffset, pos3,
- p3WidthOffset, skipZeroCheck, areaInfo.areaIPD.getShrink()));
+ elements.addAll(getElementsForJustifiedText(mapping, pos2, p2WidthOffset, pos3,
+ p3WidthOffset, skipZeroCheck, mapping.areaIPD.getShrink()));
break;
default:
// last line justified, the other lines unjustified:
// use only the space stretch
- elements.addAll(getElementsForJustifiedText(areaInfo, pos2, p2WidthOffset, pos3,
+ elements.addAll(getElementsForJustifiedText(mapping, pos2, p2WidthOffset, pos3,
p3WidthOffset, skipZeroCheck, 0));
}
return elements;
}
- private List getElementsForJustifiedText(
- AreaInfo areaInfo, Position pos2, int p2WidthOffset,
- Position pos3, int p3WidthOffset, boolean skipZeroCheck,
- int shrinkability) {
+ private List getElementsForJustifiedText(GlyphMapping mapping, Position pos2, int p2WidthOffset,
+ Position pos3, int p3WidthOffset, boolean skipZeroCheck,
+ int shrinkability) {
- int stretchability = areaInfo.areaIPD.getStretch();
+ int stretchability = mapping.areaIPD.getStretch();
List elements = new ArrayList();
if (skipZeroCheck || lineStartBAP != 0 || lineEndBAP != 0) {
@@ -1543,34 +1288,34 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
elements.add(makeZeroWidthPenalty(KnuthElement.INFINITE));
elements.add(new KnuthGlue(lineStartBAP + p3WidthOffset, 0, 0, pos3, false));
} else {
- elements.add(new KnuthGlue(areaInfo.areaIPD.getOpt(), stretchability, shrinkability,
+ elements.add(new KnuthGlue(mapping.areaIPD.getOpt(), stretchability, shrinkability,
pos2, false));
}
return elements;
}
- private void addElementsForAWordFragment(List baseList, int alignment, AreaInfo areaInfo,
+ private void addElementsForAWordFragment(List baseList, int alignment, GlyphMapping mapping,
int leafValue) {
LeafPosition mainPosition = new LeafPosition(this, leafValue);
// if the last character of the word fragment is '-' or '/',
// the fragment could end a line; in this case, it loses one
// of its letter spaces;
- boolean suppressibleLetterSpace = areaInfo.breakOppAfter && !areaInfo.isHyphenated;
+ boolean suppressibleLetterSpace = mapping.breakOppAfter && !mapping.isHyphenated;
if (letterSpaceIPD.isStiff()) {
// constant letter spacing
baseList.add(new KnuthInlineBox(suppressibleLetterSpace
- ? areaInfo.areaIPD.getOpt() - letterSpaceIPD.getOpt()
- : areaInfo.areaIPD.getOpt(),
+ ? mapping.areaIPD.getOpt() - letterSpaceIPD.getOpt()
+ : mapping.areaIPD.getOpt(),
alignmentContext, notifyPos(mainPosition), false));
} else {
// adjustable letter spacing
int unsuppressibleLetterSpaces = suppressibleLetterSpace
- ? areaInfo.letterSpaceCount - 1
- : areaInfo.letterSpaceCount;
- baseList.add(new KnuthInlineBox(areaInfo.areaIPD.getOpt()
- - areaInfo.letterSpaceCount * letterSpaceIPD.getOpt(),
+ ? mapping.letterSpaceCount - 1
+ : mapping.letterSpaceCount;
+ baseList.add(new KnuthInlineBox(mapping.areaIPD.getOpt()
+ - mapping.letterSpaceCount * letterSpaceIPD.getOpt(),
alignmentContext, notifyPos(mainPosition), false));
baseList.add(makeZeroWidthPenalty(KnuthElement.INFINITE));
baseList.add(new KnuthGlue(letterSpaceIPD.mult(unsuppressibleLetterSpaces),
@@ -1580,19 +1325,19 @@ public class TextLayoutManager extends LeafNodeLayoutManager {
// extra-elements if the word fragment is the end of a syllable,
// or it ends with a character that can be used as a line break
- if (areaInfo.isHyphenated) {
+ if (mapping.isHyphenated) {
MinOptMax widthIfNoBreakOccurs = null;
- if (areaInfo.breakIndex < foText.length()) {
+ if (mapping.endIndex < foText.length()) {
//Add in kerning in no-break condition
- widthIfNoBreakOccurs = letterSpaceAdjustArray[areaInfo.breakIndex];
+ widthIfNoBreakOccurs = letterSpaceAdjustArray[mapping.endIndex];
}
- //if (areaInfo.breakIndex)
+ //if (mapping.breakIndex)
// the word fragment ends at the end of a syllable:
// if a break occurs the content width increases,
// otherwise nothing happens
addElementsForAHyphen(baseList, alignment, hyphIPD, widthIfNoBreakOccurs,
- areaInfo.breakOppAfter && areaInfo.isHyphenated);
+ mapping.breakOppAfter && mapping.isHyphenated);
} else if (suppressibleLetterSpace) {
// the word fragment ends with a character that acts as a hyphen
// if a break occurs the width does not increase,
diff --git a/src/java/org/apache/fop/render/AbstractGenericSVGHandler.java b/src/java/org/apache/fop/render/AbstractGenericSVGHandler.java
index 392eb875e..983a5ad90 100644
--- a/src/java/org/apache/fop/render/AbstractGenericSVGHandler.java
+++ b/src/java/org/apache/fop/render/AbstractGenericSVGHandler.java
@@ -31,6 +31,7 @@ import org.apache.batik.bridge.GVTBuilder;
import org.apache.batik.dom.AbstractDocument;
import org.apache.batik.dom.svg.SVGDOMImplementation;
import org.apache.batik.gvt.GraphicsNode;
+import org.apache.batik.gvt.font.DefaultFontFamilyResolver;
import org.apache.xmlgraphics.java2d.Graphics2DImagePainter;
@@ -122,7 +123,8 @@ public abstract class AbstractGenericSVGHandler implements XMLHandler, RendererC
//Prepare
FOUserAgent userAgent = rendererContext.getUserAgent();
- SVGUserAgent svgUserAgent = new SVGUserAgent(userAgent, new AffineTransform());
+ SVGUserAgent svgUserAgent = new SVGUserAgent(userAgent, DefaultFontFamilyResolver.SINGLETON,
+ new AffineTransform());
//Create Batik BridgeContext
final BridgeContext bridgeContext = new BridgeContext(svgUserAgent);
diff --git a/src/java/org/apache/fop/render/afp/AFPSVGHandler.java b/src/java/org/apache/fop/render/afp/AFPSVGHandler.java
index 4549a26af..ccb4cc678 100644
--- a/src/java/org/apache/fop/render/afp/AFPSVGHandler.java
+++ b/src/java/org/apache/fop/render/afp/AFPSVGHandler.java
@@ -29,6 +29,7 @@ import org.w3c.dom.Document;
import org.apache.batik.bridge.BridgeContext;
import org.apache.batik.dom.svg.SVGDOMImplementation;
import org.apache.batik.gvt.GraphicsNode;
+import org.apache.batik.gvt.font.DefaultFontFamilyResolver;
import org.apache.xmlgraphics.image.loader.ImageManager;
import org.apache.xmlgraphics.image.loader.ImageSessionContext;
@@ -43,6 +44,7 @@ import org.apache.fop.afp.AFPResourceInfo;
import org.apache.fop.afp.AFPResourceManager;
import org.apache.fop.afp.AFPUnitConverter;
import org.apache.fop.afp.svg.AFPBridgeContext;
+import org.apache.fop.afp.svg.AFPFontFamilyResolver;
import org.apache.fop.apps.FOUserAgent;
import org.apache.fop.fonts.FontInfo;
import org.apache.fop.image.loader.batik.BatikUtil;
@@ -53,6 +55,7 @@ import org.apache.fop.render.RendererContext;
import org.apache.fop.render.RendererContext.RendererContextWrapper;
import org.apache.fop.svg.SVGEventProducer;
import org.apache.fop.svg.SVGUserAgent;
+import org.apache.fop.svg.font.AggregatingFontFamilyResolver;
/**
* AFP XML handler for SVG. Uses Apache Batik for SVG processing.
@@ -196,15 +199,13 @@ public class AFPSVGHandler extends AbstractGenericSVGHandler {
*/
public static BridgeContext createBridgeContext(FOUserAgent userAgent, AFPGraphics2D g2d) {
ImageManager imageManager = userAgent.getImageManager();
-
- SVGUserAgent svgUserAgent
- = new SVGUserAgent(userAgent, new AffineTransform());
-
- ImageSessionContext imageSessionContext = userAgent.getImageSessionContext();
-
FontInfo fontInfo = g2d.getFontInfo();
+ SVGUserAgent svgUserAgent = new SVGUserAgent(userAgent, new AggregatingFontFamilyResolver(
+ new AFPFontFamilyResolver(fontInfo, userAgent.getEventBroadcaster()), DefaultFontFamilyResolver.SINGLETON),
+ new AffineTransform());
+ ImageSessionContext imageSessionContext = userAgent.getImageSessionContext();
return new AFPBridgeContext(svgUserAgent, fontInfo, imageManager, imageSessionContext,
- new AffineTransform(), g2d);
+ new AffineTransform(), g2d, userAgent.getEventBroadcaster());
}
/** {@inheritDoc} */
diff --git a/src/java/org/apache/fop/render/java2d/CustomFontMetricsMapper.java b/src/java/org/apache/fop/render/java2d/CustomFontMetricsMapper.java
index 2f8865b14..5e9372a75 100644
--- a/src/java/org/apache/fop/render/java2d/CustomFontMetricsMapper.java
+++ b/src/java/org/apache/fop/render/java2d/CustomFontMetricsMapper.java
@@ -21,6 +21,7 @@ package org.apache.fop.render.java2d;
import java.awt.Font;
import java.awt.FontFormatException;
+import java.awt.Rectangle;
import java.io.IOException;
import java.io.InputStream;
import java.util.Map;
@@ -183,11 +184,31 @@ public class CustomFontMetricsMapper extends Typeface implements FontMetricsMapp
return typeface.getWidths();
}
+ public Rectangle getBoundingBox(int glyphIndex, int size) {
+ return typeface.getBoundingBox(glyphIndex, size);
+ }
+
/** {@inheritDoc} */
public final int getXHeight(final int size) {
return typeface.getXHeight(size);
}
+ public int getUnderlinePosition(int size) {
+ return typeface.getUnderlinePosition(size);
+ }
+
+ public int getUnderlineThickness(int size) {
+ return typeface.getUnderlineThickness(size);
+ }
+
+ public int getStrikeoutPosition(int size) {
+ return typeface.getStrikeoutPosition(size);
+ }
+
+ public int getStrikeoutThickness(int size) {
+ return typeface.getStrikeoutThickness(size);
+ }
+
/** {@inheritDoc} */
public final boolean hasKerningInfo() {
return typeface.hasKerningInfo();
diff --git a/src/java/org/apache/fop/render/java2d/Java2DFontMetrics.java b/src/java/org/apache/fop/render/java2d/Java2DFontMetrics.java
index fc4af3574..5e681e9e2 100644
--- a/src/java/org/apache/fop/render/java2d/Java2DFontMetrics.java
+++ b/src/java/org/apache/fop/render/java2d/Java2DFontMetrics.java
@@ -223,6 +223,26 @@ public class Java2DFontMetrics {
return xHeight * 1000;
}
+ public int getUnderlinePosition(String family, int style, int size) {
+ setFont(family, style, size);
+ return -Math.round(lineMetrics.getUnderlineOffset());
+ }
+
+ public int getUnderlineThickness(String family, int style, int size) {
+ setFont(family, style, size);
+ return Math.round(lineMetrics.getUnderlineThickness());
+ }
+
+ public int getStrikeoutPosition(String family, int style, int size) {
+ setFont(family, style, size);
+ return -Math.round(lineMetrics.getStrikethroughOffset());
+ }
+
+ public int getStrikeoutThickness(String family, int style, int size) {
+ setFont(family, style, size);
+ return Math.round(lineMetrics.getStrikethroughThickness());
+ }
+
/**
* Returns width (in 1/1000ths of point size) of character at
* code point i
diff --git a/src/java/org/apache/fop/render/java2d/Java2DSVGHandler.java b/src/java/org/apache/fop/render/java2d/Java2DSVGHandler.java
index 110581f27..f3128c600 100644
--- a/src/java/org/apache/fop/render/java2d/Java2DSVGHandler.java
+++ b/src/java/org/apache/fop/render/java2d/Java2DSVGHandler.java
@@ -31,6 +31,7 @@ import org.apache.commons.logging.LogFactory;
import org.apache.batik.bridge.BridgeContext;
import org.apache.batik.bridge.GVTBuilder;
import org.apache.batik.gvt.GraphicsNode;
+import org.apache.batik.gvt.font.DefaultFontFamilyResolver;
import org.apache.fop.image.loader.batik.BatikUtil;
import org.apache.fop.render.AbstractGenericSVGHandler;
@@ -128,8 +129,8 @@ public class Java2DSVGHandler extends AbstractGenericSVGHandler
int x = info.currentXPosition;
int y = info.currentYPosition;
-
- SVGUserAgent ua = new SVGUserAgent(context.getUserAgent(), new AffineTransform());
+ SVGUserAgent ua = new SVGUserAgent(context.getUserAgent(), DefaultFontFamilyResolver.SINGLETON,
+ new AffineTransform());
BridgeContext ctx = new BridgeContext(ua);
diff --git a/src/java/org/apache/fop/render/java2d/SystemFontMetricsMapper.java b/src/java/org/apache/fop/render/java2d/SystemFontMetricsMapper.java
index 06c975b86..f922e3f05 100644
--- a/src/java/org/apache/fop/render/java2d/SystemFontMetricsMapper.java
+++ b/src/java/org/apache/fop/render/java2d/SystemFontMetricsMapper.java
@@ -20,6 +20,7 @@
package org.apache.fop.render.java2d;
// Java
+import java.awt.Rectangle;
import java.util.Map;
import java.util.Set;
@@ -133,6 +134,22 @@ public class SystemFontMetricsMapper extends Typeface implements FontMetricsMapp
return java2DFontMetrics.getXHeight(family, style, size);
}
+ public int getUnderlinePosition(int size) {
+ return java2DFontMetrics.getUnderlinePosition(family, style, size);
+ }
+
+ public int getUnderlineThickness(int size) {
+ return java2DFontMetrics.getUnderlineThickness(family, style, size);
+ }
+
+ public int getStrikeoutPosition(int size) {
+ return java2DFontMetrics.getStrikeoutPosition(family, style, size);
+ }
+
+ public int getStrikeoutThickness(int size) {
+ return java2DFontMetrics.getStrikeoutThickness(family, style, size);
+ }
+
/**
* {@inheritDoc}
*/
@@ -148,6 +165,10 @@ public class SystemFontMetricsMapper extends Typeface implements FontMetricsMapp
return java2DFontMetrics.getWidths(family, style, Java2DFontMetrics.FONT_SIZE);
}
+ public Rectangle getBoundingBox(int glyphIndex, int size) {
+ throw new UnsupportedOperationException("Not implemented");
+ }
+
/**
* {@inheritDoc}
*/
diff --git a/src/java/org/apache/fop/render/pdf/PDFImageHandlerSVG.java b/src/java/org/apache/fop/render/pdf/PDFImageHandlerSVG.java
index 45d8dff2d..3b7084a65 100644
--- a/src/java/org/apache/fop/render/pdf/PDFImageHandlerSVG.java
+++ b/src/java/org/apache/fop/render/pdf/PDFImageHandlerSVG.java
@@ -52,6 +52,7 @@ import org.apache.fop.svg.PDFBridgeContext;
import org.apache.fop.svg.PDFGraphics2D;
import org.apache.fop.svg.SVGEventProducer;
import org.apache.fop.svg.SVGUserAgent;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
/**
* Image Handler implementation which handles SVG images.
@@ -76,7 +77,8 @@ public class PDFImageHandlerSVG implements ImageHandler {
}
final float uaResolution = userAgent.getSourceResolution();
- SVGUserAgent ua = new SVGUserAgent(userAgent, new AffineTransform());
+ SVGUserAgent ua = new SVGUserAgent(userAgent, new FOPFontFamilyResolverImpl(pdfContext.getFontInfo()),
+ new AffineTransform());
GVTBuilder builder = new GVTBuilder();
diff --git a/src/java/org/apache/fop/render/ps/AbstractPSTranscoder.java b/src/java/org/apache/fop/render/ps/AbstractPSTranscoder.java
index 1d464cae6..96886910e 100644
--- a/src/java/org/apache/fop/render/ps/AbstractPSTranscoder.java
+++ b/src/java/org/apache/fop/render/ps/AbstractPSTranscoder.java
@@ -40,6 +40,7 @@ import org.apache.fop.apps.FOPException;
import org.apache.fop.fonts.FontInfo;
import org.apache.fop.svg.AbstractFOPTranscoder;
import org.apache.fop.svg.PDFDocumentGraphics2DConfigurator;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
/**
* <p>This class enables to transcode an input to a PostScript document.</p>
@@ -115,6 +116,8 @@ public abstract class AbstractPSTranscoder extends AbstractFOPTranscoder {
this.fontInfo = PDFDocumentGraphics2DConfigurator.createFontInfo(
getEffectiveConfiguration(), useComplexScriptFeatures);
graphics.setCustomTextHandler(new NativeTextHandler(graphics, fontInfo));
+ ((FOPTranscoderUserAgent) userAgent).setFontFamilyResolver(
+ new FOPFontFamilyResolverImpl(fontInfo));
} catch (FOPException fe) {
throw new TranscoderException(fe);
}
diff --git a/src/java/org/apache/fop/render/ps/PSImageHandlerSVG.java b/src/java/org/apache/fop/render/ps/PSImageHandlerSVG.java
index 3ade34522..bd8e97063 100644
--- a/src/java/org/apache/fop/render/ps/PSImageHandlerSVG.java
+++ b/src/java/org/apache/fop/render/ps/PSImageHandlerSVG.java
@@ -41,6 +41,7 @@ import org.apache.fop.render.ImageHandler;
import org.apache.fop.render.RenderingContext;
import org.apache.fop.svg.SVGEventProducer;
import org.apache.fop.svg.SVGUserAgent;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
/**
* Image handler implementation which handles SVG images for PostScript output.
@@ -62,8 +63,8 @@ public class PSImageHandlerSVG implements ImageHandler {
boolean strokeText = false;
//TODO Configure text stroking
- SVGUserAgent ua
- = new SVGUserAgent(context.getUserAgent(), new AffineTransform());
+ SVGUserAgent ua = new SVGUserAgent(context.getUserAgent(),
+ new FOPFontFamilyResolverImpl(psContext.getFontInfo()), new AffineTransform());
PSGraphics2D graphics = new PSGraphics2D(strokeText, gen);
graphics.setGraphicContext(new org.apache.xmlgraphics.java2d.GraphicContext());
diff --git a/src/java/org/apache/fop/render/ps/PSSVGHandler.java b/src/java/org/apache/fop/render/ps/PSSVGHandler.java
index e30f6391b..dfbaf60b7 100644
--- a/src/java/org/apache/fop/render/ps/PSSVGHandler.java
+++ b/src/java/org/apache/fop/render/ps/PSSVGHandler.java
@@ -249,8 +249,7 @@ public class PSSVGHandler extends AbstractGenericSVGHandler
strokeText = cfg.getChild("stroke-text", true).getValueAsBoolean(strokeText);
}
- SVGUserAgent ua
- = new SVGUserAgent(context.getUserAgent(), new AffineTransform());
+ SVGUserAgent ua = new SVGUserAgent(context.getUserAgent(), null /* TODO */, new AffineTransform());
PSGraphics2D graphics = new PSGraphics2D(strokeText, gen);
graphics.setGraphicContext(new org.apache.xmlgraphics.java2d.GraphicContext());
diff --git a/src/java/org/apache/fop/render/ps/PSTextPainter.java b/src/java/org/apache/fop/render/ps/PSTextPainter.java
index eb2188026..44b7b08a3 100644
--- a/src/java/org/apache/fop/render/ps/PSTextPainter.java
+++ b/src/java/org/apache/fop/render/ps/PSTextPainter.java
@@ -19,25 +19,21 @@
package org.apache.fop.render.ps;
-import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.Paint;
import java.awt.Shape;
import java.awt.Stroke;
import java.awt.geom.AffineTransform;
-import java.awt.geom.Ellipse2D;
-import java.awt.geom.GeneralPath;
import java.awt.geom.PathIterator;
import java.awt.geom.Point2D;
+import java.awt.geom.Point2D.Double;
import java.io.IOException;
-import java.text.AttributedCharacterIterator;
import java.util.Iterator;
+import java.util.LinkedList;
import java.util.List;
-import org.apache.batik.gvt.font.GVTGlyphVector;
import org.apache.batik.gvt.text.TextPaintInfo;
-import org.apache.batik.gvt.text.TextSpanLayout;
import org.apache.xmlgraphics.java2d.ps.PSGraphics2D;
import org.apache.xmlgraphics.ps.PSGenerator;
@@ -48,7 +44,6 @@ import org.apache.fop.fonts.FontMetrics;
import org.apache.fop.fonts.LazyFont;
import org.apache.fop.fonts.MultiByteFont;
import org.apache.fop.svg.NativeTextPainter;
-import org.apache.fop.util.CharUtilities;
import org.apache.fop.util.HexEncoder;
/**
@@ -62,10 +57,20 @@ import org.apache.fop.util.HexEncoder;
*/
public class PSTextPainter extends NativeTextPainter {
- private static final boolean DEBUG = false;
-
private FontResourceCache fontResources;
+ private PSGraphics2D ps;
+
+ private PSGenerator gen;
+
+ private TextUtil textUtil;
+
+ private boolean flushCurrentRun;
+
+ private PSTextRun psRun;
+
+ private Double relPos;
+
private static final AffineTransform IDENTITY_TRANSFORM = new AffineTransform();
/**
@@ -82,165 +87,26 @@ public class PSTextPainter extends NativeTextPainter {
return g2d instanceof PSGraphics2D;
}
- /** {@inheritDoc} */
- protected void paintTextRun(TextRun textRun, Graphics2D g2d) throws IOException {
- AttributedCharacterIterator runaci = textRun.getACI();
- runaci.first();
-
- TextPaintInfo tpi = (TextPaintInfo)runaci.getAttribute(PAINT_INFO);
- if (tpi == null || !tpi.visible) {
- return;
- }
- if ((tpi != null) && (tpi.composite != null)) {
- g2d.setComposite(tpi.composite);
- }
-
- //------------------------------------
- TextSpanLayout layout = textRun.getLayout();
- logTextRun(runaci, layout);
- CharSequence chars = collectCharacters(runaci);
- runaci.first(); //Reset ACI
-
- final PSGraphics2D ps = (PSGraphics2D)g2d;
- final PSGenerator gen = ps.getPSGenerator();
+ @Override
+ protected void preparePainting(Graphics2D g2d) {
+ ps = (PSGraphics2D) g2d;
+ gen = ps.getPSGenerator();
ps.preparePainting();
+ }
- if (DEBUG) {
- log.debug("Text: " + chars);
- gen.commentln("%Text: " + chars);
- }
-
- GeneralPath debugShapes = null;
- if (DEBUG) {
- debugShapes = new GeneralPath();
- }
-
- TextUtil textUtil = new TextUtil(gen);
- textUtil.setupFonts(runaci);
- if (!textUtil.hasFonts()) {
- //Draw using Java2D when no native fonts are available
- textRun.getLayout().draw(g2d);
- return;
- }
-
+ @Override
+ protected void saveGraphicsState() throws IOException {
gen.saveGraphicsState();
- gen.concatMatrix(g2d.getTransform());
- Shape imclip = g2d.getClip();
- clip(ps, imclip);
-
- gen.writeln("BT"); //beginTextObject()
-
- AffineTransform localTransform = new AffineTransform();
- Point2D prevPos = null;
- GVTGlyphVector gv = layout.getGlyphVector();
- PSTextRun psRun = new PSTextRun(); //Used to split a text run into smaller runs
- for (int index = 0, c = gv.getNumGlyphs(); index < c; index++) {
- char ch = chars.charAt(index);
- boolean visibleChar = gv.isGlyphVisible(index)
- || (CharUtilities.isAnySpace(ch) && !CharUtilities.isZeroWidthSpace(ch));
- logCharacter(ch, layout, index, visibleChar);
- if (!visibleChar) {
- continue;
- }
- Point2D glyphPos = gv.getGlyphPosition(index);
-
- AffineTransform glyphTransform = gv.getGlyphTransform(index);
- if (log.isTraceEnabled()) {
- log.trace("pos " + glyphPos + ", transform " + glyphTransform);
- }
- if (DEBUG) {
- Shape sh = gv.getGlyphLogicalBounds(index);
- if (sh == null) {
- sh = new Ellipse2D.Double(glyphPos.getX(), glyphPos.getY(), 2, 2);
- }
- debugShapes.append(sh, false);
- }
-
- //Exact position of the glyph
- localTransform.setToIdentity();
- localTransform.translate(glyphPos.getX(), glyphPos.getY());
- if (glyphTransform != null) {
- localTransform.concatenate(glyphTransform);
- }
- localTransform.scale(1, -1);
-
- boolean flushCurrentRun = false;
- //Try to optimize by combining characters using the same font and on the same line.
- if (glyphTransform != null) {
- //Happens for text-on-a-path
- flushCurrentRun = true;
- }
- if (psRun.getRunLength() >= 128) {
- //Don't let a run get too long
- flushCurrentRun = true;
- }
-
- //Note the position of the glyph relative to the previous one
- Point2D relPos;
- if (prevPos == null) {
- relPos = new Point2D.Double(0, 0);
- } else {
- relPos = new Point2D.Double(
- glyphPos.getX() - prevPos.getX(),
- glyphPos.getY() - prevPos.getY());
- }
- if (psRun.vertChanges == 0
- && psRun.getHorizRunLength() > 2
- && relPos.getY() != 0) {
- //new line
- flushCurrentRun = true;
- }
-
- //Select the actual character to paint
- char paintChar = (CharUtilities.isAnySpace(ch) ? ' ' : ch);
-
- //Select (sub)font for character
- Font f = textUtil.selectFontForChar(paintChar);
- char mapped = f.mapChar(ch);
- boolean fontChanging = textUtil.isFontChanging(f, mapped);
- if (fontChanging) {
- flushCurrentRun = true;
- }
-
- if (flushCurrentRun) {
- //Paint the current run and reset for the next run
- psRun.paint(ps, textUtil, tpi);
- psRun.reset();
- }
-
- //Track current run
- psRun.addCharacter(paintChar, relPos);
- psRun.noteStartingTransformation(localTransform);
-
- //Change font if necessary
- if (fontChanging) {
- textUtil.setCurrentFont(f, mapped);
- }
+ }
- //Update last position
- prevPos = glyphPos;
- }
- psRun.paint(ps, textUtil, tpi);
- gen.writeln("ET"); //endTextObject()
+ @Override
+ protected void restoreGraphicsState() throws IOException {
gen.restoreGraphicsState();
-
- if (DEBUG) {
- //Paint debug shapes
- g2d.setStroke(new BasicStroke(0));
- g2d.setColor(Color.LIGHT_GRAY);
- g2d.draw(debugShapes);
- }
}
- private void applyColor(Paint paint, final PSGenerator gen) throws IOException {
- if (paint == null) {
- return;
- } else if (paint instanceof Color) {
- Color col = (Color)paint;
- gen.useColor(col);
- } else {
- log.warn("Paint not supported: " + paint.toString());
- }
+ @Override
+ protected void setInitialTransform(AffineTransform transform) throws IOException {
+ gen.concatMatrix(transform);
}
private PSFontResource getResourceForFont(Font f, String postfix) {
@@ -248,7 +114,8 @@ public class PSTextPainter extends NativeTextPainter {
return this.fontResources.getFontResourceForFontKey(key);
}
- private void clip(PSGraphics2D ps, Shape shape) throws IOException {
+ @Override
+ protected void clip(Shape shape) throws IOException {
if (shape == null) {
return;
}
@@ -258,17 +125,76 @@ public class PSTextPainter extends NativeTextPainter {
ps.getPSGenerator().writeln("clip");
}
+ @Override
+ protected void beginTextObject() throws IOException {
+ gen.writeln("BT");
+ textUtil = new TextUtil();
+ psRun = new PSTextRun(); //Used to split a text run into smaller runs
+ }
+
+ @Override
+ protected void endTextObject() throws IOException {
+ psRun.paint(ps, textUtil, tpi);
+ gen.writeln("ET");
+ }
+
+ @Override
+ protected void positionGlyph(Point2D prevPos, Point2D glyphPos, boolean reposition) {
+ flushCurrentRun = false;
+ //Try to optimize by combining characters using the same font and on the same line.
+ if (reposition) {
+ //Happens for text-on-a-path
+ flushCurrentRun = true;
+ }
+ if (psRun.getRunLength() >= 128) {
+ //Don't let a run get too long
+ flushCurrentRun = true;
+ }
+
+ //Note the position of the glyph relative to the previous one
+ if (prevPos == null) {
+ relPos = new Point2D.Double(0, 0);
+ } else {
+ relPos = new Point2D.Double(
+ glyphPos.getX() - prevPos.getX(),
+ glyphPos.getY() - prevPos.getY());
+ }
+ if (psRun.vertChanges == 0
+ && psRun.getHorizRunLength() > 2
+ && relPos.getY() != 0) {
+ //new line
+ flushCurrentRun = true;
+ }
+ }
+
+ @Override
+ protected void writeGlyph(char glyph, AffineTransform localTransform) throws IOException {
+ boolean fontChanging = textUtil.isFontChanging(font, glyph);
+ if (fontChanging) {
+ flushCurrentRun = true;
+ }
+
+ if (flushCurrentRun) {
+ //Paint the current run and reset for the next run
+ psRun.paint(ps, textUtil, tpi);
+ psRun.reset();
+ }
+
+ //Track current run
+ psRun.addGlyph(glyph, relPos);
+ psRun.noteStartingTransformation(localTransform);
+
+ //Change font if necessary
+ if (fontChanging) {
+ textUtil.setCurrentFont(font, glyph);
+ }
+ }
+
private class TextUtil {
- private PSGenerator gen;
- private Font[] fonts;
private Font currentFont;
private int currentEncoding = -1;
- public TextUtil(PSGenerator gen) {
- this.gen = gen;
- }
-
public boolean isMultiByte(Font f) {
FontMetrics metrics = f.getFontMetrics();
boolean multiByte = metrics instanceof MultiByteFont || metrics instanceof LazyFont
@@ -276,15 +202,6 @@ public class PSTextPainter extends NativeTextPainter {
return multiByte;
}
- public Font selectFontForChar(char ch) {
- for (int i = 0, c = fonts.length; i < c; i++) {
- if (fonts[i].hasChar(ch)) {
- return fonts[i];
- }
- }
- return fonts[0]; //TODO Maybe fall back to painting with shapes
- }
-
public void writeTextMatrix(AffineTransform transform) throws IOException {
double[] matrix = new double[6];
transform.getMatrix(matrix);
@@ -335,27 +252,19 @@ public class PSTextPainter extends NativeTextPainter {
setCurrentFont(font, encoding);
}
- public void setupFonts(AttributedCharacterIterator runaci) {
- this.fonts = findFonts(runaci);
- }
-
- public boolean hasFonts() {
- return (fonts != null) && (fonts.length > 0);
- }
-
}
private class PSTextRun {
private AffineTransform textTransform;
- private List relativePositions = new java.util.LinkedList();
- private StringBuffer currentChars = new StringBuffer();
+ private List<Point2D> relativePositions = new LinkedList<Point2D>();
+ private StringBuffer currentGlyphs = new StringBuffer();
private int horizChanges = 0;
private int vertChanges = 0;
public void reset() {
textTransform = null;
- currentChars.setLength(0);
+ currentGlyphs.setLength(0);
horizChanges = 0;
vertChanges = 0;
relativePositions.clear();
@@ -369,9 +278,9 @@ public class PSTextPainter extends NativeTextPainter {
return 0;
}
- public void addCharacter(char paintChar, Point2D relPos) {
+ public void addGlyph(char glyph, Point2D relPos) {
addRelativePosition(relPos);
- currentChars.append(paintChar);
+ currentGlyphs.append(glyph);
}
private void addRelativePosition(Point2D relPos) {
@@ -393,7 +302,7 @@ public class PSTextPainter extends NativeTextPainter {
}
public int getRunLength() {
- return currentChars.length();
+ return currentGlyphs.length();
}
private boolean isXShow() {
@@ -407,9 +316,6 @@ public class PSTextPainter extends NativeTextPainter {
public void paint(PSGraphics2D g2d, TextUtil textUtil, TextPaintInfo tpi)
throws IOException {
if (getRunLength() > 0) {
- if (log.isDebugEnabled()) {
- log.debug("Text run: " + currentChars);
- }
textUtil.writeTextMatrix(this.textTransform);
if (isXShow()) {
log.debug("Horizontal text: xshow");
@@ -431,25 +337,20 @@ public class PSTextPainter extends NativeTextPainter {
private void paintXYShow(PSGraphics2D g2d, TextUtil textUtil, Paint paint,
boolean x, boolean y) throws IOException {
- PSGenerator gen = textUtil.gen;
- char firstChar = this.currentChars.charAt(0);
- //Font only has to be setup up before the first character
- Font f = textUtil.selectFontForChar(firstChar);
- char mapped = f.mapChar(firstChar);
- textUtil.selectFont(f, mapped);
- textUtil.setCurrentFont(f, mapped);
- applyColor(paint, gen);
-
- boolean multiByte = textUtil.isMultiByte(f);
+ char glyph = currentGlyphs.charAt(0);
+ textUtil.selectFont(font, glyph);
+ textUtil.setCurrentFont(font, glyph);
+ applyColor(paint);
+
+ boolean multiByte = textUtil.isMultiByte(font);
StringBuffer sb = new StringBuffer();
sb.append(multiByte ? '<' : '(');
- for (int i = 0, c = this.currentChars.length(); i < c; i++) {
- char ch = this.currentChars.charAt(i);
- mapped = f.mapChar(ch);
+ for (int i = 0, c = this.currentGlyphs.length(); i < c; i++) {
+ glyph = this.currentGlyphs.charAt(i);
if (multiByte) {
- sb.append(HexEncoder.encode(mapped));
+ sb.append(HexEncoder.encode(glyph));
} else {
- char codepoint = (char) (mapped % 256);
+ char codepoint = (char) (glyph % 256);
PSGenerator.escapeChar(codepoint, sb);
}
}
@@ -457,9 +358,9 @@ public class PSTextPainter extends NativeTextPainter {
if (x || y) {
sb.append("\n[");
int idx = 0;
- Iterator iter = this.relativePositions.iterator();
+ Iterator<Point2D> iter = this.relativePositions.iterator();
while (iter.hasNext()) {
- Point2D pt = (Point2D)iter.next();
+ Point2D pt = iter.next();
if (idx > 0) {
if (x) {
sb.append(format(gen, pt.getX()));
@@ -500,6 +401,17 @@ public class PSTextPainter extends NativeTextPainter {
gen.writeln(sb.toString());
}
+ private void applyColor(Paint paint) throws IOException {
+ if (paint == null) {
+ return;
+ } else if (paint instanceof Color) {
+ Color col = (Color) paint;
+ gen.useColor(col);
+ } else {
+ log.warn("Paint not supported: " + paint.toString());
+ }
+ }
+
private String format(PSGenerator gen, double coord) {
if (Math.abs(coord) < 0.00001) {
return "0";
@@ -510,30 +422,21 @@ public class PSTextPainter extends NativeTextPainter {
private void paintStrokedGlyphs(PSGraphics2D g2d, TextUtil textUtil,
Paint strokePaint, Stroke stroke) throws IOException {
- PSGenerator gen = textUtil.gen;
-
- applyColor(strokePaint, gen);
+ applyColor(strokePaint);
PSGraphics2D.applyStroke(stroke, gen);
- Font f = null;
- Iterator iter = this.relativePositions.iterator();
+ Iterator<Point2D> iter = this.relativePositions.iterator();
iter.next();
Point2D pos = new Point2D.Double(0, 0);
gen.writeln("0 0 M");
- for (int i = 0, c = this.currentChars.length(); i < c; i++) {
- char ch = this.currentChars.charAt(0);
+ for (int i = 0, c = this.currentGlyphs.length(); i < c; i++) {
+ char mapped = this.currentGlyphs.charAt(i);
if (i == 0) {
- //Font only has to be setup up before the first character
- f = textUtil.selectFontForChar(ch);
- }
- char mapped = f.mapChar(ch);
- if (i == 0) {
- textUtil.selectFont(f, mapped);
- textUtil.setCurrentFont(f, mapped);
+ textUtil.selectFont(font, mapped);
+ textUtil.setCurrentFont(font, mapped);
}
//add glyph outlines to current path
- mapped = f.mapChar(this.currentChars.charAt(i));
- FontMetrics metrics = f.getFontMetrics();
+ FontMetrics metrics = font.getFontMetrics();
boolean multiByte = metrics instanceof MultiByteFont
|| metrics instanceof LazyFont
&& ((LazyFont) metrics).getRealFont() instanceof MultiByteFont;
@@ -542,14 +445,14 @@ public class PSTextPainter extends NativeTextPainter {
gen.write(HexEncoder.encode(mapped));
gen.write(">");
} else {
- char codepoint = (char)(mapped % 256);
+ char codepoint = (char) (mapped % 256);
gen.write("(" + codepoint + ")");
}
gen.writeln(" false charpath");
if (iter.hasNext()) {
//Position for the next character
- Point2D pt = (Point2D)iter.next();
+ Point2D pt = iter.next();
pos.setLocation(pos.getX() + pt.getX(), pos.getY() - pt.getY());
gen.writeln(gen.formatDouble5(pos.getX()) + " "
+ gen.formatDouble5(pos.getY()) + " M");
diff --git a/src/java/org/apache/fop/svg/ACIUtils.java b/src/java/org/apache/fop/svg/ACIUtils.java
index f54a026e1..deba98f21 100644
--- a/src/java/org/apache/fop/svg/ACIUtils.java
+++ b/src/java/org/apache/fop/svg/ACIUtils.java
@@ -38,7 +38,8 @@ import org.apache.batik.gvt.text.GVTAttributedCharacterIterator;
import org.apache.fop.fonts.Font;
import org.apache.fop.fonts.FontInfo;
-import org.apache.fop.fonts.FontTriplet;
+import org.apache.fop.svg.font.FOPGVTFont;
+import org.apache.fop.svg.font.FOPGVTFontFamily;
/**
* Utilities for java.text.AttributedCharacterIterator.
@@ -64,57 +65,43 @@ public final class ACIUtils {
@SuppressWarnings("unchecked")
List<GVTFontFamily> gvtFonts = (List<GVTFontFamily>) aci.getAttribute(
GVTAttributedCharacterIterator.TextAttribute.GVT_FONT_FAMILIES);
- Float posture = (Float) aci.getAttribute(TextAttribute.POSTURE);
- Float taWeight = (Float) aci.getAttribute(TextAttribute.WEIGHT);
- Float fontSize = (Float) aci.getAttribute(TextAttribute.SIZE);
-
- String style = toStyle(posture);
- int weight = toCSSWeight(taWeight);
- int fsize = (int)(fontSize.floatValue() * 1000);
+ String style = toStyle((Float) aci.getAttribute(TextAttribute.POSTURE));
+ int weight = toCSSWeight((Float) aci.getAttribute(TextAttribute.WEIGHT));
+ float fontSize = ((Float) aci.getAttribute(TextAttribute.SIZE)).floatValue();
String firstFontFamily = null;
-
//GVT_FONT can sometimes be different from the fonts in GVT_FONT_FAMILIES
//or GVT_FONT_FAMILIES can even be empty and only GVT_FONT is set
- /* The following code section is not available until Batik 1.7 is released. */
- GVTFont gvtFont = (GVTFont)aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.GVT_FONT);
+ GVTFont gvtFont = (GVTFont) aci.getAttribute(GVTAttributedCharacterIterator.TextAttribute.GVT_FONT);
if (gvtFont != null) {
String gvtFontFamily = gvtFont.getFamilyName();
- if (fontInfo.hasFont(gvtFontFamily, style, weight)) {
- FontTriplet triplet = fontInfo.fontLookup(gvtFontFamily, style,
- weight);
- Font f = fontInfo.getFontInstance(triplet, fsize);
+ if (gvtFont instanceof FOPGVTFont) {
+ Font font = ((FOPGVTFont) gvtFont).getFont();
if (LOG.isDebugEnabled()) {
LOG.debug("Found a font that matches the GVT font: "
+ gvtFontFamily + ", " + weight + ", " + style
- + " -> " + f);
+ + " -> " + font);
}
- fonts.add(f);
+ fonts.add(font);
}
firstFontFamily = gvtFontFamily;
}
if (gvtFonts != null) {
boolean haveInstanceOfSVGFontFamily = false;
- for (GVTFontFamily fam : gvtFonts) {
- if (fam instanceof SVGFontFamily) {
+ for (GVTFontFamily fontFamily : gvtFonts) {
+ if (fontFamily instanceof SVGFontFamily) {
haveInstanceOfSVGFontFamily = true;
- }
- String fontFamily = fam.getFamilyName();
- if (fontInfo.hasFont(fontFamily, style, weight)) {
- FontTriplet triplet = fontInfo.fontLookup(fontFamily, style,
- weight);
- Font f = fontInfo.getFontInstance(triplet, fsize);
+ } else if (fontFamily instanceof FOPGVTFontFamily) {
+ Font font = ((FOPGVTFontFamily) fontFamily).deriveFont(fontSize, aci).getFont();
if (LOG.isDebugEnabled()) {
LOG.debug("Found a font that matches the GVT font family: "
- + fontFamily + ", " + weight + ", " + style
- + " -> " + f);
+ + fontFamily.getFamilyName() + ", " + weight + ", " + style + " -> " + font);
}
- fonts.add(f);
+ fonts.add(font);
}
if (firstFontFamily == null) {
- firstFontFamily = fontFamily;
+ firstFontFamily = fontFamily.getFamilyName();
}
}
// SVG fonts are embedded fonts in the SVG document and are rarely used; however if they
@@ -126,25 +113,10 @@ public final class ACIUtils {
return null; // Let Batik paint this text!
}
}
- if (fonts.isEmpty()) {
- if (firstFontFamily == null) {
- //This will probably never happen. Just to be on the safe side.
- firstFontFamily = "any";
- }
- //lookup with fallback possibility (incl. substitution notification)
- FontTriplet triplet = fontInfo.fontLookup(firstFontFamily, style, weight);
- Font f = fontInfo.getFontInstance(triplet, fsize);
- if (LOG.isDebugEnabled()) {
- LOG.debug("Falling back to adjustable font lookup up for: "
- + firstFontFamily + ", " + weight + ", " + style
- + " -> " + f);
- }
- fonts.add(f);
- }
- return fonts.toArray(new Font[fonts.size()]);
+ return fonts.isEmpty() ? null : fonts.toArray(new Font[fonts.size()]);
}
- private static int toCSSWeight(Float weight) {
+ public static int toCSSWeight(Float weight) {
if (weight == null) {
return 400;
} else if (weight <= TextAttribute.WEIGHT_EXTRA_LIGHT.floatValue()) {
@@ -170,7 +142,7 @@ public final class ACIUtils {
}
}
- private static String toStyle(Float posture) {
+ public static String toStyle(Float posture) {
return ((posture != null) && (posture.floatValue() > 0.0))
? Font.STYLE_ITALIC
: Font.STYLE_NORMAL;
diff --git a/src/java/org/apache/fop/svg/AbstractFOPTextPainter.java b/src/java/org/apache/fop/svg/AbstractFOPTextPainter.java
index 3e16799f4..7df127d3a 100644
--- a/src/java/org/apache/fop/svg/AbstractFOPTextPainter.java
+++ b/src/java/org/apache/fop/svg/AbstractFOPTextPainter.java
@@ -23,20 +23,15 @@ import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.Paint;
import java.awt.Shape;
-import java.awt.font.TextAttribute;
-import java.awt.geom.AffineTransform;
import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;
import java.io.IOException;
import java.text.AttributedCharacterIterator;
import java.text.CharacterIterator;
-import java.util.Iterator;
-import java.util.List;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
-import org.apache.batik.dom.svg.SVGOMTextElement;
import org.apache.batik.gvt.TextNode;
import org.apache.batik.gvt.TextPainter;
import org.apache.batik.gvt.renderer.StrokingTextPainter;
@@ -66,15 +61,15 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
* Use the stroking text painter to get the bounds and shape.
* Also used as a fallback to draw the string with strokes.
*/
- protected static final TextPainter
- PROXY_PAINTER = StrokingTextPainter.getInstance();
+ private final TextPainter proxyTextPainter;
/**
* Create a new PS text painter with the given font information.
* @param nativeTextHandler the NativeTextHandler instance used for text painting
*/
- public AbstractFOPTextPainter(FOPTextHandler nativeTextHandler) {
+ public AbstractFOPTextPainter(FOPTextHandler nativeTextHandler, TextPainter proxyTextPainter) {
this.nativeTextHandler = nativeTextHandler;
+ this.proxyTextPainter = proxyTextPainter;
}
/**
@@ -85,19 +80,10 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
* @param g2d the Graphics2D to use
*/
public void paint(TextNode node, Graphics2D g2d) {
- Point2D loc = node.getLocation();
- if (!isSupportedGraphics2D(g2d) || hasUnsupportedAttributes(node)) {
- if (log.isDebugEnabled()) {
- log.debug("painting text node " + node
- + " by stroking due to unsupported attributes or an incompatible Graphics2D");
- }
- PROXY_PAINTER.paint(node, g2d);
- } else {
- if (log.isDebugEnabled()) {
- log.debug("painting text node " + node + " normally.");
- }
- paintTextRuns(node.getTextRuns(), g2d, loc);
+ if (isSupportedGraphics2D(g2d)) {
+ new TextRunPainter().paintTextRuns(node.getTextRuns(), g2d, node.getLocation());
}
+ proxyTextPainter.paint(node, g2d);
}
/**
@@ -109,190 +95,99 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
*/
protected abstract boolean isSupportedGraphics2D(Graphics2D g2d);
- private boolean hasUnsupportedAttributes(TextNode node) {
- Iterator iter = node.getTextRuns().iterator();
- while (iter.hasNext()) {
- StrokingTextPainter.TextRun
- run = (StrokingTextPainter.TextRun)iter.next();
- AttributedCharacterIterator aci = run.getACI();
- boolean hasUnsupported = hasUnsupportedAttributes(aci);
- if (hasUnsupported) {
- return true;
- }
- }
- return false;
- }
-
- private boolean hasUnsupportedAttributes(AttributedCharacterIterator aci) {
- boolean hasUnsupported = false;
-
- Font font = getFont(aci);
- String text = getText(aci);
- if (hasUnsupportedGlyphs(text, font)) {
- log.trace("-> Unsupported glyphs found");
- hasUnsupported = true;
- }
+ private class TextRunPainter {
- TextPaintInfo tpi = (TextPaintInfo) aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.PAINT_INFO);
- if ((tpi != null)
- && ((tpi.strokeStroke != null && tpi.strokePaint != null)
- || (tpi.strikethroughStroke != null)
- || (tpi.underlineStroke != null)
- || (tpi.overlineStroke != null))) {
- log.trace("-> under/overlines etc. found");
- hasUnsupported = true;
- }
+ private Point2D currentLocation;
- //Alpha is not supported
- Paint foreground = (Paint) aci.getAttribute(TextAttribute.FOREGROUND);
- if (foreground instanceof Color) {
- Color col = (Color)foreground;
- if (col.getAlpha() != 255) {
- log.trace("-> transparency found");
- hasUnsupported = true;
+ public void paintTextRuns(Iterable<StrokingTextPainter.TextRun> textRuns, Graphics2D g2d,
+ Point2D nodeLocation) {
+ currentLocation = new Point2D.Double(nodeLocation.getX(), nodeLocation.getY());
+ for (StrokingTextPainter.TextRun run : textRuns) {
+ paintTextRun(run, g2d);
}
}
- Object letSpace = aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.LETTER_SPACING);
- if (letSpace != null) {
- log.trace("-> letter spacing found");
- hasUnsupported = true;
- }
-
- Object wordSpace = aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.WORD_SPACING);
- if (wordSpace != null) {
- log.trace("-> word spacing found");
- hasUnsupported = true;
- }
-
- Object lengthAdjust = aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.LENGTH_ADJUST);
- if (lengthAdjust != null) {
- log.trace("-> length adjustments found");
- hasUnsupported = true;
- }
-
- Object writeMod = aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.WRITING_MODE);
- if (writeMod != null
- && !GVTAttributedCharacterIterator.TextAttribute.WRITING_MODE_LTR.equals(
- writeMod)) {
- log.trace("-> Unsupported writing modes found");
- hasUnsupported = true;
- }
-
- Object vertOr = aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.VERTICAL_ORIENTATION);
- if (GVTAttributedCharacterIterator.TextAttribute.ORIENTATION_ANGLE.equals(
- vertOr)) {
- log.trace("-> vertical orientation found");
- hasUnsupported = true;
- }
-
- Object rcDel = aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.TEXT_COMPOUND_DELIMITER);
- //Batik 1.6 returns null here which makes it impossible to determine whether this can
- //be painted or not, i.e. fall back to stroking. :-(
- if (rcDel != null && !(rcDel instanceof SVGOMTextElement)) {
- log.trace("-> spans found");
- hasUnsupported = true; //Filter spans
- }
-
- if (hasUnsupported) {
- log.trace("Unsupported attributes found in ACI, using StrokingTextPainter");
- }
- return hasUnsupported;
- }
-
- /**
- * Paint a list of text runs on the Graphics2D at a given location.
- * @param textRuns the list of text runs
- * @param g2d the Graphics2D to paint to
- * @param loc the current location of the "cursor"
- */
- protected void paintTextRuns(List textRuns, Graphics2D g2d, Point2D loc) {
- Point2D currentloc = loc;
- Iterator i = textRuns.iterator();
- while (i.hasNext()) {
- StrokingTextPainter.TextRun
- run = (StrokingTextPainter.TextRun)i.next();
- currentloc = paintTextRun(run, g2d, currentloc);
- }
- }
-
- /**
- * Paint a single text run on the Graphics2D at a given location.
- * @param run the text run to paint
- * @param g2d the Graphics2D to paint to
- * @param loc the current location of the "cursor"
- * @return the new location of the "cursor" after painting the text run
- */
- protected Point2D paintTextRun(StrokingTextPainter.TextRun run, Graphics2D g2d, Point2D loc) {
- AttributedCharacterIterator aci = run.getACI();
- aci.first();
-
- updateLocationFromACI(aci, loc);
- AffineTransform at = g2d.getTransform();
- loc = at.transform(loc, null);
-
- // font
- Font font = getFont(aci);
- if (font != null) {
- nativeTextHandler.setOverrideFont(font);
- }
-
- // color
- TextPaintInfo tpi = (TextPaintInfo) aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.PAINT_INFO);
- if (tpi == null) {
- return loc;
- }
- Paint foreground = tpi.fillPaint;
- if (foreground instanceof Color) {
- Color col = (Color)foreground;
- g2d.setColor(col);
- }
- g2d.setPaint(foreground);
-
- // text anchor
- TextNode.Anchor anchor = (TextNode.Anchor)aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.ANCHOR_TYPE);
-
- // text
- String txt = getText(aci);
- float advance = getStringWidth(txt, font);
- float tx = 0;
- if (anchor != null) {
- switch (anchor.getType()) {
- case TextNode.Anchor.ANCHOR_MIDDLE:
- tx = -advance / 2;
- break;
- case TextNode.Anchor.ANCHOR_END:
- tx = -advance;
- break;
- default: //nop
+ private void paintTextRun(StrokingTextPainter.TextRun run, Graphics2D g2d) {
+ AttributedCharacterIterator aci = run.getACI();
+ aci.first();
+ updateLocationFromACI(aci, currentLocation);
+ // font
+ Font font = getFont(aci);
+ if (font != null) {
+ nativeTextHandler.setOverrideFont(font);
}
- }
-
- // draw string
- double x = loc.getX();
- double y = loc.getY();
- try {
+ // color
+ TextPaintInfo tpi = (TextPaintInfo) aci.getAttribute(
+ GVTAttributedCharacterIterator.TextAttribute.PAINT_INFO);
+ if (tpi == null) {
+ return;
+ }
+ Paint foreground = tpi.fillPaint;
+ if (foreground instanceof Color) {
+ Color col = (Color) foreground;
+ g2d.setColor(col);
+ }
+ g2d.setPaint(foreground);
+ // text anchor
+ TextNode.Anchor anchor = (TextNode.Anchor) aci.getAttribute(
+ GVTAttributedCharacterIterator.TextAttribute.ANCHOR_TYPE);
+ // text
+ String txt = getText(aci);
+ double advance = font == null ? run.getLayout().getAdvance2D().getX() : getStringWidth(txt, font);
+ double tx = 0;
+ if (anchor != null) {
+ switch (anchor.getType()) {
+ case TextNode.Anchor.ANCHOR_MIDDLE:
+ tx = -advance / 2;
+ break;
+ case TextNode.Anchor.ANCHOR_END:
+ tx = -advance;
+ break;
+ default: //nop
+ }
+ }
+ // draw string
+ Point2D outputLocation = g2d.getTransform().transform(currentLocation, null);
+ double x = outputLocation.getX();
+ double y = outputLocation.getY();
try {
- nativeTextHandler.drawString(g2d, txt, (float)x + tx, (float)y);
- } catch (IOException ioe) {
- if (g2d instanceof AFPGraphics2D) {
- ((AFPGraphics2D)g2d).handleIOException(ioe);
+ try {
+ //TODO draw underline and overline if set
+ nativeTextHandler.drawString(g2d, txt, (float) (x + tx), (float) y);
+ //TODO draw strikethrough if set
+ } catch (IOException ioe) {
+ if (g2d instanceof AFPGraphics2D) {
+ ((AFPGraphics2D) g2d).handleIOException(ioe);
+ }
}
+ } finally {
+ nativeTextHandler.setOverrideFont(null);
+ }
+ currentLocation.setLocation(currentLocation.getX() + advance, currentLocation.getY());
+ }
+ private void updateLocationFromACI(AttributedCharacterIterator aci, Point2D loc) {
+ //Adjust position of span
+ Float xpos = (Float) aci.getAttribute(
+ GVTAttributedCharacterIterator.TextAttribute.X);
+ Float ypos = (Float) aci.getAttribute(
+ GVTAttributedCharacterIterator.TextAttribute.Y);
+ Float dxpos = (Float) aci.getAttribute(
+ GVTAttributedCharacterIterator.TextAttribute.DX);
+ Float dypos = (Float) aci.getAttribute(
+ GVTAttributedCharacterIterator.TextAttribute.DY);
+ if (xpos != null) {
+ loc.setLocation(xpos.doubleValue(), loc.getY());
+ }
+ if (ypos != null) {
+ loc.setLocation(loc.getX(), ypos.doubleValue());
+ }
+ if (dxpos != null) {
+ loc.setLocation(loc.getX() + dxpos.doubleValue(), loc.getY());
+ }
+ if (dypos != null) {
+ loc.setLocation(loc.getX(), loc.getY() + dypos.doubleValue());
}
- } finally {
- nativeTextHandler.setOverrideFont(null);
}
- loc.setLocation(loc.getX() + advance, loc.getY());
- return loc;
}
/**
@@ -305,36 +200,9 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
for (char c = aci.first(); c != CharacterIterator.DONE; c = aci.next()) {
sb.append(c);
}
- aci.first();
return sb.toString();
}
- private void updateLocationFromACI(
- AttributedCharacterIterator aci,
- Point2D loc) {
- //Adjust position of span
- Float xpos = (Float)aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.X);
- Float ypos = (Float)aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.Y);
- Float dxpos = (Float)aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.DX);
- Float dypos = (Float)aci.getAttribute(
- GVTAttributedCharacterIterator.TextAttribute.DY);
- if (xpos != null) {
- loc.setLocation(xpos.doubleValue(), loc.getY());
- }
- if (ypos != null) {
- loc.setLocation(loc.getX(), ypos.doubleValue());
- }
- if (dxpos != null) {
- loc.setLocation(loc.getX() + dxpos.doubleValue(), loc.getY());
- }
- if (dypos != null) {
- loc.setLocation(loc.getX(), loc.getY() + dypos.doubleValue());
- }
- }
-
private Font getFont(AttributedCharacterIterator aci) {
Font[] fonts = ACIUtils.findFontsForBatikACI(aci, nativeTextHandler.getFontInfo());
return fonts == null ? null : fonts[0];
@@ -360,21 +228,6 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
return wordWidth / 1000f;
}
- private boolean hasUnsupportedGlyphs(String str, Font font) {
- if (font == null) {
- return true;
- }
- for (int i = 0; i < str.length(); i++) {
- char c = str.charAt(i);
- if (!((c == ' ') || (c == '\n') || (c == '\r') || (c == '\t'))) {
- if (!font.hasChar(c)) {
- return true;
- }
- }
- }
- return false;
- }
-
/**
* Get the outline shape of the text characters.
* This uses the StrokingTextPainter to get the outline
@@ -384,7 +237,7 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
* @return the outline shape of the text characters
*/
public Shape getOutline(TextNode node) {
- return PROXY_PAINTER.getOutline(node);
+ return proxyTextPainter.getOutline(node);
}
/**
@@ -399,7 +252,7 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
/* (todo) getBounds2D() is too slow
* because it uses the StrokingTextPainter. We should implement this
* method ourselves. */
- return PROXY_PAINTER.getBounds2D(node);
+ return proxyTextPainter.getBounds2D(node);
}
/**
@@ -411,7 +264,7 @@ public abstract class AbstractFOPTextPainter implements TextPainter {
* @return the bounds of the text
*/
public Rectangle2D getGeometryBounds(TextNode node) {
- return PROXY_PAINTER.getGeometryBounds(node);
+ return proxyTextPainter.getGeometryBounds(node);
}
// Methods that have no purpose for PS
diff --git a/src/java/org/apache/fop/svg/AbstractFOPTranscoder.java b/src/java/org/apache/fop/svg/AbstractFOPTranscoder.java
index e04bf0d35..9b1606953 100644
--- a/src/java/org/apache/fop/svg/AbstractFOPTranscoder.java
+++ b/src/java/org/apache/fop/svg/AbstractFOPTranscoder.java
@@ -36,9 +36,9 @@ import org.apache.avalon.framework.configuration.DefaultConfiguration;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.impl.SimpleLog;
-import org.apache.batik.bridge.UserAgent;
import org.apache.batik.dom.svg.SVGDOMImplementation;
import org.apache.batik.dom.util.DocumentFactory;
+import org.apache.batik.gvt.font.FontFamilyResolver;
import org.apache.batik.transcoder.ErrorHandler;
import org.apache.batik.transcoder.SVGAbstractTranscoder;
import org.apache.batik.transcoder.TranscoderException;
@@ -56,6 +56,8 @@ import org.apache.xmlgraphics.image.loader.ImageSessionContext;
import org.apache.xmlgraphics.image.loader.impl.AbstractImageSessionContext;
import org.apache.xmlgraphics.util.UnitConv;
+import org.apache.fop.svg.font.FOPFontFamilyResolver;
+
/**
* This is the common base class of all of FOP's transcoders.
*/
@@ -86,11 +88,6 @@ public abstract class AbstractFOPTranscoder extends SVGAbstractTranscoder implem
/** The value to turn off text stroking. */
public static final Boolean VALUE_FORMAT_OFF = Boolean.FALSE;
- /**
- * The user agent dedicated to this Transcoder.
- */
- protected UserAgent userAgent = createUserAgent();
-
private Log logger;
private EntityResolver resolver;
private Configuration cfg = null;
@@ -113,7 +110,7 @@ public abstract class AbstractFOPTranscoder extends SVGAbstractTranscoder implem
* this method if you need non-default behaviour.
* @return UserAgent the newly created user agent
*/
- protected UserAgent createUserAgent() {
+ protected FOPTranscoderUserAgent createUserAgent() {
return new FOPTranscoderUserAgent();
}
@@ -331,6 +328,8 @@ public abstract class AbstractFOPTranscoder extends SVGAbstractTranscoder implem
*/
protected class FOPTranscoderUserAgent extends SVGAbstractTranscoderUserAgent {
+ private FOPFontFamilyResolver fontFamilyResolver;
+
/**
* Displays the specified error message using the {@link ErrorHandler}.
* @param message the message to display
@@ -386,6 +385,15 @@ public abstract class AbstractFOPTranscoder extends SVGAbstractTranscoder implem
return "print";
}
+ public void setFontFamilyResolver(FOPFontFamilyResolver resolver) {
+ fontFamilyResolver = resolver;
+ }
+
+ @Override
+ public FontFamilyResolver getFontFamilyResolver() {
+ return fontFamilyResolver;
+ }
+
}
}
diff --git a/src/java/org/apache/fop/svg/NativeTextPainter.java b/src/java/org/apache/fop/svg/NativeTextPainter.java
index 4513e0101..200f6558b 100644
--- a/src/java/org/apache/fop/svg/NativeTextPainter.java
+++ b/src/java/org/apache/fop/svg/NativeTextPainter.java
@@ -19,7 +19,14 @@
package org.apache.fop.svg;
+import java.awt.BasicStroke;
+import java.awt.Color;
import java.awt.Graphics2D;
+import java.awt.Shape;
+import java.awt.geom.AffineTransform;
+import java.awt.geom.Ellipse2D;
+import java.awt.geom.GeneralPath;
+import java.awt.geom.Point2D;
import java.io.IOException;
import java.text.AttributedCharacterIterator;
import java.util.List;
@@ -27,11 +34,17 @@ import java.util.List;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
+import org.apache.batik.bridge.SVGGVTFont;
+import org.apache.batik.gvt.font.FontFamilyResolver;
+import org.apache.batik.gvt.font.GVTGlyphVector;
import org.apache.batik.gvt.renderer.StrokingTextPainter;
+import org.apache.batik.gvt.text.TextPaintInfo;
import org.apache.batik.gvt.text.TextSpanLayout;
import org.apache.fop.fonts.Font;
import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
+import org.apache.fop.svg.font.FOPGVTFont;
import org.apache.fop.util.CharUtilities;
/**
@@ -41,17 +54,26 @@ import org.apache.fop.util.CharUtilities;
public abstract class NativeTextPainter extends StrokingTextPainter {
/** the logger for this class */
- protected Log log = LogFactory.getLog(NativeTextPainter.class);
+ protected static final Log log = LogFactory.getLog(NativeTextPainter.class);
+
+ private static final boolean DEBUG = false;
/** the font collection */
protected final FontInfo fontInfo;
+ protected final FontFamilyResolver fontFamilyResolver;
+
+ protected Font font;
+
+ protected TextPaintInfo tpi;
+
/**
* Creates a new instance.
* @param fontInfo the font collection
*/
public NativeTextPainter(FontInfo fontInfo) {
this.fontInfo = fontInfo;
+ this.fontFamilyResolver = new FOPFontFamilyResolverImpl(fontInfo);
}
/**
@@ -68,11 +90,93 @@ public abstract class NativeTextPainter extends StrokingTextPainter {
* @param g2d the target Graphics2D instance
* @throws IOException if an I/O error occurs while rendering the text
*/
- protected abstract void paintTextRun(TextRun textRun, Graphics2D g2d) throws IOException;
+ protected final void paintTextRun(TextRun textRun, Graphics2D g2d) throws IOException {
+ AttributedCharacterIterator runaci = textRun.getACI();
+ runaci.first();
+
+ tpi = (TextPaintInfo) runaci.getAttribute(PAINT_INFO);
+ if (tpi == null || !tpi.visible) {
+ return;
+ }
+ if (tpi.composite != null) {
+ g2d.setComposite(tpi.composite);
+ }
+
+ //------------------------------------
+ TextSpanLayout layout = textRun.getLayout();
+ logTextRun(runaci, layout);
+ runaci.first(); //Reset ACI
+
+ GeneralPath debugShapes = null;
+ if (DEBUG) {
+ debugShapes = new GeneralPath();
+ }
+
+ preparePainting(g2d);
+
+ GVTGlyphVector gv = layout.getGlyphVector();
+ if (!(gv.getFont() instanceof FOPGVTFont)) {
+ assert gv.getFont() == null || gv.getFont() instanceof SVGGVTFont;
+ //Draw using Java2D when no native fonts are available
+ textRun.getLayout().draw(g2d);
+ return;
+ }
+ font = ((FOPGVTFont) gv.getFont()).getFont();
+
+ saveGraphicsState();
+ setInitialTransform(g2d.getTransform());
+ clip(g2d.getClip());
+ beginTextObject();
+
+ AffineTransform localTransform = new AffineTransform();
+ Point2D prevPos = null;
+ AffineTransform prevGlyphTransform = null;
+ for (int index = 0, c = gv.getNumGlyphs(); index < c; index++) {
+ if (!gv.isGlyphVisible(index)) {
+ continue;
+ }
+ Point2D glyphPos = gv.getGlyphPosition(index);
+
+ AffineTransform glyphTransform = gv.getGlyphTransform(index);
+ if (log.isTraceEnabled()) {
+ log.trace("pos " + glyphPos + ", transform " + glyphTransform);
+ }
+ if (DEBUG) {
+ Shape sh = gv.getGlyphLogicalBounds(index);
+ if (sh == null) {
+ sh = new Ellipse2D.Double(glyphPos.getX(), glyphPos.getY(), 2, 2);
+ }
+ debugShapes.append(sh, false);
+ }
+
+ //Exact position of the glyph
+ localTransform.setToIdentity();
+ localTransform.translate(glyphPos.getX(), glyphPos.getY());
+ if (glyphTransform != null) {
+ localTransform.concatenate(glyphTransform);
+ }
+ localTransform.scale(1, -1);
+
+ positionGlyph(prevPos, glyphPos, glyphTransform != null || prevGlyphTransform != null);
+ char glyph = (char) gv.getGlyphCode(index);
+ //Update last position
+ prevPos = glyphPos;
+ prevGlyphTransform = glyphTransform;
+
+ writeGlyph(glyph, localTransform);
+ }
+ endTextObject();
+ restoreGraphicsState();
+ if (DEBUG) {
+ //Paint debug shapes
+ g2d.setStroke(new BasicStroke(0));
+ g2d.setColor(Color.LIGHT_GRAY);
+ g2d.draw(debugShapes);
+ }
+ }
- /** {@inheritDoc} */
@Override
- protected void paintTextRuns(List textRuns, Graphics2D g2d) {
+ protected void paintTextRuns(@SuppressWarnings("rawtypes") List textRuns, Graphics2D g2d) {
if (log.isTraceEnabled()) {
log.trace("paintTextRuns: count = " + textRuns.size());
}
@@ -81,7 +185,7 @@ public abstract class NativeTextPainter extends StrokingTextPainter {
return;
}
for (int i = 0; i < textRuns.size(); i++) {
- TextRun textRun = (TextRun)textRuns.get(i);
+ TextRun textRun = (TextRun) textRuns.get(i);
try {
paintTextRun(textRun, g2d);
} catch (IOException ioe) {
@@ -92,16 +196,6 @@ public abstract class NativeTextPainter extends StrokingTextPainter {
}
/**
- * Finds an array of suitable fonts for a given AttributedCharacterIterator.
- * @param aci the character iterator
- * @return the array of fonts
- */
- protected Font[] findFonts(AttributedCharacterIterator aci) {
- Font[] fonts = ACIUtils.findFontsForBatikACI(aci, fontInfo);
- return fonts;
- }
-
- /**
* Collects all characters from an {@link AttributedCharacterIterator}.
* @param runaci the character iterator
* @return the characters
@@ -115,6 +209,25 @@ public abstract class NativeTextPainter extends StrokingTextPainter {
return chars;
}
+ protected abstract void preparePainting(Graphics2D g2d);
+
+ protected abstract void saveGraphicsState() throws IOException;
+
+ protected abstract void restoreGraphicsState() throws IOException;
+
+ protected abstract void setInitialTransform(AffineTransform transform) throws IOException;
+
+ protected abstract void clip(Shape clip) throws IOException;
+
+ protected abstract void beginTextObject() throws IOException;
+
+ protected abstract void endTextObject() throws IOException;
+
+ protected abstract void positionGlyph(Point2D prevPos, Point2D glyphPos, boolean reposition);
+
+ protected abstract void writeGlyph(char glyph, AffineTransform transform) throws IOException;
+
+
/**
* @param runaci an attributed character iterator
* @param layout a text span layout
@@ -155,5 +268,9 @@ public abstract class NativeTextPainter extends StrokingTextPainter {
}
}
+ @Override
+ protected FontFamilyResolver getFontFamilyResolver() {
+ return this.fontFamilyResolver;
+ }
}
diff --git a/src/java/org/apache/fop/svg/PDFTextPainter.java b/src/java/org/apache/fop/svg/PDFTextPainter.java
index ef376663f..c5fa9f04e 100644
--- a/src/java/org/apache/fop/svg/PDFTextPainter.java
+++ b/src/java/org/apache/fop/svg/PDFTextPainter.java
@@ -19,25 +19,17 @@
package org.apache.fop.svg;
-import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.Paint;
import java.awt.Shape;
import java.awt.Stroke;
import java.awt.geom.AffineTransform;
-import java.awt.geom.Ellipse2D;
-import java.awt.geom.GeneralPath;
import java.awt.geom.Point2D;
-import java.text.AttributedCharacterIterator;
-import org.apache.batik.gvt.font.GVTGlyphVector;
import org.apache.batik.gvt.text.TextPaintInfo;
-import org.apache.batik.gvt.text.TextSpanLayout;
-import org.apache.fop.fonts.Font;
import org.apache.fop.fonts.FontInfo;
-import org.apache.fop.util.CharUtilities;
/**
* Renders the attributed character iterator of a {@link org.apache.batik.gvt.TextNode}.
@@ -51,11 +43,19 @@ import org.apache.fop.util.CharUtilities;
*/
class PDFTextPainter extends NativeTextPainter {
- private static final boolean DEBUG = false;
+ private PDFGraphics2D pdf;
+
+ private PDFTextUtil textUtil;
+
+ private double prevVisibleGlyphWidth;
+
+ private boolean repositionNextGlyph;
/**
* Create a new PDF text painter with the given font information.
+ *
* @param fi the font info
+ * @param fontFamilyResolver the Font Family Resolver
*/
public PDFTextPainter(FontInfo fi) {
super(fi);
@@ -67,29 +67,29 @@ class PDFTextPainter extends NativeTextPainter {
return g2d instanceof PDFGraphics2D;
}
- /** {@inheritDoc} */
@Override
- protected void paintTextRun(TextRun textRun, Graphics2D g2d) {
- AttributedCharacterIterator runaci = textRun.getACI();
- runaci.first();
+ protected void preparePainting(Graphics2D g2d) {
+ pdf = (PDFGraphics2D) g2d;
+ }
- TextPaintInfo tpi = (TextPaintInfo)runaci.getAttribute(PAINT_INFO);
- if (tpi == null || !tpi.visible) {
- return;
- }
- if ((tpi != null) && (tpi.composite != null)) {
- g2d.setComposite(tpi.composite);
- }
+ @Override
+ protected void saveGraphicsState() {
+ pdf.saveGraphicsState();
+ }
- //------------------------------------
- TextSpanLayout layout = textRun.getLayout();
- logTextRun(runaci, layout);
- CharSequence chars = collectCharacters(runaci);
- runaci.first(); //Reset ACI
+ @Override
+ protected void restoreGraphicsState() {
+ pdf.restoreGraphicsState();
+ }
- final PDFGraphics2D pdf = (PDFGraphics2D)g2d;
+ @Override
+ protected void setInitialTransform(AffineTransform transform) {
+ createTextUtil();
+ textUtil.concatMatrix(transform);
+ }
- PDFTextUtil textUtil = new PDFTextUtil(pdf.fontInfo) {
+ private void createTextUtil() {
+ textUtil = new PDFTextUtil(pdf.fontInfo) {
protected void write(String code) {
pdf.currentStream.write(code);
}
@@ -97,142 +97,39 @@ class PDFTextPainter extends NativeTextPainter {
pdf.currentStream.append(code);
}
};
+ }
- if (DEBUG) {
- log.debug("Text: " + chars);
- pdf.currentStream.write("%Text: " + chars + "\n");
- }
-
- GeneralPath debugShapes = null;
- if (DEBUG) {
- debugShapes = new GeneralPath();
- }
-
- Font[] fonts = findFonts(runaci);
- if (fonts == null || fonts.length == 0) {
- //Draw using Java2D when no native fonts are available
- textRun.getLayout().draw(g2d);
- return;
- }
-
- pdf.saveGraphicsState();
- textUtil.concatMatrix(g2d.getTransform());
- Shape imclip = g2d.getClip();
- pdf.writeClip(imclip);
-
- applyColorAndPaint(tpi, pdf);
+ @Override
+ protected void clip(Shape clip) {
+ pdf.writeClip(clip);
+ }
+ @Override
+ protected void beginTextObject() {
+ applyColorAndPaint(tpi);
textUtil.beginTextObject();
- textUtil.setFonts(fonts);
- boolean stroke = (tpi.strokePaint != null)
- && (tpi.strokeStroke != null);
+ boolean stroke = (tpi.strokePaint != null) && (tpi.strokeStroke != null);
textUtil.setTextRenderingMode(tpi.fillPaint != null, stroke, false);
+ }
- AffineTransform localTransform = new AffineTransform();
- Point2D prevPos = null;
- double prevVisibleCharWidth = 0.0;
- GVTGlyphVector gv = layout.getGlyphVector();
- for (int index = 0, c = gv.getNumGlyphs(); index < c; index++) {
- char ch = chars.charAt(index);
- boolean visibleChar = gv.isGlyphVisible(index)
- || (CharUtilities.isAnySpace(ch) && !CharUtilities.isZeroWidthSpace(ch));
- logCharacter(ch, layout, index, visibleChar);
- if (!visibleChar) {
- continue;
- }
- Point2D glyphPos = gv.getGlyphPosition(index);
-
- AffineTransform glyphTransform = gv.getGlyphTransform(index);
- //TODO Glyph transforms could be refined so not every char has to be painted
- //with its own TJ command (stretch/squeeze case could be optimized)
- if (log.isTraceEnabled()) {
- log.trace("pos " + glyphPos + ", transform " + glyphTransform);
- }
- if (DEBUG) {
- Shape sh = gv.getGlyphLogicalBounds(index);
- if (sh == null) {
- sh = new Ellipse2D.Double(glyphPos.getX(), glyphPos.getY(), 2, 2);
- }
- debugShapes.append(sh, false);
- }
-
- //Exact position of the glyph
- localTransform.setToIdentity();
- localTransform.translate(glyphPos.getX(), glyphPos.getY());
- if (glyphTransform != null) {
- localTransform.concatenate(glyphTransform);
- }
- localTransform.scale(1, -1);
-
- boolean yPosChanged = (prevPos == null
- || prevPos.getY() != glyphPos.getY()
- || glyphTransform != null);
- if (yPosChanged) {
- if (index > 0) {
- textUtil.writeTJ();
- textUtil.writeTextMatrix(localTransform);
- }
- } else {
- double xdiff = glyphPos.getX() - prevPos.getX();
- //Width of previous character
- Font font = textUtil.getCurrentFont();
- double cw = prevVisibleCharWidth;
- double effxdiff = (1000 * xdiff) - cw;
- if (effxdiff != 0) {
- double adjust = (-effxdiff / font.getFontSize());
- textUtil.adjustGlyphTJ(adjust * 1000);
- }
- if (log.isTraceEnabled()) {
- log.trace("==> x diff: " + xdiff + ", " + effxdiff
- + ", charWidth: " + cw);
- }
- }
- Font f = textUtil.selectFontForChar(ch);
- char paintChar = (CharUtilities.isAnySpace(ch) ? ' ' : ch);
- char mappedChar = f.mapChar(paintChar);
- boolean encodingChanging = false; // used for single byte
- if (!textUtil.isMultiByteFont(f.getFontName())) {
- int encoding = mappedChar / 256;
- mappedChar = (char) (mappedChar % 256);
- if (textUtil.getCurrentEncoding() != encoding) {
- textUtil.setCurrentEncoding(encoding);
- encodingChanging = true;
- }
- }
- if (f != textUtil.getCurrentFont() || encodingChanging) {
- textUtil.writeTJ();
- textUtil.setCurrentFont(f);
- textUtil.writeTf(f);
- textUtil.writeTextMatrix(localTransform);
- }
- textUtil.writeTJMappedChar(mappedChar);
-
- //Update last position
- prevPos = glyphPos;
- prevVisibleCharWidth = textUtil.getCurrentFont().getCharWidth(chars.charAt(index));
- }
+ @Override
+ protected void endTextObject() {
textUtil.writeTJ();
textUtil.endTextObject();
- pdf.restoreGraphicsState();
- if (DEBUG) {
- g2d.setStroke(new BasicStroke(0));
- g2d.setColor(Color.LIGHT_GRAY);
- g2d.draw(debugShapes);
- }
}
- private void applyColorAndPaint(TextPaintInfo tpi, PDFGraphics2D pdf) {
+ private void applyColorAndPaint(TextPaintInfo tpi) {
Paint fillPaint = tpi.fillPaint;
Paint strokePaint = tpi.strokePaint;
Stroke stroke = tpi.strokeStroke;
int fillAlpha = PDFGraphics2D.OPAQUE;
if (fillPaint instanceof Color) {
- Color col = (Color)fillPaint;
+ Color col = (Color) fillPaint;
pdf.applyColor(col, true);
fillAlpha = col.getAlpha();
}
if (strokePaint instanceof Color) {
- Color col = (Color)strokePaint;
+ Color col = (Color) strokePaint;
pdf.applyColor(col, false);
}
pdf.applyPaint(fillPaint, true);
@@ -243,4 +140,46 @@ class PDFTextPainter extends NativeTextPainter {
pdf.applyAlpha(fillAlpha, PDFGraphics2D.OPAQUE);
}
+ @Override
+ protected void positionGlyph(Point2D prevPos, Point2D glyphPos, boolean reposition) {
+ // TODO Glyph transforms could be refined so not every char has to be painted
+ // with its own TJ command (stretch/squeeze case could be optimized)
+ repositionNextGlyph = (prevPos == null
+ || prevPos.getY() != glyphPos.getY()
+ || reposition);
+ if (!repositionNextGlyph) {
+ double xdiff = glyphPos.getX() - prevPos.getX();
+ //Width of previous character
+ double cw = prevVisibleGlyphWidth;
+ double effxdiff = (1000 * xdiff) - cw;
+ if (effxdiff != 0) {
+ double adjust = (-effxdiff / font.getFontSize());
+ textUtil.adjustGlyphTJ(adjust * 1000);
+ }
+ }
+ }
+
+ @Override
+ protected void writeGlyph(char glyph, AffineTransform transform) {
+ prevVisibleGlyphWidth = font.getWidth(glyph);
+ boolean encodingChanging = false; // used for single byte
+ if (!textUtil.isMultiByteFont(font.getFontName())) {
+ int encoding = glyph / 256;
+ glyph = (char) (glyph % 256);
+ if (textUtil.getCurrentEncoding() != encoding) {
+ textUtil.setCurrentEncoding(encoding);
+ encodingChanging = true;
+ }
+ }
+ if (repositionNextGlyph || encodingChanging) {
+ textUtil.writeTJ();
+ if (font != textUtil.getCurrentFont() || encodingChanging) {
+ textUtil.setCurrentFont(font);
+ textUtil.writeTf(font);
+ }
+ textUtil.writeTextMatrix(transform);
+ }
+ textUtil.writeTJMappedChar(glyph);
+ }
+
}
diff --git a/src/java/org/apache/fop/svg/PDFTextUtil.java b/src/java/org/apache/fop/svg/PDFTextUtil.java
index 8325dce1f..d525ecefc 100644
--- a/src/java/org/apache/fop/svg/PDFTextUtil.java
+++ b/src/java/org/apache/fop/svg/PDFTextUtil.java
@@ -30,7 +30,6 @@ import org.apache.fop.fonts.Typeface;
public abstract class PDFTextUtil extends org.apache.fop.pdf.PDFTextUtil {
private FontInfo fontInfo;
- private Font[] fonts;
private Font font;
private int encoding;
@@ -50,23 +49,6 @@ public abstract class PDFTextUtil extends org.apache.fop.pdf.PDFTextUtil {
}
/**
- * Sets the current fonts for the text object. For every character, the suitable font will
- * be selected.
- * @param fonts the new fonts
- */
- public void setFonts(Font[] fonts) {
- this.fonts = fonts;
- }
-
- /**
- * Sets the current font for the text object.
- * @param font the new font
- */
- public void setFont(Font font) {
- setFonts(new Font[] {font});
- }
-
- /**
* Returns the current font in use.
* @return the current font or null if no font is currently active.
*/
@@ -123,27 +105,4 @@ public abstract class PDFTextUtil extends org.apache.fop.pdf.PDFTextUtil {
}
}
- /**
- * Selects a font from the font list suitable to display the given character.
- * @param ch the character
- * @return the recommended Font to use
- */
- public Font selectFontForChar(char ch) {
- for (int i = 0, c = fonts.length; i < c; i++) {
- if (fonts[i].hasChar(ch)) {
- return fonts[i];
- }
- }
- return fonts[0]; //TODO Maybe fall back to painting with shapes
- }
-
- /**
- * Writes a char to the "TJ-Buffer".
- * @param ch the unmapped character
- */
- public void writeTJChar(char ch) {
- char mappedChar = font.mapChar(ch);
- writeTJMappedChar(mappedChar);
- }
-
}
diff --git a/src/java/org/apache/fop/svg/PDFTranscoder.java b/src/java/org/apache/fop/svg/PDFTranscoder.java
index 9f0345657..1a3f154cb 100644
--- a/src/java/org/apache/fop/svg/PDFTranscoder.java
+++ b/src/java/org/apache/fop/svg/PDFTranscoder.java
@@ -31,7 +31,6 @@ import org.apache.avalon.framework.configuration.Configuration;
import org.apache.batik.bridge.BridgeContext;
import org.apache.batik.bridge.UnitProcessor;
-import org.apache.batik.bridge.UserAgent;
import org.apache.batik.ext.awt.RenderingHintsKeyExt;
import org.apache.batik.transcoder.TranscoderException;
import org.apache.batik.transcoder.TranscoderOutput;
@@ -39,6 +38,7 @@ import org.apache.batik.transcoder.image.ImageTranscoder;
import org.apache.fop.Version;
import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
/**
* <p>This class enables to transcode an input to a PDF document.</p>
@@ -88,7 +88,7 @@ public class PDFTranscoder extends AbstractFOPTranscoder {
/**
* {@inheritDoc}
*/
- protected UserAgent createUserAgent() {
+ protected FOPTranscoderUserAgent createUserAgent() {
return new AbstractFOPTranscoder.FOPTranscoderUserAgent() {
// The PDF stuff wants everything at 72dpi
public float getPixelUnitToMillimeter() {
@@ -131,6 +131,8 @@ public class PDFTranscoder extends AbstractFOPTranscoder {
} else {
graphics.setupDefaultFontInfo();
}
+ ((FOPTranscoderUserAgent) userAgent).setFontFamilyResolver(
+ new FOPFontFamilyResolverImpl(graphics.getFontInfo()));
} catch (Exception e) {
throw new TranscoderException(
"Error while setting up PDFDocumentGraphics2D", e);
diff --git a/src/java/org/apache/fop/svg/SVGUserAgent.java b/src/java/org/apache/fop/svg/SVGUserAgent.java
index d43552289..a265b4fef 100644
--- a/src/java/org/apache/fop/svg/SVGUserAgent.java
+++ b/src/java/org/apache/fop/svg/SVGUserAgent.java
@@ -21,6 +21,8 @@ package org.apache.fop.svg;
import java.awt.geom.AffineTransform;
+import org.apache.batik.gvt.font.FontFamilyResolver;
+
import org.apache.fop.apps.FOUserAgent;
/**
@@ -34,10 +36,11 @@ public class SVGUserAgent extends SimpleSVGUserAgent {
/**
* Creates a new SVGUserAgent.
* @param foUserAgent the FO user agent to associate with this SVG user agent
+ * @param fontFamilyResolver the font family resolver
* @param at the current transform
*/
- public SVGUserAgent(FOUserAgent foUserAgent, AffineTransform at) {
- super(foUserAgent.getSourcePixelUnitToMillimeter(), at);
+ public SVGUserAgent(FOUserAgent foUserAgent, FontFamilyResolver fontFamilyResolver, AffineTransform at) {
+ super(foUserAgent.getSourcePixelUnitToMillimeter(), at, fontFamilyResolver);
this.eventProducer = SVGEventProducer.Provider.get(foUserAgent.getEventBroadcaster());
}
@@ -45,8 +48,8 @@ public class SVGUserAgent extends SimpleSVGUserAgent {
* Creates a new SVGUserAgent.
* @param foUserAgent the FO user agent to associate with this SVG user agent
*/
- public SVGUserAgent(FOUserAgent foUserAgent) {
- this(foUserAgent, new AffineTransform());
+ public SVGUserAgent(FOUserAgent foUserAgent, FontFamilyResolver fontFamilyResolver) {
+ this(foUserAgent, fontFamilyResolver, new AffineTransform());
}
/**
diff --git a/src/java/org/apache/fop/svg/SimpleSVGUserAgent.java b/src/java/org/apache/fop/svg/SimpleSVGUserAgent.java
index 2b27945a4..42a18b17c 100644
--- a/src/java/org/apache/fop/svg/SimpleSVGUserAgent.java
+++ b/src/java/org/apache/fop/svg/SimpleSVGUserAgent.java
@@ -26,6 +26,7 @@ import java.awt.geom.Dimension2D;
import javax.xml.parsers.SAXParserFactory;
import org.apache.batik.bridge.UserAgentAdapter;
+import org.apache.batik.gvt.font.FontFamilyResolver;
/**
* A simple SVG user agent.
@@ -35,14 +36,18 @@ import org.apache.batik.bridge.UserAgentAdapter;
public class SimpleSVGUserAgent extends UserAgentAdapter {
private AffineTransform currentTransform = null;
+
private float pixelUnitToMillimeter = 0.0f;
+ private final FontFamilyResolver fontFamilyResolver;
+
/**
* Creates a new user agent.
* @param pixelUnitToMM the pixel to millimeter conversion factor currently in effect
* @param at the current transform
*/
- public SimpleSVGUserAgent(float pixelUnitToMM, AffineTransform at) {
+ public SimpleSVGUserAgent(float pixelUnitToMM, AffineTransform at, FontFamilyResolver fontFamilyResolver) {
+ this.fontFamilyResolver = fontFamilyResolver;
pixelUnitToMillimeter = pixelUnitToMM;
currentTransform = at;
}
@@ -122,5 +127,10 @@ public class SimpleSVGUserAgent extends UserAgentAdapter {
return new Dimension(100, 100);
}
+ @Override
+ public FontFamilyResolver getFontFamilyResolver() {
+ return fontFamilyResolver;
+ }
+
}
diff --git a/src/java/org/apache/fop/svg/font/AggregatingFontFamilyResolver.java b/src/java/org/apache/fop/svg/font/AggregatingFontFamilyResolver.java
new file mode 100644
index 000000000..313a6a74b
--- /dev/null
+++ b/src/java/org/apache/fop/svg/font/AggregatingFontFamilyResolver.java
@@ -0,0 +1,70 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.util.Arrays;
+import java.util.List;
+
+import org.apache.batik.gvt.font.FontFamilyResolver;
+import org.apache.batik.gvt.font.GVTFontFamily;
+
+public class AggregatingFontFamilyResolver implements FontFamilyResolver {
+
+ private final List<FontFamilyResolver> resolvers;
+
+ public AggregatingFontFamilyResolver(FontFamilyResolver... resolvers) {
+ this.resolvers = Arrays.<FontFamilyResolver>asList(resolvers);
+ }
+
+ public String lookup(String familyName) {
+ for (FontFamilyResolver resolver : resolvers) {
+ String lookup = resolver.lookup(familyName);
+ if (lookup != null) {
+ return lookup;
+ }
+ }
+ return null;
+ }
+
+ public GVTFontFamily resolve(String familyName) {
+ for (FontFamilyResolver resolver : resolvers) {
+ GVTFontFamily family = resolver.resolve(familyName);
+ if (family != null) {
+ return family;
+ }
+ }
+ return null;
+ }
+
+ public GVTFontFamily getDefault() {
+ return resolve("any");
+ }
+
+ public GVTFontFamily getFamilyThatCanDisplay(char c) {
+ for (FontFamilyResolver resolver : resolvers) {
+ GVTFontFamily family = resolver.getFamilyThatCanDisplay(c);
+ if (family != null) {
+ return family;
+ }
+ }
+ return null;
+ }
+
+}
diff --git a/src/java/org/apache/fop/svg/font/FOPFontFamilyResolver.java b/src/java/org/apache/fop/svg/font/FOPFontFamilyResolver.java
new file mode 100644
index 000000000..7af5f0b4f
--- /dev/null
+++ b/src/java/org/apache/fop/svg/font/FOPFontFamilyResolver.java
@@ -0,0 +1,31 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import org.apache.batik.gvt.font.FontFamilyResolver;
+
+public interface FOPFontFamilyResolver extends FontFamilyResolver {
+
+ FOPGVTFontFamily resolve(String familyName);
+
+ FOPGVTFontFamily getDefault();
+
+ FOPGVTFontFamily getFamilyThatCanDisplay(char c);
+}
diff --git a/src/java/org/apache/fop/svg/font/FOPFontFamilyResolverImpl.java b/src/java/org/apache/fop/svg/font/FOPFontFamilyResolverImpl.java
new file mode 100644
index 000000000..a9a631691
--- /dev/null
+++ b/src/java/org/apache/fop/svg/font/FOPFontFamilyResolverImpl.java
@@ -0,0 +1,67 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.util.Map;
+
+import org.apache.fop.fonts.Font;
+import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.fonts.FontTriplet;
+import org.apache.fop.fonts.Typeface;
+
+public class FOPFontFamilyResolverImpl implements FOPFontFamilyResolver {
+
+ private final FontInfo fontInfo;
+
+ public FOPFontFamilyResolverImpl(FontInfo fontInfo) {
+ this.fontInfo = fontInfo;
+ }
+
+ public String lookup(String familyName) {
+ // TODO Auto-generated method stub
+ throw new UnsupportedOperationException();
+ }
+
+ public FOPGVTFontFamily resolve(String familyName) {
+ FOPGVTFontFamily gvtFontFamily = null;
+ FontTriplet triplet = fontInfo.fontLookup(familyName, Font.STYLE_NORMAL, Font.WEIGHT_NORMAL);
+ if (fontInfo.hasFont(familyName, Font.STYLE_NORMAL, Font.WEIGHT_NORMAL)) {
+ gvtFontFamily = new FOPGVTFontFamily(fontInfo, familyName, triplet);
+ }
+ return gvtFontFamily;
+ }
+
+ public FOPGVTFontFamily getDefault() {
+ return resolve("any");
+ }
+
+ public FOPGVTFontFamily getFamilyThatCanDisplay(char c) {
+ Map<String, Typeface> fonts = fontInfo.getFonts();
+ for (Typeface font : fonts.values()) {
+ if (font.hasChar(c)) {
+ String fontFamily = font.getFamilyNames().iterator().next();
+ return new FOPGVTFontFamily(fontInfo, fontFamily,
+ new FontTriplet(fontFamily, Font.STYLE_NORMAL, Font.WEIGHT_NORMAL));
+ }
+ }
+ return null;
+ }
+
+}
diff --git a/src/java/org/apache/fop/svg/font/FOPGVTFont.java b/src/java/org/apache/fop/svg/font/FOPGVTFont.java
new file mode 100644
index 000000000..c55e2fa8b
--- /dev/null
+++ b/src/java/org/apache/fop/svg/font/FOPGVTFont.java
@@ -0,0 +1,159 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.awt.font.FontRenderContext;
+import java.text.CharacterIterator;
+import java.text.StringCharacterIterator;
+
+import org.apache.batik.gvt.font.GVTFont;
+import org.apache.batik.gvt.font.GVTFontFamily;
+import org.apache.batik.gvt.font.GVTGlyphVector;
+import org.apache.batik.gvt.font.GVTLineMetrics;
+
+import org.apache.fop.fonts.Font;
+import org.apache.fop.fonts.FontMetrics;
+
+public class FOPGVTFont implements GVTFont {
+
+ private final Font font;
+
+ private final GVTFontFamily fontFamily;
+
+ public FOPGVTFont(Font font, GVTFontFamily fontFamily) {
+ this.font = font;
+ this.fontFamily = fontFamily;
+ }
+
+ public Font getFont() {
+ return font;
+ }
+
+ public boolean canDisplay(char c) {
+ return font.hasChar(c);
+ }
+
+ public int canDisplayUpTo(char[] text, int start, int limit) {
+ for (int i = start; i < limit; i++) {
+ if (!canDisplay(text[i])) {
+ return i;
+ }
+ }
+ return -1;
+ }
+
+
+ public int canDisplayUpTo(CharacterIterator iter, int start, int limit) {
+ for (char c = iter.setIndex(start); iter.getIndex() < limit; c = iter.next()) {
+ if (!canDisplay(c)) {
+ return iter.getIndex();
+ }
+ }
+ return -1;
+ }
+
+ public int canDisplayUpTo(String str) {
+ for (int i = 0; i < str.length(); i++) {
+ if (!canDisplay(str.charAt(i))) {
+ return i;
+ }
+ }
+ return -1;
+ }
+
+ public GVTGlyphVector createGlyphVector(FontRenderContext frc, char[] chars) {
+ return createGlyphVector(frc, new String(chars));
+ }
+
+ public GVTGlyphVector createGlyphVector(FontRenderContext frc, CharacterIterator ci) {
+ // TODO Batik does manual glyph shaping for Arabic. Replace with complex scripts implementation
+ return new FOPGVTGlyphVector(this, ci, frc);
+ }
+
+ public GVTGlyphVector createGlyphVector(FontRenderContext frc,
+ int[] glyphCodes,
+ CharacterIterator ci) {
+ throw new UnsupportedOperationException("Not implemented");
+ }
+
+ public GVTGlyphVector createGlyphVector(FontRenderContext frc, String str) {
+ StringCharacterIterator sci = new StringCharacterIterator(str);
+ return createGlyphVector(frc, sci);
+ }
+
+ public FOPGVTFont deriveFont(float size) {
+ throw new UnsupportedOperationException("Not implemented");
+ }
+
+ public String getFamilyName() {
+ return fontFamily.getFamilyName();
+ }
+
+ public GVTLineMetrics getLineMetrics(char[] chars, int beginIndex, int limit, FontRenderContext frc) {
+ return getLineMetrics(limit - beginIndex);
+ }
+
+ GVTLineMetrics getLineMetrics(int numChars) {
+ numChars = numChars < 0 ? 0 : numChars;
+ FontMetrics metrics = font.getFontMetrics();
+ int size = font.getFontSize();
+ return new GVTLineMetrics(
+ metrics.getCapHeight(size) / 1000000f,
+ java.awt.Font.ROMAN_BASELINE, // Not actually used by Batik
+ null, // Not actually used by Batik
+ -metrics.getDescender(size) / 1000000f,
+ 0, // Not actually used by Batik
+ 0, // Not actually used by Batik
+ numChars,
+ -metrics.getStrikeoutPosition(size) / 1000000f,
+ metrics.getStrikeoutThickness(size) / 1000000f,
+ -metrics.getUnderlinePosition(size) / 1000000f,
+ metrics.getUnderlineThickness(size) / 1000000f,
+ -metrics.getCapHeight(size) / 1000000f, // Because this is what Batik does in GVTLineMetrics
+ metrics.getUnderlineThickness(size) / 1000000f);
+ }
+
+ public GVTLineMetrics getLineMetrics(CharacterIterator ci, int beginIndex, int limit,
+ FontRenderContext frc) {
+ return getLineMetrics(limit - beginIndex);
+ }
+
+ public GVTLineMetrics getLineMetrics(String str, FontRenderContext frc) {
+ return getLineMetrics(str.length());
+ }
+
+ public GVTLineMetrics getLineMetrics(String str, int beginIndex, int limit, FontRenderContext frc) {
+ return getLineMetrics(limit - beginIndex);
+ }
+
+ public float getSize() {
+ return font.getFontSize() / 1000f;
+ }
+
+ public float getVKern(int glyphCode1, int glyphCode2) {
+ return 0;
+ }
+
+ public float getHKern(int glyphCode1, int glyphCode2) {
+ // TODO Cannot be implemented until getKernValue takes glyph indices instead of character codes
+ return 0;
+ }
+
+}
diff --git a/src/java/org/apache/fop/svg/font/FOPGVTFontFamily.java b/src/java/org/apache/fop/svg/font/FOPGVTFontFamily.java
new file mode 100644
index 000000000..036b560a0
--- /dev/null
+++ b/src/java/org/apache/fop/svg/font/FOPGVTFontFamily.java
@@ -0,0 +1,73 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.awt.font.TextAttribute;
+import java.text.AttributedCharacterIterator;
+import java.util.Map;
+
+import org.apache.batik.gvt.font.GVTFontFace;
+import org.apache.batik.gvt.font.GVTFontFamily;
+
+import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.fonts.FontTriplet;
+import org.apache.fop.svg.ACIUtils;
+
+public class FOPGVTFontFamily implements GVTFontFamily {
+
+ private final FontInfo fontInfo;
+
+ private final FontTriplet fontTriplet;
+
+ private final String familyName;
+
+ public FOPGVTFontFamily(FontInfo fontInfo, String familyName, FontTriplet triplet) {
+ this.fontInfo = fontInfo;
+ this.fontTriplet = triplet;
+ this.familyName = familyName;
+ }
+
+ public String getFamilyName() {
+ return familyName;
+ }
+
+ public GVTFontFace getFontFace() {
+ // TODO Auto-generated method stub
+ throw new UnsupportedOperationException();
+ }
+
+ public FOPGVTFont deriveFont(float size, AttributedCharacterIterator aci) {
+ return deriveFont(size, aci.getAttributes());
+ }
+
+ public FOPGVTFont deriveFont(float size, @SuppressWarnings("rawtypes") Map attrs) {
+ Float fontWeight = (Float) attrs.get(TextAttribute.WEIGHT);
+ int weight = fontWeight == null ? fontTriplet.getWeight() : ACIUtils.toCSSWeight(fontWeight);
+ Float fontStyle = (Float) attrs.get(TextAttribute.POSTURE);
+ String style = fontStyle == null ? fontTriplet.getStyle() : ACIUtils.toStyle(fontStyle);
+ FontTriplet triplet = fontInfo.fontLookup(fontTriplet.getName(), style, weight);
+ return new FOPGVTFont(fontInfo.getFontInstance(triplet, (int) (size * 1000)), this);
+ }
+
+ public boolean isComplex() {
+ return false;
+ }
+
+}
diff --git a/src/java/org/apache/fop/svg/font/FOPGVTGlyphVector.java b/src/java/org/apache/fop/svg/font/FOPGVTGlyphVector.java
new file mode 100644
index 000000000..3567bb508
--- /dev/null
+++ b/src/java/org/apache/fop/svg/font/FOPGVTGlyphVector.java
@@ -0,0 +1,339 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.awt.Graphics2D;
+import java.awt.Rectangle;
+import java.awt.Shape;
+import java.awt.font.FontRenderContext;
+import java.awt.font.GlyphJustificationInfo;
+import java.awt.font.GlyphMetrics;
+import java.awt.geom.AffineTransform;
+import java.awt.geom.GeneralPath;
+import java.awt.geom.Point2D;
+import java.awt.geom.Rectangle2D;
+import java.text.AttributedCharacterIterator;
+import java.text.CharacterIterator;
+import java.text.StringCharacterIterator;
+import java.util.Arrays;
+
+import org.apache.batik.gvt.font.GVTFont;
+import org.apache.batik.gvt.font.GVTGlyphMetrics;
+import org.apache.batik.gvt.font.GVTGlyphVector;
+import org.apache.batik.gvt.font.GVTLineMetrics;
+
+import org.apache.fop.fonts.Font;
+import org.apache.fop.fonts.FontMetrics;
+import org.apache.fop.fonts.GlyphMapping;
+import org.apache.fop.fonts.TextFragment;
+import org.apache.fop.traits.MinOptMax;
+
+class FOPGVTGlyphVector implements GVTGlyphVector {
+
+ private final CharacterIterator charIter;
+
+ private final FOPGVTFont font;
+
+ private final int fontSize;
+
+ private final FontMetrics fontMetrics;
+
+ private final FontRenderContext frc;
+
+ private int[] glyphs;
+
+ private float[] positions;
+
+ private Rectangle2D[] boundingBoxes;
+
+ private GeneralPath outline;
+
+ private AffineTransform[] glyphTransforms;
+
+ private boolean[] glyphVisibilities;
+
+ private Rectangle2D logicalBounds;
+
+ FOPGVTGlyphVector(FOPGVTFont font, final CharacterIterator iter, FontRenderContext frc) {
+ this.charIter = iter;
+ this.font = font;
+ Font f = font.getFont();
+ this.fontSize = f.getFontSize();
+ this.fontMetrics = f.getFontMetrics();
+ this.frc = frc;
+ }
+
+ public void performDefaultLayout() {
+ Font f = font.getFont();
+ TextFragment text = new SVGTextFragment(charIter);
+ MinOptMax letterSpaceIPD = MinOptMax.ZERO;
+ MinOptMax[] letterSpaceAdjustments = new MinOptMax[charIter.getEndIndex() - charIter.getBeginIndex()];
+ GlyphMapping mapping = GlyphMapping.doGlyphMapping(text, charIter.getBeginIndex(), charIter.getEndIndex(),
+ f, letterSpaceIPD, letterSpaceAdjustments, '\0', '\0', false, 0 /* TODO */);
+ glyphs = buildGlyphs(f, mapping.mapping != null ? new StringCharacterIterator(mapping.mapping) : charIter);
+ buildGlyphPositions(mapping, letterSpaceAdjustments);
+ this.glyphVisibilities = new boolean[this.glyphs.length];
+ Arrays.fill(glyphVisibilities, true);
+ this.glyphTransforms = new AffineTransform[this.glyphs.length];
+ }
+
+ private static class SVGTextFragment implements TextFragment {
+
+ private final CharacterIterator charIter;
+
+ SVGTextFragment(CharacterIterator charIter) {
+ this.charIter = charIter;
+ }
+
+ public CharSequence subSequence(int startIndex, int endIndex) {
+ StringBuilder sb = new StringBuilder();
+ for (char c = charIter.first(); c != CharacterIterator.DONE; c = charIter.next()) {
+ sb.append(c);
+ }
+ return sb.toString();
+ }
+
+ public String getScript() {
+ return "DFLT"; // TODO pass on script value from SVG
+ }
+
+ public String getLanguage() {
+ return "dflt"; // TODO pass on language value from SVG
+ }
+
+ public char charAt(int index) {
+ return charIter.setIndex(index - charIter.getBeginIndex());
+ }
+ }
+
+ private int[] buildGlyphs(Font font, final CharacterIterator charIter) {
+ int[] glyphs = new int[charIter.getEndIndex() - charIter.getBeginIndex()];
+ int index = 0;
+ for (char c = charIter.first(); c != CharacterIterator.DONE; c = charIter.next()) {
+ glyphs[index] = font.mapChar(c);
+ index++;
+ }
+ return glyphs;
+ }
+
+ private void buildGlyphPositions(GlyphMapping ai, MinOptMax[] letterSpaceAdjustments) {
+ positions = new float[2 * glyphs.length + 2];
+ if (ai.gposAdjustments != null) {
+ assert ai.gposAdjustments.length == glyphs.length;
+ for (int glyphIndex = 0; glyphIndex < glyphs.length; glyphIndex++) {
+ int n = 2 * glyphIndex;
+ if (ai.gposAdjustments[glyphIndex] != null) {
+ for (int p = 0; p < 4; p++) {
+ positions[n + p] += ai.gposAdjustments[glyphIndex][p] / 1000f;
+ }
+ }
+ positions[n + 2] += positions[n] + getGlyphWidth(glyphIndex);
+ }
+ } else {
+ for (int i = 0, n = 2; i < glyphs.length; i++, n += 2) {
+ int kern = i < glyphs.length - 1 && letterSpaceAdjustments[i + 1] != null
+ ? letterSpaceAdjustments[i + 1].getOpt()
+ : 0;
+ positions[n] = positions[n - 2] + getGlyphWidth(i) + kern / 1000f;
+ positions[n + 1] = 0;
+ }
+ }
+ }
+
+ private float getGlyphWidth(int index) {
+ return fontMetrics.getWidth(glyphs[index], fontSize) / 1000000f;
+ }
+
+ public GVTFont getFont() {
+ return font;
+ }
+
+ public FontRenderContext getFontRenderContext() {
+ return frc;
+ }
+
+ public int getGlyphCode(int glyphIndex) {
+ return glyphs[glyphIndex];
+ }
+
+ public int[] getGlyphCodes(int beginGlyphIndex, int numEntries,
+ int[] codeReturn) {
+ if (codeReturn == null) {
+ codeReturn = new int[numEntries];
+ }
+ System.arraycopy(glyphs, beginGlyphIndex, codeReturn, 0, numEntries);
+ return codeReturn;
+ }
+
+ public GlyphJustificationInfo getGlyphJustificationInfo(int glyphIndex) {
+ // TODO Auto-generated method stub
+ throw new UnsupportedOperationException();
+ }
+
+ public Shape getGlyphLogicalBounds(int glyphIndex) {
+ GVTGlyphMetrics metrics = getGlyphMetrics(glyphIndex);
+ Point2D pos = getGlyphPosition(glyphIndex);
+ GVTLineMetrics fontMetrics = font.getLineMetrics(0);
+ Rectangle2D bounds = new Rectangle2D.Float(0, -fontMetrics.getDescent(), metrics.getHorizontalAdvance(),
+ fontMetrics.getAscent() + fontMetrics.getDescent());
+ AffineTransform t = AffineTransform.getTranslateInstance(pos.getX(), pos.getY());
+ AffineTransform transf = getGlyphTransform(glyphIndex);
+ if (transf != null) {
+ t.concatenate(transf);
+ }
+ t.scale(1, -1); // Translate from glyph coordinate system to user
+ return t.createTransformedShape(bounds);
+ }
+
+ public GVTGlyphMetrics getGlyphMetrics(int glyphIndex) {
+ Rectangle2D bbox = getBoundingBoxes()[glyphIndex];
+ return new GVTGlyphMetrics(positions[2 * (glyphIndex + 1)] - positions[2 * glyphIndex],
+ (fontMetrics.getAscender(fontSize) - fontMetrics.getDescender(fontSize)) / 1000000f,
+ bbox, GlyphMetrics.STANDARD);
+ }
+
+ public Shape getGlyphOutline(int glyphIndex) {
+ Shape glyphBox = getBoundingBoxes()[glyphIndex];
+ AffineTransform tr = AffineTransform.getTranslateInstance(positions[glyphIndex * 2],
+ positions[glyphIndex * 2 + 1]);
+ AffineTransform glyphTransform = getGlyphTransform(glyphIndex);
+ if (glyphTransform != null) {
+ tr.concatenate(glyphTransform);
+ }
+ return tr.createTransformedShape(glyphBox);
+ }
+
+ public Rectangle2D getGlyphCellBounds(int glyphIndex) {
+ // TODO Auto-generated method stub
+ throw new UnsupportedOperationException();
+ }
+
+ public Point2D getGlyphPosition(int glyphIndex) {
+ int positionIndex = glyphIndex * 2;
+ return new Point2D.Float(positions[positionIndex], positions[positionIndex + 1]);
+ }
+
+ public float[] getGlyphPositions(int beginGlyphIndex, int numEntries, float[] positionReturn) {
+ if (positionReturn == null) {
+ positionReturn = new float[numEntries * 2];
+ }
+ System.arraycopy(positions, beginGlyphIndex * 2, positionReturn, 0, numEntries * 2);
+ return positionReturn;
+ }
+
+ public AffineTransform getGlyphTransform(int glyphIndex) {
+ return glyphTransforms[glyphIndex];
+ }
+
+ public Shape getGlyphVisualBounds(int glyphIndex) {
+ Rectangle2D bbox = getBoundingBoxes()[glyphIndex];
+ Point2D pos = getGlyphPosition(glyphIndex);
+ AffineTransform t = AffineTransform.getTranslateInstance(pos.getX(), pos.getY());
+ AffineTransform transf = getGlyphTransform(glyphIndex);
+ if (transf != null) {
+ t.concatenate(transf);
+ }
+ return t.createTransformedShape(bbox);
+ }
+
+ public Rectangle2D getLogicalBounds() {
+ if (logicalBounds == null) {
+ GeneralPath logicalBoundsPath = new GeneralPath();
+ for (int i = 0; i < getNumGlyphs(); i++) {
+ Shape glyphLogicalBounds = getGlyphLogicalBounds(i);
+ logicalBoundsPath.append(glyphLogicalBounds, false);
+ }
+ logicalBounds = logicalBoundsPath.getBounds2D();
+ }
+ return logicalBounds;
+ }
+
+ public int getNumGlyphs() {
+ return glyphs.length;
+ }
+
+ public Shape getOutline() {
+ if (outline == null) {
+ outline = new GeneralPath();
+ for (int i = 0; i < glyphs.length; i++) {
+ outline.append(getGlyphOutline(i), false);
+ }
+ }
+ return outline;
+ }
+
+ public Shape getOutline(float x, float y) {
+ // TODO Auto-generated method stub
+ throw new UnsupportedOperationException();
+ }
+
+ public Rectangle2D getGeometricBounds() {
+ // TODO Auto-generated method stub
+ throw new UnsupportedOperationException();
+ }
+
+ public Rectangle2D getBounds2D(AttributedCharacterIterator aci) {
+ return getOutline().getBounds2D();
+ }
+
+ public void setGlyphPosition(int glyphIndex, Point2D newPos) {
+ int idx = glyphIndex * 2;
+ positions[idx] = (float) newPos.getX();
+ positions[idx + 1] = (float) newPos.getY();
+ }
+
+ public void setGlyphTransform(int glyphIndex, AffineTransform newTX) {
+ glyphTransforms[glyphIndex] = newTX;
+ }
+
+ public void setGlyphVisible(int glyphIndex, boolean visible) {
+ glyphVisibilities[glyphIndex] = visible;
+ }
+
+ public boolean isGlyphVisible(int glyphIndex) {
+ return glyphVisibilities[glyphIndex];
+ }
+
+ public int getCharacterCount(int startGlyphIndex, int endGlyphIndex) {
+ // TODO Not that simple if complex scripts are involved
+ return endGlyphIndex - startGlyphIndex + 1;
+ }
+
+ public void draw(Graphics2D graphics2d, AttributedCharacterIterator aci) {
+ // NOP
+ }
+
+ private Rectangle2D[] getBoundingBoxes() {
+ if (boundingBoxes == null) {
+ buildBoundingBoxes();
+ }
+ return boundingBoxes;
+ }
+
+ private void buildBoundingBoxes() {
+ boundingBoxes = new Rectangle2D[glyphs.length];
+ for (int i = 0; i < glyphs.length; i++) {
+ Rectangle bbox = fontMetrics.getBoundingBox(glyphs[i], fontSize);
+ boundingBoxes[i] = new Rectangle2D.Float(bbox.x / 1000000f, -(bbox.y + bbox.height) / 1000000f,
+ bbox.width / 1000000f, bbox.height / 1000000f);
+ }
+ }
+
+}
diff --git a/src/java/org/apache/fop/svg/font/FilteringFontFamilyResolver.java b/src/java/org/apache/fop/svg/font/FilteringFontFamilyResolver.java
new file mode 100644
index 000000000..319d6b2b8
--- /dev/null
+++ b/src/java/org/apache/fop/svg/font/FilteringFontFamilyResolver.java
@@ -0,0 +1,47 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+
+public class FilteringFontFamilyResolver implements FOPFontFamilyResolver {
+
+ private final FOPFontFamilyResolver delegate;
+
+ public FilteringFontFamilyResolver(FOPFontFamilyResolver delegate) {
+ this.delegate = delegate;
+ }
+
+ public String lookup(String familyName) {
+ return delegate.lookup(familyName);
+ }
+
+ public FOPGVTFontFamily resolve(String familyName) {
+ return delegate.resolve(familyName);
+ }
+
+ public FOPGVTFontFamily getDefault() {
+ return delegate.getDefault();
+ }
+
+ public FOPGVTFontFamily getFamilyThatCanDisplay(char c) {
+ return delegate.getFamilyThatCanDisplay(c);
+ }
+
+}
diff --git a/test/java/org/apache/fop/afp/fonts/IntegerKeyStoreTestCase.java b/test/java/org/apache/fop/afp/fonts/IntegerKeyStoreTestCase.java
new file mode 100644
index 000000000..96c0618a7
--- /dev/null
+++ b/test/java/org/apache/fop/afp/fonts/IntegerKeyStoreTestCase.java
@@ -0,0 +1,48 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.afp.fonts;
+
+import org.junit.Test;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertNull;
+import static org.junit.Assert.fail;
+
+public class IntegerKeyStoreTestCase {
+
+ @Test
+ public void getAndPut() {
+ IntegerKeyStore<Integer> sut = new IntegerKeyStore<Integer>();
+ assertNull(sut.get(0));
+ sut.put(0, 0);
+ assertEquals(Integer.valueOf(0), sut.get(0));
+ sut.put(0, 1);
+ assertEquals(Integer.valueOf(1), sut.get(0));
+ sut.put(0, null);
+ assertNull(sut.get(0));
+ try {
+ sut.put(-1, 0);
+ fail("Negative index");
+ } catch (IndexOutOfBoundsException e) {
+ // As expected
+ }
+ }
+
+}
diff --git a/test/java/org/apache/fop/fonts/DejaVuLGCSerifTestCase.java b/test/java/org/apache/fop/fonts/DejaVuLGCSerifTestCase.java
index 20212b002..4c89d00da 100644
--- a/test/java/org/apache/fop/fonts/DejaVuLGCSerifTestCase.java
+++ b/test/java/org/apache/fop/fonts/DejaVuLGCSerifTestCase.java
@@ -24,11 +24,11 @@ import java.io.File;
import org.junit.Before;
import org.junit.Test;
+import static org.junit.Assert.assertEquals;
+
import org.apache.fop.apps.io.InternalResourceResolver;
import org.apache.fop.apps.io.ResourceResolverFactory;
-import static org.junit.Assert.assertEquals;
-
/**
*
*/
@@ -58,4 +58,17 @@ public class DejaVuLGCSerifTestCase {
public void testFontName() {
assertEquals("DejaVuLGCSerif", font.getFontName());
}
+
+ @Test
+ public void testUnderline() {
+ assertEquals(-840, font.getUnderlinePosition(10));
+ assertEquals(430, font.getUnderlineThickness(10));
+ }
+
+ @Test
+ public void testStrikeout() {
+ assertEquals(2340, font.getStrikeoutPosition(10));
+ assertEquals(490, font.getStrikeoutThickness(10));
+ }
+
}
diff --git a/test/java/org/apache/fop/fonts/FontEventProcessingTestCase.java b/test/java/org/apache/fop/fonts/FontEventProcessingTestCase.java
index 509ee56f4..b4271ba60 100644
--- a/test/java/org/apache/fop/fonts/FontEventProcessingTestCase.java
+++ b/test/java/org/apache/fop/fonts/FontEventProcessingTestCase.java
@@ -52,13 +52,4 @@ public class FontEventProcessingTestCase {
MimeConstants.MIME_PDF);
}
- @Test
- public void testSVGFontStrokedAsShapes() throws Exception {
- // svg-fonts.fo embeds two fonts; one that is present in the system and the other is not; the
- // missing font is stroked as shapes while the fonts that exists is stroked as text
- InputStream inStream = getClass().getResourceAsStream("svg-fonts.fo");
- eventsTests.doTest(inStream, null, FontEventProducer.class.getName() + ".svgTextStrokedAsShapes",
- MimeConstants.MIME_PDF);
- }
-
}
diff --git a/test/java/org/apache/fop/fonts/svg-fonts.fo b/test/java/org/apache/fop/fonts/svg-fonts.fo
deleted file mode 100644
index 0c5f3f599..000000000
--- a/test/java/org/apache/fop/fonts/svg-fonts.fo
+++ /dev/null
@@ -1,37 +0,0 @@
-<?xml version="1.0" standalone="no"?>
-<fo:root xmlns:fo="http://www.w3.org/1999/XSL/Format" xmlns:svg="http://www.w3.org/2000/svg">
- <fo:layout-master-set>
- <fo:simple-page-master master-name="page">
- <fo:region-body />
- </fo:simple-page-master>
- </fo:layout-master-set>
- <fo:page-sequence master-reference="page">
- <fo:flow flow-name="xsl-region-body">
- <fo:block>
- <fo:instream-foreign-object>
- <svg:svg width="250" height="50">
- <svg:font horiz-adv-x="1000">
- <svg:font-face font-family="Missing" units-per-em="1000" underline-position="-100"
- underline-thickness="50" />
- <svg:glyph unicode="A" horiz-adv-x="686" d="M162,186l362,0l78,-186l84,0l-308,708l-70,0l-308,-708l84,0M343,624l153,-372l-307,0z" />
- <svg:glyph unicode="C" horiz-adv-x="704" d="M620,154C567,72 491,48 417,48C235,48 126,191 126,354C126,517 235,660 417,660C492,660 571,613 599,567l63,47C600,693 505,726 417,726C206,726 48,569 48,354C48,139 206,-18 417,-18C534,-18 632,39 679,112z" />
- <svg:glyph unicode="F" horiz-adv-x="556" d="M168,335l330,0l0,66l-330,0l0,241l355,0l0,66l-427,0l0,-708l72,0z" />
- <svg:glyph unicode="G" horiz-adv-x="778" d="M673,631C610,694 529,726 417,726C206,726 48,569 48,354C48,139 206,-18 417,-18C503,-18 606,7 685,54l0,347l-241,0l0,-66l169,0l0,-237C560,68 490,48 417,48C235,48 126,191 126,354C126,517 235,660 417,660C504,660 571,629 619,578z" />
- </svg:font>
- <svg:font horiz-adv-x="1000">
- <!-- this is not Helvetica but it is here to increase coverage and show the code takes expected path -->
- <svg:font-face font-family="Helvetica" units-per-em="1000" underline-position="-100"
- underline-thickness="50" />
- <svg:glyph unicode="A" horiz-adv-x="686" d="M162,186l362,0l78,-186l84,0l-308,708l-70,0l-308,-708l84,0M343,624l153,-372l-307,0z" />
- <svg:glyph unicode="C" horiz-adv-x="704" d="M620,154C567,72 491,48 417,48C235,48 126,191 126,354C126,517 235,660 417,660C492,660 571,613 599,567l63,47C600,693 505,726 417,726C206,726 48,569 48,354C48,139 206,-18 417,-18C534,-18 632,39 679,112z" />
- <svg:glyph unicode="F" horiz-adv-x="556" d="M168,335l330,0l0,66l-330,0l0,241l355,0l0,66l-427,0l0,-708l72,0z" />
- <svg:glyph unicode="G" horiz-adv-x="778" d="M673,631C610,694 529,726 417,726C206,726 48,569 48,354C48,139 206,-18 417,-18C503,-18 606,7 685,54l0,347l-241,0l0,-66l169,0l0,-237C560,68 490,48 417,48C235,48 126,191 126,354C126,517 235,660 417,660C504,660 571,629 619,578z" />
- </svg:font>
- <svg:text x="20" y="20" font-family="Missing" font-size="12">ACFG</svg:text>
- <svg:text x="20" y="40" font-family="Helvetica" font-size="12">ACFG</svg:text>
- </svg:svg>
- </fo:instream-foreign-object>
- </fo:block>
- </fo:flow>
- </fo:page-sequence>
-</fo:root>
diff --git a/test/java/org/apache/fop/fonts/truetype/TTFFileTestCase.java b/test/java/org/apache/fop/fonts/truetype/TTFFileTestCase.java
index e04347032..2d583dd3e 100644
--- a/test/java/org/apache/fop/fonts/truetype/TTFFileTestCase.java
+++ b/test/java/org/apache/fop/fonts/truetype/TTFFileTestCase.java
@@ -26,12 +26,12 @@ import java.util.Map;
import org.junit.Test;
-import org.apache.fop.fonts.truetype.TTFFile.PostScriptVersion;
-
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
+import org.apache.fop.fonts.truetype.TTFFile.PostScriptVersion;
+
/**
* Class for testing org.apache.fop.fonts.truetype.TTFFile
*/
@@ -426,6 +426,24 @@ public class TTFFileTestCase {
assertEquals(true, droidmonoTTFFile.isEmbeddable());
}
+ /** Underline position and thickness. */
+ @Test
+ public void testUnderline() {
+ assertEquals(-63, dejavuTTFFile.getUnderlinePosition());
+ assertEquals(43, dejavuTTFFile.getUnderlineThickness());
+ assertEquals(-75, droidmonoTTFFile.getUnderlinePosition());
+ assertEquals(49, droidmonoTTFFile.getUnderlineThickness());
+ }
+
+ /** Strikeout position and thickness. */
+ @Test
+ public void testStrikeout() {
+ assertEquals(258, dejavuTTFFile.getStrikeoutPosition());
+ assertEquals(49, dejavuTTFFile.getStrikeoutThickness());
+ assertEquals(243, droidmonoTTFFile.getStrikeoutPosition());
+ assertEquals(49, droidmonoTTFFile.getStrikeoutThickness());
+ }
+
/**
* Test readFont() - Add implementation if necessary.
*/
diff --git a/test/java/org/apache/fop/fonts/type1/AFMParserTestCase.java b/test/java/org/apache/fop/fonts/type1/AFMParserTestCase.java
index 93443a0d9..31a613567 100644
--- a/test/java/org/apache/fop/fonts/type1/AFMParserTestCase.java
+++ b/test/java/org/apache/fop/fonts/type1/AFMParserTestCase.java
@@ -19,9 +19,6 @@
package org.apache.fop.fonts.type1;
-import static org.junit.Assert.assertEquals;
-import static org.junit.Assert.assertTrue;
-
import java.awt.Rectangle;
import java.io.IOException;
import java.io.InputStream;
@@ -29,6 +26,9 @@ import java.util.List;
import org.junit.Test;
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertTrue;
+
/**
* Test case for {@link AFMParser}.
*/
@@ -128,4 +128,13 @@ public class AFMParserTestCase {
private boolean objectEquals(Object o1, Object o2) {
return o1 == null ? o2 == null : (o1 == o2 || o1.equals(o2));
}
+
+ @Test
+ public void testUnderlinePositionAndThickness() throws IOException {
+ AFMFile afm = sut.parse(getClass().getResourceAsStream("underline.afm"), null);
+ AFMWritingDirectionMetrics metrics = afm.getWritingDirectionMetrics(0);
+ assertEquals(-96, metrics.getUnderlinePosition());
+ assertEquals(58, metrics.getUnderlineThickness());
+ }
+
}
diff --git a/test/java/org/apache/fop/fonts/type1/underline.afm b/test/java/org/apache/fop/fonts/type1/underline.afm
new file mode 100644
index 000000000..8137b41eb
--- /dev/null
+++ b/test/java/org/apache/fop/fonts/type1/underline.afm
@@ -0,0 +1,4 @@
+StartFontMetrics 2.0
+UnderlinePosition -96
+UnderlineThickness 58
+EndFontMetrics
diff --git a/test/java/org/apache/fop/svg/NativeTextPainterTest.java b/test/java/org/apache/fop/svg/NativeTextPainterTest.java
new file mode 100644
index 000000000..1c2c3b582
--- /dev/null
+++ b/test/java/org/apache/fop/svg/NativeTextPainterTest.java
@@ -0,0 +1,78 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg;
+
+import java.awt.Graphics2D;
+import java.awt.geom.AffineTransform;
+import java.io.File;
+import java.io.IOException;
+
+import org.w3c.dom.Document;
+
+import org.apache.batik.bridge.BridgeContext;
+import org.apache.batik.bridge.GVTBuilder;
+import org.apache.batik.dom.svg.SAXSVGDocumentFactory;
+import org.apache.batik.gvt.GraphicsNode;
+import org.apache.batik.gvt.TextPainter;
+
+import org.apache.fop.apps.FOUserAgent;
+import org.apache.fop.apps.FopFactory;
+import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.fonts.base14.Base14FontCollection;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
+
+abstract class NativeTextPainterTest {
+
+ protected final void runTest(String testcase, OperatorValidator validator) throws Exception {
+ FontInfo fontInfo = createFontInfo();
+ BridgeContext bridgeContext = createBridgeContext(fontInfo);
+ GraphicsNode svg = loadSVG(bridgeContext, testcase);
+ Graphics2D g2d = createGraphics2D(fontInfo, validator);
+ svg.paint(g2d);
+ validator.end();
+ }
+
+ private FontInfo createFontInfo() {
+ FontInfo fontInfo = new FontInfo();
+ new Base14FontCollection(true).setup(0, fontInfo);
+ return fontInfo;
+ }
+
+ private BridgeContext createBridgeContext(FontInfo fontInfo) {
+ FOUserAgent userAgent = FopFactory.newInstance(new File(".").toURI()).newFOUserAgent();
+ SVGUserAgent svgUserAgent = new SVGUserAgent(userAgent, new FOPFontFamilyResolverImpl(fontInfo),
+ new AffineTransform());
+ BridgeContext bridgeContext = new BridgeContext(svgUserAgent);
+ bridgeContext.setTextPainter(createTextPainter(fontInfo));
+ return bridgeContext;
+ }
+
+ protected abstract TextPainter createTextPainter(FontInfo fontInfo);
+
+ private GraphicsNode loadSVG(BridgeContext bridgeContext, String resourceName) throws IOException {
+ SAXSVGDocumentFactory factory = new SAXSVGDocumentFactory(null);
+ Document svg = factory.createDocument(null, getClass().getResourceAsStream(resourceName));
+ GVTBuilder builder = new GVTBuilder();
+ return builder.build(bridgeContext, svg);
+ }
+
+ protected abstract Graphics2D createGraphics2D(FontInfo fontInfo, OperatorValidator validator);
+
+}
diff --git a/test/java/org/apache/fop/svg/OperatorValidator.java b/test/java/org/apache/fop/svg/OperatorValidator.java
new file mode 100644
index 000000000..959adc9a1
--- /dev/null
+++ b/test/java/org/apache/fop/svg/OperatorValidator.java
@@ -0,0 +1,108 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg;
+
+import java.util.LinkedList;
+import java.util.Queue;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertTrue;
+
+class OperatorValidator {
+
+ private static interface Match {
+
+ boolean match(String line);
+ }
+
+ private static class MatchSequence implements OperatorValidator.Match {
+
+ private final Queue<OperatorValidator.Match> expectedMatches = new LinkedList<OperatorValidator.Match>();
+
+ private OperatorValidator.Match currentMatch;
+
+ private static final OperatorValidator.Match FINAL_MATCH = new Match() {
+
+ public boolean match(String line) {
+ return false;
+ }
+ };
+
+ public boolean isExhausted() {
+ return currentMatch == FINAL_MATCH;
+ }
+
+ public void addMatch(OperatorValidator.Match match) {
+ if (currentMatch == null) {
+ currentMatch = match;
+ } else {
+ expectedMatches.add(match);
+ }
+ }
+
+ public boolean match(String line) {
+ boolean match = currentMatch.match(line);
+ if (match) {
+ if(expectedMatches.isEmpty()) {
+ currentMatch = FINAL_MATCH;
+ } else {
+ currentMatch = expectedMatches.remove();
+ }
+ }
+ return match;
+ }
+ }
+
+ private static class OperatorMatch implements OperatorValidator.Match {
+
+ final String operator;
+
+ final String line;
+
+ OperatorMatch(String operator, String line) {
+ this.operator = operator;
+ this.line = line;
+ }
+
+ public boolean match(String line) {
+ if (line.contains(operator)) {
+ assertEquals(this.line, line);
+ return true;
+ }
+ return false;
+ }
+ }
+
+ private final OperatorValidator.MatchSequence matchSequence = new MatchSequence();
+
+ public OperatorValidator addOperatorMatch(String operator, String expectedLine) {
+ matchSequence.addMatch(new OperatorMatch(operator, expectedLine));
+ return this;
+ }
+
+ public void check(String line) {
+ matchSequence.match(line);
+ }
+
+ public void end() {
+ assertTrue("Expected operators remain", matchSequence.isExhausted());
+ }
+
+} \ No newline at end of file
diff --git a/test/java/org/apache/fop/svg/PDFTextPainterTestCase.java b/test/java/org/apache/fop/svg/PDFTextPainterTestCase.java
new file mode 100644
index 000000000..52f18bb5e
--- /dev/null
+++ b/test/java/org/apache/fop/svg/PDFTextPainterTestCase.java
@@ -0,0 +1,149 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg;
+
+import java.awt.Graphics2D;
+import java.io.StringWriter;
+
+import org.junit.Test;
+
+import org.apache.batik.gvt.TextPainter;
+
+import org.apache.xmlgraphics.java2d.GraphicContext;
+
+import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.pdf.PDFDocument;
+import org.apache.fop.svg.font.FOPFontFamilyResolverImpl;
+
+public class PDFTextPainterTestCase extends NativeTextPainterTest {
+
+ private static class OperatorCheckingPDFGraphics2D extends PDFGraphics2D {
+
+ OperatorCheckingPDFGraphics2D(FontInfo fontInfo, final OperatorValidator validator) {
+ super(false, fontInfo, new PDFDocument("test"), null, null, null, 0);
+ this.currentStream = new StringWriter() {
+
+ @Override
+ public void write(String str) {
+ validator.check(str);
+ }
+
+ };
+ }
+ }
+
+ @Override
+ protected TextPainter createTextPainter(FontInfo fontInfo) {
+ return new PDFTextPainter(fontInfo);
+ }
+
+ @Override
+ protected Graphics2D createGraphics2D(FontInfo fontInfo, OperatorValidator validator) {
+ PDFGraphics2D g2d = new OperatorCheckingPDFGraphics2D(fontInfo, validator);
+ g2d.setGraphicContext(new GraphicContext());
+ return g2d;
+ }
+
+ @Test
+ public void testRotatedGlyph() throws Exception {
+ runTest("rotated-glyph.svg", new OperatorValidator()
+ .addOperatorMatch("Tm", "1 0 0 -1 40 110 Tm ")
+ .addOperatorMatch("TJ", "[(A)] TJ\n")
+ .addOperatorMatch("Tm", "0.70710677 0.7071068 0.7071068 -0.70710677 106.69999695 110 Tm ")
+ .addOperatorMatch("TJ", "[(B)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 173.3999939 110 Tm ")
+ .addOperatorMatch("TJ", "[(C)] TJ\n"));
+ }
+
+ @Test
+ public void testDxDy() throws Exception {
+ runTest("dx-dy.svg", new OperatorValidator()
+ .addOperatorMatch("Tm", "1 0 0 -1 55 35 Tm ")
+ .addOperatorMatch("TJ", "[(ABCDE)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 55 75 Tm ")
+ .addOperatorMatch("TJ", "[(A)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 69 85 Tm ")
+ .addOperatorMatch("TJ", "[(B)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 109 80 Tm ")
+ .addOperatorMatch("TJ", "[(C)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 91 65 Tm ")
+ .addOperatorMatch("TJ", "[(D)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 127 75 Tm ")
+ .addOperatorMatch("TJ", "[(E)] TJ\n"));
+ }
+
+ @Test
+ public void testSpacing() throws Exception {
+ runTest("spacing.svg", new OperatorValidator()
+ .addOperatorMatch("Tm", "1 0 0 -1 0 0 Tm ")
+ .addOperatorMatch("TJ", "[(V) 80 (A) 70 (V)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 0 0 Tm ")
+ .addOperatorMatch("TJ", "[(V) 80 (A) 70 (V)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 0 0 Tm ")
+ .addOperatorMatch("TJ", "[(V) -20 (A) -30 (V)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 0 0 Tm ")
+ .addOperatorMatch("TJ", "[(ab) -111 ( ) -389 (cd)] TJ\n"));
+ }
+
+ @Test
+ public void testGlyphOrientation() throws Exception {
+ runTest("glyph-orientation.svg", new OperatorValidator()
+ .addOperatorMatch("Tm", "0 1 1 0 738.5 0 Tm ")
+ .addOperatorMatch("TJ", "[(A)] TJ\n")
+ .addOperatorMatch("Tm", "0 1 1 0 738.5 667 Tm ")
+ .addOperatorMatch("TJ", "[(B)] TJ\n")
+ .addOperatorMatch("Tm", "0 1 1 0 738.5 1334 Tm ")
+ .addOperatorMatch("TJ", "[(C)] TJ\n")
+ .addOperatorMatch("Tm", "0 1 1 0 738.5 2056 Tm ")
+ .addOperatorMatch("TJ", "[(D)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 2149 718 Tm ")
+ .addOperatorMatch("TJ", "[(E)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 2165.5 1643 Tm ")
+ .addOperatorMatch("TJ", "[(F)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 2124 2568 Tm ")
+ .addOperatorMatch("TJ", "[(G)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 2138.5 3493 Tm ")
+ .addOperatorMatch("TJ", "[(H)] TJ\n")
+ .addOperatorMatch("Tm", "0 -1 -1 0 718 5000 Tm ")
+ .addOperatorMatch("TJ", "[(I)] TJ\n")
+ .addOperatorMatch("Tm", "0 -1 -1 0 1643 5000 Tm ")
+ .addOperatorMatch("TJ", "[(J)] TJ\n")
+ .addOperatorMatch("Tm", "0 -1 -1 0 2568 5000 Tm ")
+ .addOperatorMatch("TJ", "[(K)] TJ\n")
+ .addOperatorMatch("Tm", "0 -1 -1 0 3493 5000 Tm ")
+ .addOperatorMatch("TJ", "[(L)] TJ\n"));
+ }
+
+ @Test
+ public void testBaselineShift() throws Exception {
+ runTest("baseline-shift.svg", new OperatorValidator()
+ .addOperatorMatch("Tm", "1 0 0 -1 0 0 Tm ")
+ .addOperatorMatch("TJ", "[(AB)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 1334 -462.5 Tm ")
+ .addOperatorMatch("TJ", "[(CD)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 2778 0 Tm ")
+ .addOperatorMatch("TJ", "[(EF)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 4056 462.5 Tm ")
+ .addOperatorMatch("TJ", "[(GH)] TJ\n")
+ .addOperatorMatch("Tm", "1 0 0 -1 5556 0 Tm ")
+ .addOperatorMatch("TJ", "[(IJ)] TJ\n"));
+ }
+
+}
diff --git a/test/java/org/apache/fop/svg/PSTextPainterTestCase.java b/test/java/org/apache/fop/svg/PSTextPainterTestCase.java
new file mode 100644
index 000000000..2d5de7455
--- /dev/null
+++ b/test/java/org/apache/fop/svg/PSTextPainterTestCase.java
@@ -0,0 +1,77 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg;
+
+import java.awt.Graphics2D;
+import java.io.IOException;
+
+import org.junit.Test;
+
+import org.apache.commons.io.output.NullOutputStream;
+
+import org.apache.batik.gvt.TextPainter;
+
+import org.apache.xmlgraphics.java2d.GraphicContext;
+import org.apache.xmlgraphics.java2d.ps.PSGraphics2D;
+import org.apache.xmlgraphics.ps.PSGenerator;
+
+import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.render.ps.PSTextPainter;
+
+public class PSTextPainterTestCase extends NativeTextPainterTest {
+
+ private static class OperatorCheckingPSGraphics2D extends PSGraphics2D {
+
+ OperatorCheckingPSGraphics2D(FontInfo fontInfo, final OperatorValidator validator) {
+ super(false, new PSGenerator(new NullOutputStream()) {
+
+ @Override
+ public void writeln(String cmd) throws IOException {
+ validator.check(cmd);
+ }
+
+ });
+ }
+ }
+
+ @Override
+ protected TextPainter createTextPainter(FontInfo fontInfo) {
+ return new PSTextPainter(fontInfo);
+ }
+
+ @Override
+ protected Graphics2D createGraphics2D(FontInfo fontInfo, OperatorValidator validator) {
+ PSGraphics2D g2d = new OperatorCheckingPSGraphics2D(fontInfo, validator);
+ g2d.setGraphicContext(new GraphicContext());
+ return g2d;
+ }
+
+ @Test
+ public void testRotatedGlyph() throws Exception {
+ runTest("rotated-glyph.svg", new OperatorValidator()
+ .addOperatorMatch("Tm", "1 0 0 -1 40 110 Tm")
+ .addOperatorMatch("xshow", "(A)\n[0] xshow")
+ .addOperatorMatch("Tm", "0.70711 0.70711 0.70711 -0.70711 106.7 110 Tm")
+ .addOperatorMatch("xshow", "(B)\n[0] xshow")
+ .addOperatorMatch("Tm", "1 0 0 -1 173.39999 110 Tm")
+ .addOperatorMatch("xshow", "(C)\n[0] xshow"));
+ }
+
+}
diff --git a/test/java/org/apache/fop/svg/baseline-shift.svg b/test/java/org/apache/fop/svg/baseline-shift.svg
new file mode 100644
index 000000000..0f375b9af
--- /dev/null
+++ b/test/java/org/apache/fop/svg/baseline-shift.svg
@@ -0,0 +1,9 @@
+<?xml version="1.0"?>
+<svg width="150" height="60" xmlns="http://www.w3.org/2000/svg">
+<rect x="0" y="0" width="100%" height="100%" fill="none" stroke="black" stroke-width="2"/>
+<g transform="translate(10, 40) scale(20) scale(0.001)">
+<text font-family="sans-serif" font-size="1000">
+AB<tspan baseline-shift="super">CD</tspan>EF<tspan baseline-shift="sub">GH</tspan>IJ
+</text>
+</g>
+</svg>
diff --git a/test/java/org/apache/fop/svg/dx-dy.svg b/test/java/org/apache/fop/svg/dx-dy.svg
new file mode 100644
index 000000000..cfdc2de01
--- /dev/null
+++ b/test/java/org/apache/fop/svg/dx-dy.svg
@@ -0,0 +1,8 @@
+<?xml version="1.0"?>
+<svg width="200" height="100" xmlns="http://www.w3.org/2000/svg">
+<rect x="0" y="0" width="100%" height="100%" stroke="black" stroke-width="2" fill="none"/>
+<g font-family="monospace" font-size="30">
+ <text x="55" y="35">ABCDE</text>
+ <text x="55" y="75" dx="0 -4 22 -36 18" dy="0 10 -5 -15 10">ABCDE</text>
+</g>
+</svg>
diff --git a/test/java/org/apache/fop/svg/font/BasicGlyphVectorTestCase.java b/test/java/org/apache/fop/svg/font/BasicGlyphVectorTestCase.java
new file mode 100644
index 000000000..c6f062a84
--- /dev/null
+++ b/test/java/org/apache/fop/svg/font/BasicGlyphVectorTestCase.java
@@ -0,0 +1,193 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.awt.Rectangle;
+import java.awt.font.GlyphMetrics;
+import java.awt.geom.AffineTransform;
+import java.awt.geom.Point2D;
+import java.awt.geom.Rectangle2D;
+import java.text.CharacterIterator;
+import java.text.StringCharacterIterator;
+
+import org.junit.Before;
+import org.junit.Test;
+
+import static org.junit.Assert.assertArrayEquals;
+import static org.junit.Assert.assertEquals;
+import static org.mockito.Matchers.anyInt;
+import static org.mockito.Matchers.eq;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+import org.apache.batik.gvt.font.GVTGlyphMetrics;
+import org.apache.batik.gvt.font.GVTLineMetrics;
+
+import org.apache.fop.fonts.Font;
+import org.apache.fop.fonts.FontMetrics;
+
+/**
+ * Tests all the methods of {@link FOPGVTGlyphVector} with a mocked font.
+ */
+public class BasicGlyphVectorTestCase extends FOPGVTGlyphVectorTest {
+
+ private final int fontSize = 10000;
+
+ @Before
+ public void createGlyphVector() {
+ FontMetrics metrics = mockFontMetrics();
+ Font font = mockFont(metrics);
+ FOPGVTFont gvtFont = mockGVTFont(font);
+ CharacterIterator it = new StringCharacterIterator("ABC");
+ glyphVector = new FOPGVTGlyphVector(gvtFont, it, null);
+ glyphVector.performDefaultLayout();
+ }
+
+ private FontMetrics mockFontMetrics() {
+ FontMetrics metrics = mock(FontMetrics.class);
+ when(metrics.getAscender(eq(fontSize))).thenReturn(8000000);
+ when(metrics.getDescender(eq(fontSize))).thenReturn(-4000000);
+ when(metrics.getWidth(eq(1), eq(fontSize))).thenReturn(10000000);
+ when(metrics.getBoundingBox(eq(1), eq(fontSize))).thenReturn(
+ new Rectangle(-1000000, -2000000, 3000000, 4000000));
+ when(metrics.getWidth(eq(2), eq(fontSize))).thenReturn(11000000);
+ when(metrics.getBoundingBox(eq(2), eq(fontSize))).thenReturn(
+ new Rectangle(-5000000, -6000000, 7000000, 9000000));
+ when(metrics.getWidth(eq(3), eq(fontSize))).thenReturn(12000000);
+ when(metrics.getBoundingBox(eq(3), eq(fontSize))).thenReturn(
+ new Rectangle(-9000000, -10000000, 11000000, 14000000));
+ return metrics;
+ }
+
+ private Font mockFont(FontMetrics metrics) {
+ Font font = mock(Font.class);
+ when(font.getFontMetrics()).thenReturn(metrics);
+ when(font.getFontSize()).thenReturn(fontSize);
+ when(font.mapChar(eq('A'))).thenReturn((char) 1);
+ when(font.mapChar(eq('B'))).thenReturn((char) 2);
+ when(font.mapChar(eq('C'))).thenReturn((char) 3);
+ return font;
+ }
+
+ private FOPGVTFont mockGVTFont(Font font) {
+ FOPGVTFont gvtFont = mock(FOPGVTFont.class);
+ when(gvtFont.getFont()).thenReturn(font);
+ when(gvtFont.getLineMetrics(anyInt())).thenReturn(
+ new GVTLineMetrics(8, 0, null, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0));
+ return gvtFont;
+ }
+
+ @Test
+ public void getGlyphCodeReturnsGlyphIndex() {
+ assertEquals(1, glyphVector.getGlyphCode(0));
+ assertEquals(2, glyphVector.getGlyphCode(1));
+ assertEquals(3, glyphVector.getGlyphCode(2));
+ }
+
+ @Test
+ public void testGetGlyphCodes() {
+ assertArrayEquals(new int[] {1, 2, 3}, glyphVector.getGlyphCodes(0, 3, null));
+ assertArrayEquals(new int[] {2, 3}, glyphVector.getGlyphCodes(1, 2, null));
+ }
+
+ @Test
+ public void testGetGlyphMetrics() {
+ assertGlyphMetricsEqual(new GVTGlyphMetrics(10, 12, new Rectangle(-1, -2, 3, 4), GlyphMetrics.STANDARD),
+ glyphVector.getGlyphMetrics(0));
+ assertGlyphMetricsEqual(new GVTGlyphMetrics(11, 12, new Rectangle(-5, -3, 7, 9), GlyphMetrics.STANDARD),
+ glyphVector.getGlyphMetrics(1));
+ assertGlyphMetricsEqual(new GVTGlyphMetrics(12, 12, new Rectangle(-9, -4, 11, 14), GlyphMetrics.STANDARD),
+ glyphVector.getGlyphMetrics(2));
+ }
+
+ private void assertGlyphMetricsEqual(GVTGlyphMetrics expected, GVTGlyphMetrics actual) {
+ assertEquals(expected.getHorizontalAdvance(), actual.getHorizontalAdvance(), 0);
+ assertEquals(expected.getVerticalAdvance(), actual.getVerticalAdvance(), 0);
+ assertEquals(expected.getBounds2D(), actual.getBounds2D());
+ assertEquals(expected.getLSB(), actual.getLSB(), 0);
+ assertEquals(expected.getRSB(), actual.getRSB(), 0);
+ assertEquals(expected.getType(), actual.getType());
+ assertEquals(expected.isCombining(), actual.isCombining());
+ assertEquals(expected.isComponent(), actual.isComponent());
+ assertEquals(expected.isLigature(), actual.isLigature());
+ assertEquals(expected.isStandard(), actual.isStandard());
+ assertEquals(expected.isWhitespace(), actual.isWhitespace());
+ }
+
+ @Test
+ public void testGetGlyphPosition() {
+ assertEquals(new Point2D.Float(0, 0), glyphVector.getGlyphPosition(0));
+ assertEquals(new Point2D.Float(10, 0), glyphVector.getGlyphPosition(1));
+ assertEquals(new Point2D.Float(21, 0), glyphVector.getGlyphPosition(2));
+ assertEquals(new Point2D.Float(33, 0), glyphVector.getGlyphPosition(3));
+ }
+
+ @Test
+ public void testGetGlyphPositions() {
+ float[] expectedPositions = new float[] {0, 0, 10, 0, 21, 0, 33, 0};
+ assertArrayEquals(expectedPositions, glyphVector.getGlyphPositions(0, 4, null), 0);
+ assertArrayEquals(expectedPositions, glyphVector.getGlyphPositions(0, 4, new float[8]), 0);
+ }
+
+ @Test
+ public void testGetGlyphOutline() {
+ assertEquals(new Rectangle(-1, -2, 3, 4), glyphVector.getGlyphOutline(0).getBounds());
+ assertEquals(new Rectangle(5, -3, 7, 9), glyphVector.getGlyphOutline(1).getBounds());
+ assertEquals(new Rectangle(12, -4, 11, 14), glyphVector.getGlyphOutline(2).getBounds());
+ }
+
+ @Test
+ public void testGetOutline() {
+ assertEquals(new Rectangle(-1, -4, 24, 14), glyphVector.getOutline().getBounds());
+ }
+
+ @Test
+ public void testGetLogicalBounds() {
+ assertEquals(new Rectangle(0, -8, 33, 12), glyphVector.getLogicalBounds());
+ }
+
+ @Test
+ public void testGetLogicalBoundsRotated() {
+ for (int i = 0; i < 3; i++) {
+ glyphVector.setGlyphTransform(i, new AffineTransform(0.7, 0.7, -0.7, 0.7, 0, 0));
+ }
+ assertEquals(new Rectangle2D.Float(-2.8f, -5.6f, 37.8f, 16.8f), glyphVector.getLogicalBounds());
+ }
+
+ @Test
+ public void testGetBounds() {
+ assertEquals(new Rectangle(-1, -4, 24, 14), glyphVector.getBounds2D(null));
+ }
+
+ @Test
+ public void testGetGlyphVisualBounds() {
+ assertEquals(new Rectangle(-1, -2, 3, 4), glyphVector.getGlyphVisualBounds(0).getBounds());
+ assertEquals(new Rectangle(5, -3, 7, 9), glyphVector.getGlyphVisualBounds(1).getBounds());
+ assertEquals(new Rectangle(12, -4, 11, 14), glyphVector.getGlyphVisualBounds(2).getBounds());
+ }
+
+ @Test
+ public void testGetGlyphLogicalBounds() {
+ assertEquals(new Rectangle(0, -8, 10, 12), glyphVector.getGlyphLogicalBounds(0).getBounds());
+ assertEquals(new Rectangle(10, -8, 11, 12), glyphVector.getGlyphLogicalBounds(1).getBounds());
+ assertEquals(new Rectangle(21, -8, 12, 12), glyphVector.getGlyphLogicalBounds(2).getBounds());
+ }
+
+}
diff --git a/test/java/org/apache/fop/svg/font/FOPFontFamilyResolverTestCase.java b/test/java/org/apache/fop/svg/font/FOPFontFamilyResolverTestCase.java
new file mode 100644
index 000000000..2f5668cbc
--- /dev/null
+++ b/test/java/org/apache/fop/svg/font/FOPFontFamilyResolverTestCase.java
@@ -0,0 +1,127 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.awt.FontFormatException;
+import java.awt.GraphicsEnvironment;
+import java.awt.font.FontRenderContext;
+import java.awt.font.LineMetrics;
+import java.awt.image.BufferedImage;
+import java.io.File;
+import java.io.IOException;
+import java.util.Collections;
+
+import org.junit.Before;
+import org.junit.BeforeClass;
+import org.junit.Ignore;
+import org.junit.Test;
+
+import org.apache.batik.gvt.font.GVTFontFamily;
+import org.apache.batik.gvt.font.GVTLineMetrics;
+
+import org.apache.fop.fonts.FontInfo;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertFalse;
+import static org.junit.Assert.assertNotNull;
+import static org.junit.Assert.assertNull;
+import static org.junit.Assert.assertTrue;
+
+public class FOPFontFamilyResolverTestCase {
+
+ private static FontInfo fontInfo;
+
+ private FOPFontFamilyResolver resolver;
+
+ @BeforeClass
+ public static void setUpFontInfo() {
+ fontInfo = new FontInfoBuilder()
+ .useDejaVuLGCSerif()
+ .useDroidSansMono()
+ .build();
+ }
+
+ @Before
+ public void createFontFamilyResolver() {
+ resolver = new FOPFontFamilyResolverImpl(fontInfo);
+ }
+
+ @Test
+ public void testResolve() {
+ assertNull(resolver.resolve("Unavailable"));
+ assertNotNull(resolver.resolve(FontInfoBuilder.DEJAVU_LGC_SERIF));
+ }
+
+ @Test
+ public void testGetFamilyThatCanDisplay() {
+ GVTFontFamily family = resolver.getFamilyThatCanDisplay('\u0180');
+ assertEquals(FontInfoBuilder.DEJAVU_LGC_SERIF, family.getFamilyName());
+ family = resolver.getFamilyThatCanDisplay('\u02F3');
+ assertEquals(FontInfoBuilder.DROID_SANS_MONO, family.getFamilyName());
+ family = resolver.getFamilyThatCanDisplay('\u02DF');
+ assertNull(family);
+ }
+
+ @Test
+ public void testDeriveFont() {
+ FOPGVTFontFamily family = (FOPGVTFontFamily) resolver.resolve(FontInfoBuilder.DEJAVU_LGC_SERIF);
+ FOPGVTFont font = family.deriveFont(10, Collections.emptyMap());
+ assertEquals(10, font.getSize(), 0);
+ assertTrue(font.canDisplay('\u01F6'));
+ assertFalse(font.canDisplay('\u01F7'));
+ }
+
+ @Test
+ @Ignore("FOP metrics don't match AWT, but not sure who is right and who is wrong")
+ public void testLineMetrics() throws FontFormatException, IOException {
+ FOPGVTFontFamily family = (FOPGVTFontFamily) resolver.resolve(FontInfoBuilder.DEJAVU_LGC_SERIF);
+ FOPGVTFont font = family.deriveFont(10, Collections.emptyMap());
+ GVTLineMetrics fopMetrics = font.getLineMetrics("", null);
+ LineMetrics awtMetrics = getAWTLineMetrics();
+ printDifference("Ascent", awtMetrics.getAscent(), fopMetrics.getAscent());
+ printDifference("Descent", awtMetrics.getDescent(), fopMetrics.getDescent());
+ printDifference("Height", awtMetrics.getHeight(), fopMetrics.getHeight());
+ printDifference("Leading", awtMetrics.getLeading(), fopMetrics.getLeading());
+ printDifference("StrikethroughOffset", awtMetrics.getStrikethroughOffset(),
+ fopMetrics.getStrikethroughOffset());
+ printDifference("StrikethroughThickness", awtMetrics.getStrikethroughThickness(),
+ fopMetrics.getStrikethroughThickness());
+ printDifference("UnderlineOffset", awtMetrics.getUnderlineOffset(),
+ fopMetrics.getUnderlineOffset());
+ printDifference("UnderlineThickness", awtMetrics.getUnderlineThickness(),
+ fopMetrics.getUnderlineThickness());
+ }
+
+ private LineMetrics getAWTLineMetrics() throws FontFormatException, IOException {
+ File fontFile = new File("test/resources/fonts/ttf/DejaVuLGCSerif.ttf");
+ java.awt.Font awtFont = java.awt.Font.createFont(java.awt.Font.TRUETYPE_FONT, fontFile).deriveFont(10f);
+ GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();
+ BufferedImage dummyImage = new BufferedImage(1000, 1000, BufferedImage.TYPE_INT_RGB);
+ FontRenderContext frc = ge.createGraphics(dummyImage).getFontRenderContext();
+ LineMetrics awtMetrics = awtFont.getLineMetrics("ABC", frc);
+ return awtMetrics;
+ }
+
+ private void printDifference(String value, float awt, float fop) {
+ System.out.println(String.format("%22s AWT: %10f FOP: %10f Difference: %.2f%%", value, awt, fop,
+ (fop - awt) / awt * 100));
+ }
+
+}
diff --git a/test/java/org/apache/fop/svg/font/FOPGVTFontTestCase.java b/test/java/org/apache/fop/svg/font/FOPGVTFontTestCase.java
new file mode 100644
index 000000000..c7af068ce
--- /dev/null
+++ b/test/java/org/apache/fop/svg/font/FOPGVTFontTestCase.java
@@ -0,0 +1,72 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.text.StringCharacterIterator;
+
+import org.junit.Before;
+import org.junit.Test;
+
+import static org.junit.Assert.assertEquals;
+import static org.mockito.Matchers.eq;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+import org.apache.fop.fonts.Font;
+import org.apache.fop.svg.font.FOPGVTFont;
+
+public class FOPGVTFontTestCase {
+
+ private FOPGVTFont font;
+
+ @Before
+ public void createFont() {
+ Font f = mock(Font.class);
+ when(f.hasChar(eq((char) 0))).thenReturn(false);
+ when(f.hasChar(eq((char) 1))).thenReturn(true);
+ font = new FOPGVTFont(f, null);
+ }
+
+ @Test
+ public void testCanDisplayUpTo() {
+ char[] text = new char[] {1, 1, 1};
+ testCanDisplayUpToVariants(text, -1, 0, 3);
+ testCanDisplayUpToVariants(text, -1, 1, 3);
+ text = new char[] {1, 1, 0, 1};
+ testCanDisplayUpToVariants(text, 2, 0, 4);
+ testCanDisplayUpToVariants(text, 2, 1, 4);
+ testCanDisplayUpToVariants(text, 2, 2, 4);
+ testCanDisplayUpToVariants(text, -1, 3, 4);
+ testCanDisplayUpToVariants(text, -1, 1, 2);
+ }
+
+ @Test
+ public void testCanDisplayUpToString() {
+ assertEquals(-1, font.canDisplayUpTo(new String(new char[] {1, 1, 1})));
+ assertEquals(0, font.canDisplayUpTo(new String(new char[] {0, 1, 1})));
+ assertEquals(1, font.canDisplayUpTo(new String(new char[] {1, 0, 1})));
+ assertEquals(2, font.canDisplayUpTo(new String(new char[] {1, 1, 0})));
+ }
+
+ private void testCanDisplayUpToVariants(char[] text, int expected, int start, int limit) {
+ assertEquals(expected, font.canDisplayUpTo(text, start, limit));
+ assertEquals(expected, font.canDisplayUpTo(new StringCharacterIterator(new String(text)), start, limit));
+ }
+}
diff --git a/test/java/org/apache/fop/svg/font/FOPGVTGlyphVectorTest.java b/test/java/org/apache/fop/svg/font/FOPGVTGlyphVectorTest.java
new file mode 100644
index 000000000..0995ab4df
--- /dev/null
+++ b/test/java/org/apache/fop/svg/font/FOPGVTGlyphVectorTest.java
@@ -0,0 +1,27 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+
+public abstract class FOPGVTGlyphVectorTest {
+
+ protected FOPGVTGlyphVector glyphVector;
+
+}
diff --git a/test/java/org/apache/fop/svg/font/FontInfoBuilder.java b/test/java/org/apache/fop/svg/font/FontInfoBuilder.java
new file mode 100644
index 000000000..f7a5825bd
--- /dev/null
+++ b/test/java/org/apache/fop/svg/font/FontInfoBuilder.java
@@ -0,0 +1,102 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.io.File;
+import java.io.IOException;
+import java.net.URI;
+import java.net.URISyntaxException;
+
+import org.apache.fop.apps.io.InternalResourceResolver;
+import org.apache.fop.apps.io.ResourceResolverFactory;
+import org.apache.fop.fonts.EmbeddingMode;
+import org.apache.fop.fonts.EncodingMode;
+import org.apache.fop.fonts.Font;
+import org.apache.fop.fonts.FontInfo;
+import org.apache.fop.fonts.FontMetrics;
+import org.apache.fop.fonts.truetype.TTFFontLoader;
+
+class FontInfoBuilder {
+
+ public static final String DEJAVU_LGC_SERIF = "DejaVu LGC Serif";
+
+ public static final String DROID_SANS_MONO = "Droid Sans Mono";
+
+ private static final boolean USE_ADVANCED_BY_DEFAULT = true;
+
+ private FontInfo fontInfo;
+
+ private int fontKey;
+
+ public FontInfoBuilder() {
+ reset();
+ }
+
+ private void reset() {
+ fontInfo = new FontInfo();
+ fontKey = 1;
+ }
+
+ public FontInfoBuilder useDejaVuLGCSerif() {
+ return useDejaVuLGCSerif(USE_ADVANCED_BY_DEFAULT);
+ }
+
+ public FontInfoBuilder useDejaVuLGCSerif(boolean useAdvanced) {
+ try {
+ return useFont(DEJAVU_LGC_SERIF, "DejaVuLGCSerif.ttf", useAdvanced);
+ } catch (Exception e) {
+ throw new RuntimeException(e);
+ }
+ }
+
+ public FontInfoBuilder useDroidSansMono() {
+ return useDroidSansMono(USE_ADVANCED_BY_DEFAULT);
+ }
+
+ public FontInfoBuilder useDroidSansMono(boolean useAdvanced) {
+ try {
+ return useFont(DROID_SANS_MONO, "DroidSansMono.ttf", useAdvanced);
+ } catch (Exception e) {
+ throw new RuntimeException(e);
+ }
+ }
+
+ private FontInfoBuilder useFont(String fontName, String filename, boolean useAdvanced)
+ throws IOException, URISyntaxException {
+ URI baseURI = new File("test/resources/fonts/ttf").toURI();
+ InternalResourceResolver resolver = ResourceResolverFactory.createDefaultInternalResourceResolver(baseURI);
+ TTFFontLoader fontLoader = new TTFFontLoader(new URI(filename), null, true,
+ EmbeddingMode.AUTO, EncodingMode.AUTO, true, useAdvanced, resolver);
+ FontMetrics font = fontLoader.getFont();
+ registerFont(font, "F" + fontKey++, fontName);
+ return this;
+ }
+
+ private void registerFont(FontMetrics font, String key, String familyName) {
+ fontInfo.addMetrics(key, font);
+ fontInfo.addFontProperties(key, familyName, Font.STYLE_NORMAL, Font.WEIGHT_NORMAL);
+ }
+
+ public FontInfo build() {
+ FontInfo fontInfo = this.fontInfo;
+ reset();
+ return fontInfo;
+ }
+}
diff --git a/test/java/org/apache/fop/svg/font/GlyphLayoutTestCase.java b/test/java/org/apache/fop/svg/font/GlyphLayoutTestCase.java
new file mode 100644
index 000000000..34cba604a
--- /dev/null
+++ b/test/java/org/apache/fop/svg/font/GlyphLayoutTestCase.java
@@ -0,0 +1,91 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/* $Id$ */
+
+package org.apache.fop.svg.font;
+
+import java.util.Collections;
+
+import org.junit.Test;
+
+import org.apache.fop.fonts.FontInfo;
+
+import static org.junit.Assert.assertEquals;
+
+/**
+ * Specifically tests glyph positioning from a real font.
+ */
+public class GlyphLayoutTestCase extends FOPGVTGlyphVectorTest {
+
+ /**
+ * Glyph positioning using the legacy kern table.
+ */
+ @Test
+ public void testBasicGlyphPositioning() throws Exception {
+ testGlyphLayout(false);
+ }
+
+ /**
+ * Glyph positioning using GPOS sub-tables.
+ */
+ @Test
+ public void testAdvancedGlyphPositioning() throws Exception {
+ testGlyphLayout(true);
+ }
+
+ private void testGlyphLayout(boolean useAdvanced) {
+ FOPGVTFont font = loadFont(useAdvanced);
+ glyphVector = (FOPGVTGlyphVector) font.createGlyphVector(null, "L\u201DP,V.F,A\u2019LT.");
+ glyphVector.performDefaultLayout();
+ // Values in font units (unitsPerEm = 2048), glyph width - kern
+ int[] widths = {
+ /* L */ 1360 - 491,
+ /* " */ 1047,
+ /* P */ 1378 - 415,
+ /* , */ 651,
+ /* V */ 1479 - 358,
+ /* . */ 651,
+ /* F */ 1421 - 319,
+ /* , */ 651,
+ /* A */ 1479 - 301,
+ /* ' */ 651,
+ /* L */ 1360 - 167,
+ /* T */ 1366 - 301,
+ /* . */ 651};
+ checkGlyphPositions(13, widths);
+ }
+
+ private FOPGVTFont loadFont(boolean useAdvanced) {
+ FontInfo fontInfo = new FontInfoBuilder().useDejaVuLGCSerif(useAdvanced).build();
+ FOPFontFamilyResolver resolver = new FOPFontFamilyResolverImpl(fontInfo);
+ FOPGVTFontFamily family = resolver.resolve(FontInfoBuilder.DEJAVU_LGC_SERIF);
+ return family.deriveFont(1000, Collections.emptyMap());
+ }
+
+ private void checkGlyphPositions(int expectedGlyphCount, int[] widths) {
+ assertEquals(expectedGlyphCount, glyphVector.getNumGlyphs());
+ float[] positions = new float[2 * (widths.length + 1)];
+ for (int i = 0, n = 2; i < widths.length; i++, n += 2) {
+ positions[n] = positions[n - 2] + widths[i] / 2.048f;
+ }
+ for (int i = 0; i <= widths.length; i++) {
+ assertEquals(positions[2 * i], glyphVector.getGlyphPosition(i).getX(), 3);
+ }
+ }
+
+}
diff --git a/test/java/org/apache/fop/svg/glyph-orientation.svg b/test/java/org/apache/fop/svg/glyph-orientation.svg
new file mode 100644
index 000000000..4900a3b02
--- /dev/null
+++ b/test/java/org/apache/fop/svg/glyph-orientation.svg
@@ -0,0 +1,10 @@
+<?xml version="1.0"?>
+<svg width="100" height="140" xmlns="http://www.w3.org/2000/svg">
+<rect x="0" y="0" width="100%" height="100%" fill="none" stroke="black" stroke-width="2"/>
+<g transform="translate(10) scale(20) scale(0.001) translate(0, 1000)"
+ font-family="sans-serif" font-size="1000">
+ <text x="1000" writing-mode="tb">ABCD</text>
+ <text x="2500" writing-mode="tb" glyph-orientation-vertical="0">EFGH</text>
+ <text x="0" y="5000" glyph-orientation-horizontal="270">IJKL</text>
+</g>
+</svg>
diff --git a/test/java/org/apache/fop/svg/rotated-glyph.svg b/test/java/org/apache/fop/svg/rotated-glyph.svg
new file mode 100644
index 000000000..8b942905e
--- /dev/null
+++ b/test/java/org/apache/fop/svg/rotated-glyph.svg
@@ -0,0 +1,5 @@
+<?xml version="1.0"?>
+<svg width="300" height="150" xmlns="http://www.w3.org/2000/svg">
+<rect x="0" y="0" width="100%" height="100%" fill="none" stroke="black" stroke-width="2"/>
+<text font-family="Helvetica" font-size="100" x="40" y="110" rotate="0 45 0">ABC</text>
+</svg>
diff --git a/test/java/org/apache/fop/svg/spacing.svg b/test/java/org/apache/fop/svg/spacing.svg
new file mode 100644
index 000000000..943ab1d04
--- /dev/null
+++ b/test/java/org/apache/fop/svg/spacing.svg
@@ -0,0 +1,21 @@
+<?xml version="1.0"?>
+<svg width="150" height="200" xmlns="http://www.w3.org/2000/svg">
+<rect x="0" y="0" width="100%" height="100%" stroke="black" stroke-width="2" fill="none"/>
+<g transform="translate(10) scale(40) scale(0.001) translate(0, 1000)"
+ font-family="sans-serif" font-size="1000">
+ <g transform="">
+ <text>VAV</text>
+ <line x1="667" y1="-818" x2="667" y2="100" stroke-width="10" stroke="blue"/>
+ </g>
+ <g transform="translate(0, 1000)">
+ <text kerning="0">VAV</text>
+ <line x1="667" y1="-818" x2="667" y2="100" stroke-width="10" stroke="blue"/>
+ </g>
+ <g transform="translate(0, 2000)">
+ <text letter-spacing="100">VAV</text>
+ </g>
+ <g transform="translate(0, 3000)">
+ <text word-spacing="500">ab cd</text>
+ </g>
+</g>
+</svg>