From 999ca17b4f4c879db6e40c58cca8763129bd4b07 Mon Sep 17 00:00:00 2001 From: Simon Steiner Date: Fri, 28 Jan 2022 10:27:59 +0000 Subject: FOP-3049: Align AFP SVG text in the middle git-svn-id: https://svn.apache.org/repos/asf/xmlgraphics/fop/trunk@1897561 13f79535-47bb-0310-9956-ffa450edef68 --- .../apache/fop/image/loader/batik/BatikUtil.java | 10 +++- .../org/apache/fop/svg/AbstractFOPTextPainter.java | 10 +++- .../apache/fop/render/afp/AFPTrueTypeTestCase.java | 69 +++++++++++++++++----- 3 files changed, 72 insertions(+), 17 deletions(-) (limited to 'fop-core') diff --git a/fop-core/src/main/java/org/apache/fop/image/loader/batik/BatikUtil.java b/fop-core/src/main/java/org/apache/fop/image/loader/batik/BatikUtil.java index cda57b596..7162df971 100644 --- a/fop-core/src/main/java/org/apache/fop/image/loader/batik/BatikUtil.java +++ b/fop-core/src/main/java/org/apache/fop/image/loader/batik/BatikUtil.java @@ -19,8 +19,10 @@ package org.apache.fop.image.loader.batik; +import org.w3c.dom.DOMImplementation; import org.w3c.dom.Document; +import org.apache.batik.anim.dom.SVGDOMImplementation; import org.apache.batik.dom.AbstractDocument; import org.apache.batik.dom.util.DOMUtilities; @@ -53,9 +55,13 @@ public final class BatikUtil { * @return the cloned SVG DOM */ public static Document cloneSVGDocument(Document doc) { - Document clonedDoc = DOMUtilities.deepCloneDocument(doc, doc.getImplementation()); + DOMImplementation impl = doc.getImplementation(); + if (!(impl instanceof SVGDOMImplementation)) { + impl = new SVGDOMImplementation(); + } + Document clonedDoc = DOMUtilities.deepCloneDocument(doc, impl); if (clonedDoc instanceof AbstractDocument) { - ((AbstractDocument)clonedDoc).setDocumentURI(((AbstractDocument)doc).getDocumentURI()); + clonedDoc.setDocumentURI(doc.getDocumentURI()); } return clonedDoc; } diff --git a/fop-core/src/main/java/org/apache/fop/svg/AbstractFOPTextPainter.java b/fop-core/src/main/java/org/apache/fop/svg/AbstractFOPTextPainter.java index df24dbb92..30843121d 100644 --- a/fop-core/src/main/java/org/apache/fop/svg/AbstractFOPTextPainter.java +++ b/fop-core/src/main/java/org/apache/fop/svg/AbstractFOPTextPainter.java @@ -36,6 +36,7 @@ import org.apache.batik.bridge.Mark; import org.apache.batik.bridge.StrokingTextPainter; import org.apache.batik.bridge.TextNode; import org.apache.batik.bridge.TextPainter; +import org.apache.batik.gvt.font.GVTGlyphVector; import org.apache.batik.gvt.text.GVTAttributedCharacterIterator; import org.apache.batik.gvt.text.TextPaintInfo; @@ -157,7 +158,14 @@ public abstract class AbstractFOPTextPainter implements TextPainter { if (font != null) { fontSize = (int) Math.round(afpg2d.convertToAbsoluteLength(font.getFontSize())); } - if (fontSize < 6000) { + if (anchor != null && TextNode.Anchor.ANCHOR_MIDDLE == anchor.getType()) { + GVTGlyphVector gv = run.getLayout().getGlyphVector(); + Point2D glyphPos = gv.getGlyphPosition(0); + double advanceChar = afpg2d.convertToAbsoluteLength(advance / txt.length()); + nativeTextHandler.drawString(g2d, txt, + (float) (currentLocation.getX() - advanceChar - glyphPos.getX()), + (float) (currentLocation.getY() + glyphPos.getY())); + } else if (fontSize < 6000) { nativeTextHandler.drawString(g2d, txt, (float) (x + tx), (float) y); } else { double scaleX = g2d.getTransform().getScaleX(); diff --git a/fop-core/src/test/java/org/apache/fop/render/afp/AFPTrueTypeTestCase.java b/fop-core/src/test/java/org/apache/fop/render/afp/AFPTrueTypeTestCase.java index 9a9e5bed7..f875d80d8 100644 --- a/fop-core/src/test/java/org/apache/fop/render/afp/AFPTrueTypeTestCase.java +++ b/fop-core/src/test/java/org/apache/fop/render/afp/AFPTrueTypeTestCase.java @@ -21,17 +21,16 @@ package org.apache.fop.render.afp; import java.awt.Color; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; +import java.io.DataInputStream; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.URI; -import java.net.URISyntaxException; import javax.xml.transform.Result; import javax.xml.transform.Source; import javax.xml.transform.Transformer; -import javax.xml.transform.TransformerException; import javax.xml.transform.TransformerFactory; import javax.xml.transform.sax.SAXResult; import javax.xml.transform.stream.StreamSource; @@ -39,8 +38,6 @@ import javax.xml.transform.stream.StreamSource; import org.junit.Assert; import org.junit.Test; -import org.xml.sax.SAXException; - import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; @@ -53,6 +50,8 @@ import org.apache.fop.afp.DataStream; import org.apache.fop.afp.Factory; import org.apache.fop.afp.fonts.FopCharacterSet; import org.apache.fop.afp.modca.PageObject; +import org.apache.fop.afp.parser.MODCAParser; +import org.apache.fop.afp.parser.UnparsedStructuredField; import org.apache.fop.apps.EnvironmentalProfileFactory; import org.apache.fop.apps.FOUserAgent; import org.apache.fop.apps.Fop; @@ -66,8 +65,6 @@ import org.apache.fop.fonts.FontTriplet; import org.apache.fop.fonts.MultiByteFont; import org.apache.fop.render.intermediate.IFException; - - public class AFPTrueTypeTestCase { private String font; private String fopxconf = "\n" @@ -90,7 +87,7 @@ public class AFPTrueTypeTestCase { + ""; @Test - public void testAFPTrueType() throws IOException, SAXException, TransformerException, URISyntaxException { + public void testAFPTrueType() throws Exception { String fo = "\n" + " \n" + " \n" @@ -189,7 +186,7 @@ public class AFPTrueTypeTestCase { } @Test - public void testSVGAFPTrueType() throws IOException, SAXException, TransformerException, URISyntaxException { + public void testSVGAFPTrueType() throws Exception { String fo = "\n" @@ -228,7 +225,55 @@ public class AFPTrueTypeTestCase { Assert.assertTrue(getAFP(fo).contains("DATA GRAPHICS")); } - private String getAFP(String fo) throws IOException, TransformerException, SAXException, URISyntaxException { + @Test + public void testSVGAnchorAFP() throws Exception { + String fo = "\n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + "\n" + + " \n" + + " 0000210122010000000100010004\n" + + " \n" + + "\n" + + " \n" + + " \n" + + " \n" + + " \n" + + ""; + ByteArrayOutputStream bos = getAFPBytes(fo); + MODCAParser parser = new MODCAParser(new ByteArrayInputStream(bos.toByteArray())); + UnparsedStructuredField structuredField; + while ((structuredField = parser.readNextStructuredField()) != null) { + if (structuredField.toString().contains("Data Graphics")) { + break; + } + } + DataInputStream bis = new DataInputStream(new ByteArrayInputStream(structuredField.getData())); + bis.skip(34); + //X Y coordinates: + Assert.assertEquals(bis.readShort(), 3); + Assert.assertEquals(bis.readShort(), 5); + } + + private String getAFP(String fo) throws Exception { + ByteArrayOutputStream bos = getAFPBytes(fo); + StringBuilder sb = new StringBuilder(); + InputStream bis = new ByteArrayInputStream(bos.toByteArray()); + new AFPParser(false).read(bis, sb); + return sb.toString(); + } + + private ByteArrayOutputStream getAFPBytes(String fo) throws Exception { FopFactoryBuilder confBuilder = new FopConfParser( new ByteArrayInputStream(fopxconf.getBytes()), EnvironmentalProfileFactory.createRestrictedIO(new URI("."), @@ -243,11 +288,7 @@ public class AFPTrueTypeTestCase { Result res = new SAXResult(fop.getDefaultHandler()); transformer.transform(src, res); bos.close(); - - StringBuilder sb = new StringBuilder(); - InputStream bis = new ByteArrayInputStream(bos.toByteArray()); - new AFPParser(false).read(bis, sb); - return sb.toString(); + return bos; } class MyResourceResolver implements ResourceResolver { -- cgit v1.2.3