com.kitfox.svg.MissingGlyph Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of svgSalamander Show documentation
Show all versions of svgSalamander Show documentation
A tool for displaying and playing SVG content using the Java2D.
/*
* 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 [email protected]. 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 com.kitfox.svg.pathcmd.BuildHistory;
import com.kitfox.svg.pathcmd.PathCommand;
import com.kitfox.svg.xml.StyleAttribute;
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;
/**
* Implements an embedded font.
*
* SVG specification: http://www.w3.org/TR/SVG/fonts.html
*
* @author Mark McKay
* @author Mark McKay
*/
public class MissingGlyph extends ShapeElement
{
public static final String TAG_NAME = "missingglyph";
//We may define a path
private Shape path = null;
//Alternately, we may have child graphical elements
private float horizAdvX = -1; //Inherits font's value if not set
private float vertOriginX = -1; //Inherits font's value if not set
private float vertOriginY = -1; //Inherits font's value if not set
private float 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.getFloatValue();
}
if (getPres(sty.setName("vert-origin-x")))
{
vertOriginX = sty.getFloatValue();
}
if (getPres(sty.setName("vert-origin-y")))
{
vertOriginY = sty.getFloatValue();
}
if (getPres(sty.setName("vert-adv-y")))
{
vertAdvY = sty.getFloatValue();
}
}
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 = it.next();
if (ele instanceof RenderableElement)
{
((RenderableElement) ele).render(g);
}
}
//Do not push or pop stack
}
public float getHorizAdvX()
{
if (horizAdvX == -1)
{
horizAdvX = ((Font) parent).getHorizAdvX();
}
return horizAdvX;
}
public float getVertOriginX()
{
if (vertOriginX == -1)
{
vertOriginX = getHorizAdvX() / 2;
}
return vertOriginX;
}
public float getVertOriginY()
{
if (vertOriginY == -1)
{
vertOriginY = ((Font) parent).getFontFace().getAscent();
}
return vertOriginY;
}
public float 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;
}
/**
* @param path the path to set
*/
public void setPath(Shape path)
{
this.path = path;
}
/**
* @param horizAdvX the horizAdvX to set
*/
public void setHorizAdvX(float horizAdvX)
{
this.horizAdvX = horizAdvX;
}
/**
* @param vertOriginX the vertOriginX to set
*/
public void setVertOriginX(float vertOriginX)
{
this.vertOriginX = vertOriginX;
}
/**
* @param vertOriginY the vertOriginY to set
*/
public void setVertOriginY(float vertOriginY)
{
this.vertOriginY = vertOriginY;
}
/**
* @param vertAdvY the vertAdvY to set
*/
public void setVertAdvY(float vertAdvY)
{
this.vertAdvY = vertAdvY;
}
}