/*

   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.

 */
package org.apache.batik.bridge;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.StringTokenizer;

import org.apache.batik.gvt.font.GVTFontFamily;
import org.apache.batik.gvt.font.GVTFontFace;
import org.apache.batik.gvt.font.UnresolvedFontFamily;
import org.apache.batik.util.SVGConstants;

import org.apache.batik.anim.dom.SVGOMDocument;
import org.apache.batik.css.engine.CSSEngine;
import org.apache.batik.css.engine.FontFaceRule;

import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NodeList;

Utility class for SVG fonts.
Author:Bella Robinson
Version:$Id: SVGFontUtilities.java 1804130 2017-08-04 14:41:11Z ssteiner $
/** * Utility class for SVG fonts. * * @author <a href="mailto:bella.robinson@cmis.csiro.au">Bella Robinson</a> * @version $Id: SVGFontUtilities.java 1804130 2017-08-04 14:41:11Z ssteiner $ */
public abstract class SVGFontUtilities implements SVGConstants { public static List getFontFaces(Document doc, BridgeContext ctx) { // check fontFamilyMap to see if we have already created an // FontFamily that matches Map fontFamilyMap = ctx.getFontFamilyMap(); List ret = (List)fontFamilyMap.get(doc); if (ret != null) return ret; ret = new LinkedList(); NodeList fontFaceElements = doc.getElementsByTagNameNS (SVG_NAMESPACE_URI, SVG_FONT_FACE_TAG); SVGFontFaceElementBridge fontFaceBridge; fontFaceBridge = (SVGFontFaceElementBridge)ctx.getBridge (SVG_NAMESPACE_URI, SVG_FONT_FACE_TAG); for (int i = 0; i < fontFaceElements.getLength(); i++) { Element fontFaceElement = (Element)fontFaceElements.item(i); ret.add(fontFaceBridge.createFontFace (ctx, fontFaceElement)); } CSSEngine engine = ((SVGOMDocument)doc).getCSSEngine(); List sms = engine.getFontFaces(); for (Object sm : sms) { FontFaceRule ffr = (FontFaceRule) sm; ret.add(CSSFontFace.createCSSFontFace(engine, ffr)); } return ret; }
Given a font family name tries to find a matching SVG font object. If finds one, returns an SVGFontFamily otherwise returns an UnresolvedFontFamily.
Params:
  • textElement – The text element that the font family will be attached to.
  • ctx – The bridge context, used to search for a matching SVG font element.
  • fontFamilyName – The name of the font family to search for.
  • fontWeight – The weight of the font to use when trying to match an SVG font family.
  • fontStyle – The style of the font to use when trying to match as SVG font family.
Returns:A GVTFontFamily for the specified font attributes. This will be unresolved unless a matching SVG font was found.
/** * Given a font family name tries to find a matching SVG font * object. If finds one, returns an SVGFontFamily otherwise * returns an UnresolvedFontFamily. * * @param textElement The text element that the font family will * be attached to. * @param ctx The bridge context, used to search for a matching * SVG font element. * @param fontFamilyName The name of the font family to search * for. * @param fontWeight The weight of the font to use when trying to * match an SVG font family. * @param fontStyle The style of the font to use when trying to * match as SVG font family. * * @return A GVTFontFamily for the specified font attributes. This * will be unresolved unless a matching SVG font was found. */
public static GVTFontFamily getFontFamily(Element textElement, BridgeContext ctx, String fontFamilyName, String fontWeight, String fontStyle) { // TODO: should match against font-variant as well String fontKeyName = fontFamilyName.toLowerCase() + " " + // todo locale?? fontWeight + " " + fontStyle; // check fontFamilyMap to see if we have already created an // FontFamily that matches Map fontFamilyMap = ctx.getFontFamilyMap(); GVTFontFamily fontFamily = (GVTFontFamily)fontFamilyMap.get(fontKeyName); if (fontFamily != null) { return fontFamily; } // try to find a matching SVGFontFace element Document doc = textElement.getOwnerDocument(); List fontFaces = (List)fontFamilyMap.get(doc); if (fontFaces == null) { fontFaces = getFontFaces(doc, ctx); fontFamilyMap.put(doc, fontFaces); } Iterator iter = fontFaces.iterator(); List svgFontFamilies = new LinkedList(); while (iter.hasNext()) { FontFace fontFace = (FontFace)iter.next(); if (!fontFace.hasFamilyName(fontFamilyName)) { continue; } String fontFaceStyle = fontFace.getFontStyle(); if (fontFaceStyle.equals(SVG_ALL_VALUE) || fontFaceStyle.indexOf(fontStyle) != -1) { GVTFontFamily ffam = fontFace.getFontFamily(ctx); if (ffam != null) svgFontFamilies.add(ffam); } } if (svgFontFamilies.size() == 1) { // only found one matching svg font family fontFamilyMap.put(fontKeyName, svgFontFamilies.get(0)); return (GVTFontFamily)svgFontFamilies.get(0); } else if (svgFontFamilies.size() > 1) { // need to find font face that matches the font-weight closest String fontWeightNumber = getFontWeightNumberString(fontWeight); // create lists of font weight numbers for each font family List fontFamilyWeights = new ArrayList(svgFontFamilies.size()); for (Object svgFontFamily : svgFontFamilies) { GVTFontFace fontFace; fontFace = ((GVTFontFamily) svgFontFamily).getFontFace(); String fontFaceWeight = fontFace.getFontWeight(); fontFaceWeight = getFontWeightNumberString(fontFaceWeight); fontFamilyWeights.add(fontFaceWeight); } // make sure that each possible font-weight has been // assigned to a font-face, if not then need to "fill the // holes" List newFontFamilyWeights = new ArrayList(fontFamilyWeights); for (int i = 100; i <= 900; i+= 100) { String weightString = String.valueOf(i); boolean matched = false; int minDifference = 1000; int minDifferenceIndex = 0; for (int j = 0; j < fontFamilyWeights.size(); j++) { String fontFamilyWeight = (String)fontFamilyWeights.get(j); if (fontFamilyWeight.indexOf(weightString) > -1) { matched = true; break; } StringTokenizer st = new StringTokenizer(fontFamilyWeight, " ,"); while (st.hasMoreTokens()) { int weightNum = Integer.parseInt(st.nextToken()); int difference = Math.abs(weightNum - i); if (difference < minDifference) { minDifference = difference; minDifferenceIndex = j; } } } if (!matched) { String newFontFamilyWeight = newFontFamilyWeights.get(minDifferenceIndex) + ", " + weightString; newFontFamilyWeights.set(minDifferenceIndex, newFontFamilyWeight); } } // now find matching font weight for (int i = 0; i < svgFontFamilies.size(); i++) { String fontFaceWeight = (String)newFontFamilyWeights.get(i); if (fontFaceWeight.indexOf(fontWeightNumber) > -1) { fontFamilyMap.put(fontKeyName, svgFontFamilies.get(i)); return (GVTFontFamily)svgFontFamilies.get(i); } } // should not get here, just return the first svg font family fontFamilyMap.put(fontKeyName, svgFontFamilies.get(0)); return (GVTFontFamily) svgFontFamilies.get(0); } else { // couldn't find one so return an UnresolvedFontFamily object GVTFontFamily gvtFontFamily = new UnresolvedFontFamily(fontFamilyName); fontFamilyMap.put(fontKeyName, gvtFontFamily); return gvtFontFamily; } }
Returns a string that contains all of the font weight numbers for the specified font weight attribute value.
Params:
  • fontWeight – The font-weight attribute value.
Returns:The font weight expressed as font weight numbers. e.g. "normal" becomes "400".
/** * Returns a string that contains all of the font weight numbers for the * specified font weight attribute value. * * @param fontWeight The font-weight attribute value. * * @return The font weight expressed as font weight numbers. * e.g. "normal" becomes "400". */
protected static String getFontWeightNumberString(String fontWeight) { if (fontWeight.equals(SVG_NORMAL_VALUE)) { return SVG_400_VALUE; } else if (fontWeight.equals(SVG_BOLD_VALUE)) { return SVG_700_VALUE; } else if (fontWeight.equals(SVG_ALL_VALUE)) { return "100, 200, 300, 400, 500, 600, 700, 800, 900"; } return fontWeight; } }