/* * SVG Salamander * Copyright (c) 2004, Mark McKay * All rights reserved. * * Redistribution and use in source and binary forms, with or * without modification, are permitted provided that the following * conditions are met: * * - Redistributions of source code must retain the above * copyright notice, this list of conditions and the following * disclaimer. * - Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials * provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED * OF THE POSSIBILITY OF SUCH DAMAGE. * * Mark McKay can be contacted at mark@kitfox.com. Salamander and other * projects can be found at http://www.kitfox.com * * Created on February 20, 2004, 10:00 PM */ package com.kitfox.svg; import java.awt.Graphics2D; import java.awt.Shape; import java.awt.geom.AffineTransform; import java.awt.geom.GeneralPath; import java.awt.geom.Rectangle2D; import java.util.Iterator; import com.kitfox.svg.pathcmd.BuildHistory; import com.kitfox.svg.pathcmd.PathCommand; import com.kitfox.svg.xml.StyleAttribute; /** * Implements an embedded font. * * SVG specification: http://www.w3.org/TR/SVG/fonts.html * * @author Mark McKay * @author <a href="mailto:mark@kitfox.com">Mark McKay</a> */ public class MissingGlyph extends ShapeElement { public static final String TAG_NAME = "missingglyph"; // We may define a path Shape path = null; // Alternately, we may have child graphical elements int horizAdvX = -1; // Inherits font's value if not set int vertOriginX = -1; // Inherits font's value if not set int vertOriginY = -1; // Inherits font's value if not set int vertAdvY = -1; // Inherits font's value if not set /** * Creates a new instance of Font */ public MissingGlyph() { } @Override public String getTagName() { return TAG_NAME; } /** * Called after the start element but before the end element to indicate * each child tag that has been processed */ @Override public void loaderAddChild(SVGLoaderHelper helper, SVGElement child) throws SVGElementException { super.loaderAddChild(helper, child); } @Override protected void build() throws SVGException { super.build(); StyleAttribute sty = new StyleAttribute(); String commandList = ""; if (getPres(sty.setName("d"))) { commandList = sty.getStringValue(); } // If glyph path was specified, calculate it if (commandList != null) { String fillRule = getStyle(sty.setName("fill-rule")) ? sty.getStringValue() : "nonzero"; PathCommand[] commands = parsePathList(commandList); GeneralPath buildPath = new GeneralPath(fillRule.equals("evenodd") ? GeneralPath.WIND_EVEN_ODD : GeneralPath.WIND_NON_ZERO, commands.length); BuildHistory hist = new BuildHistory(); for (int i = 0; i < commands.length; i++) { PathCommand cmd = commands[i]; cmd.appendPath(buildPath, hist); } // Reflect glyph path to put it in user coordinate system AffineTransform at = new AffineTransform(); at.scale(1, -1); path = at.createTransformedShape(buildPath); } // Read glyph spacing info if (getPres(sty.setName("horiz-adv-x"))) { horizAdvX = sty.getIntValue(); } if (getPres(sty.setName("vert-origin-x"))) { vertOriginX = sty.getIntValue(); } if (getPres(sty.setName("vert-origin-y"))) { vertOriginY = sty.getIntValue(); } if (getPres(sty.setName("vert-adv-y"))) { vertAdvY = sty.getIntValue(); } } public Shape getPath() { return path; } @Override public void render(Graphics2D g) throws SVGException { // Do not push or pop stack if (path != null) { renderShape(g, path); } Iterator it = children.iterator(); while (it.hasNext()) { SVGElement ele = (SVGElement) it.next(); if (ele instanceof RenderableElement) { ((RenderableElement) ele).render(g); } } // Do not push or pop stack } public int getHorizAdvX() { if (horizAdvX == -1) { horizAdvX = ((Font) parent).getHorizAdvX(); } return horizAdvX; } public int getVertOriginX() { if (vertOriginX == -1) { vertOriginX = getHorizAdvX() / 2; } return vertOriginX; } public int getVertOriginY() { if (vertOriginY == -1) { vertOriginY = ((Font) parent).getFontFace().getAscent(); } return vertOriginY; } public int getVertAdvY() { if (vertAdvY == -1) { vertAdvY = ((Font) parent).getFontFace().getUnitsPerEm(); } return vertAdvY; } @Override public Shape getShape() { if (path != null) { return shapeToParent(path); } return null; } @Override public Rectangle2D getBoundingBox() throws SVGException { if (path != null) { return boundsToParent(includeStrokeInBounds(path.getBounds2D())); } return null; } /** * Updates all attributes in this diagram associated with a time event. Ie, * all attributes with track information. * * @return - true if this node has changed state as a result of the time * update */ @Override public boolean updateTime(double curTime) throws SVGException { // Fonts can't change return false; } }