aboutsummaryrefslogtreecommitdiffstats
path: root/src/java/org/apache/fop/fonts/FontInfo.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/java/org/apache/fop/fonts/FontInfo.java')
-rw-r--r--src/java/org/apache/fop/fonts/FontInfo.java194
1 files changed, 114 insertions, 80 deletions
diff --git a/src/java/org/apache/fop/fonts/FontInfo.java b/src/java/org/apache/fop/fonts/FontInfo.java
index 950134eb6..9ac4c1e6a 100644
--- a/src/java/org/apache/fop/fonts/FontInfo.java
+++ b/src/java/org/apache/fop/fonts/FontInfo.java
@@ -19,19 +19,18 @@
package org.apache.fop.fonts;
-import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
+import java.util.Set;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
-
/**
- * The FontInfo for the layout and rendering of a fo document.
+ * 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
* the renderer. The font name can be retrieved for the
* family style and weight.
@@ -47,38 +46,40 @@ public class FontInfo {
protected static Log log = LogFactory.getLog(FontInfo.class);
/** Map containing fonts that have been used */
- private Map usedFonts; //Map<String,FontMetrics> (String = font key)
+ private Map/*<String,FontMetrics>*/ usedFonts = null; //(String = font key)
/** look up a font-triplet to find a font-name */
- private Map triplets; //Map<FontTriplet,String> (String = font key)
+ private Map/*<FontTriplet,String>*/ triplets = null; //(String = font key)
/** look up a font-triplet to find its priority
* (only used inside addFontProperties()) */
- private Map tripletPriorities; //Map<FontTriplet,Integer>
+ private Map/*<FontTriplet,Integer>*/ tripletPriorities = null; //Map<FontTriplet,Integer>
/** look up a font-name to get a font (that implements FontMetrics at least) */
- private Map fonts; //Map<String,FontMetrics> (String = font key)
+ private Map/*<String,FontMetrics>*/ fonts = null; //(String = font key)
- /** collection of missing fonts; used to make sure the user gets
+ /**
+ * a collection of missing fonts; used to make sure the user gets
* a warning for a missing font only once (not every time the font is used)
*/
- private Collection loggedFontKeys;
+ private Set/*<FontTriplet>*/ loggedFontKeys = null;
/** Cache for Font instances. */
- private Map fontInstanceCache = new java.util.HashMap();
-
- private FontEventListener eventListener;
+ private Map/*<FontTriplet, Map>*/ fontInstanceCache = null;
+ /** Event listener for font events */
+ private FontEventListener eventListener = null;
+
/**
* Main constructor
*/
public FontInfo() {
- this.triplets = new java.util.HashMap();
- this.tripletPriorities = new java.util.HashMap();
- this.fonts = new java.util.HashMap();
- this.usedFonts = new java.util.HashMap();
+ this.triplets = new java.util.HashMap/*<FontTriplet, String>*/();
+ this.tripletPriorities = new java.util.HashMap/*<FontTriplet, Integer>*/();
+ this.fonts = new java.util.HashMap/*<String, FontMetrics>*/();
+ this.usedFonts = new java.util.HashMap/*<String,FontMetrics>*/();
}
-
+
/**
* Sets the font event listener that can be used to receive events about particular events
* in this class.
@@ -112,16 +113,16 @@ public class FontInfo {
/**
* Adds a new font triplet.
- * @param name internal key
+ * @param internalFontKey internal font key
* @param triplet the font triplet to associate with the internal key
*/
- public void addFontProperties(String name, FontTriplet triplet) {
+ public void addFontProperties(String internalFontKey, FontTriplet triplet) {
/*
* add the given family, style and weight as a lookup for the font
* with the given name
*/
if (log.isDebugEnabled()) {
- log.debug("Registering: " + triplet + " under " + name);
+ log.debug("Registering: " + triplet + " under " + internalFontKey);
}
String oldName = (String)triplets.get(triplet);
int newPriority = triplet.getPriority();
@@ -129,18 +130,20 @@ public class FontInfo {
int oldPriority = ((Integer)tripletPriorities.get(triplet)).intValue();
if (oldPriority < newPriority) {
logDuplicateFont(triplet, false, oldName, oldPriority,
- name, newPriority);
+ internalFontKey, newPriority);
return;
} else {
logDuplicateFont(triplet, true, oldName, oldPriority,
- name, newPriority);
+ internalFontKey, newPriority);
}
}
- this.triplets.put(triplet, name);
+ this.triplets.put(triplet, internalFontKey);
this.tripletPriorities.put(triplet, new Integer(newPriority));
}
- /** Log warning about duplicate font triplets.
+ /**
+ * Log warning about duplicate font triplets.
+ *
* @param triplet the duplicate font triplet
* @param replacing true iff the new font will replace the old one
* @param oldKey the old internal font name
@@ -155,24 +158,24 @@ public class FontInfo {
log.debug(triplet
+ (replacing ? ": Replacing " : ": Not replacing ")
+ ((FontMetrics)fonts.get(triplets.get(triplet))).getFullName()
- + " (" + oldPriority + ") by "
+ + " (priority=" + oldPriority + ") by "
+ ((FontMetrics)fonts.get(newKey)).getFullName()
- + " (" + newPriority + ")");
+ + " (priority=" + newPriority + ")");
}
}
/**
* Adds font metrics for a specific font.
- * @param name internal key
+ * @param internalFontKey internal key
* @param metrics metrics to register
*/
- public void addMetrics(String name, FontMetrics metrics) {
+ public void addMetrics(String internalFontKey, FontMetrics metrics) {
// add the given metrics as a font with the given name
if (metrics instanceof Typeface) {
((Typeface)metrics).setEventListener(this.eventListener);
}
- this.fonts.put(name, metrics);
+ this.fonts.put(internalFontKey, metrics);
}
/**
@@ -187,71 +190,72 @@ public class FontInfo {
* @param weight font weight
* @param substFont true if the font may be substituted with the
* default font if not found
- * @return internal key
+ * @return internal font triplet key
*/
private FontTriplet fontLookup(String family, String style,
int weight, boolean substFont) {
if (log.isTraceEnabled()) {
log.trace("Font lookup: " + family + " " + style + " " + weight);
- }
+ }
+
FontTriplet startKey = createFontKey(family, style, weight);
- FontTriplet key = startKey;
+ FontTriplet fontTriplet = startKey;
// first try given parameters
- String f = getInternalFontKey(key);
- if (f == null) {
- key = doAdjustedLookup(family, style, weight, startKey, substFont);
+ String internalFontKey = getInternalFontKey(fontTriplet);
+ if (internalFontKey == null) {
+ fontTriplet = fuzzyFontLookup(family, style, weight, startKey, substFont);
}
- if (key != null) {
- if (key != startKey) {
- notifyFontReplacement(startKey, key);
+ if (fontTriplet != null) {
+ if (fontTriplet != startKey) {
+ notifyFontReplacement(startKey, fontTriplet);
}
- return key;
+ return fontTriplet;
} else {
return null;
}
}
- private FontTriplet doAdjustedLookup(String family, String style,
+ private FontTriplet fuzzyFontLookup(String family, String style,
int weight, FontTriplet startKey, boolean substFont) {
FontTriplet key;
- String f;
+ String internalFontKey;
if (!family.equals(startKey.getName())) {
key = createFontKey(family, style, weight);
- f = getInternalFontKey(key);
- if (f != null) {
+ internalFontKey = getInternalFontKey(key);
+ if (internalFontKey != null) {
return key;
}
}
// adjust weight, favouring normal or bold
key = findAdjustWeight(family, style, weight);
- f = getInternalFontKey(key);
+ internalFontKey = getInternalFontKey(key);
- if (!substFont && f == null) {
+ if (!substFont && internalFontKey == null) {
return null;
}
// only if the font may be substituted
// fallback 1: try the same font-family and weight with default style
- if (f == null && style != Font.STYLE_NORMAL) {
+ if (internalFontKey == null && style != Font.STYLE_NORMAL) {
key = createFontKey(family, Font.STYLE_NORMAL, weight);
- f = getInternalFontKey(key);
+ internalFontKey = getInternalFontKey(key);
}
- if (f == null && weight != Font.WEIGHT_NORMAL) {
+ if (internalFontKey == null && weight != Font.WEIGHT_NORMAL) {
int diffWeight = (Font.WEIGHT_NORMAL - weight) / 100;
int direction = diffWeight > 0 ? 1 : -1;
int tryWeight = weight;
while (tryWeight != Font.WEIGHT_NORMAL) {
tryWeight += 100 * direction;
key = createFontKey(family, style, weight);
- f = getInternalFontKey(key);
- if (f == null) {
+ internalFontKey = getInternalFontKey(key);
+ if (internalFontKey == null) {
key = createFontKey(family, Font.STYLE_NORMAL, weight);
- f = getInternalFontKey(key);
+ internalFontKey = getInternalFontKey(key);
}
- if (f != null) {
+ if (internalFontKey != null) {
break;
}
}
@@ -265,17 +269,17 @@ public class FontInfo {
}*/
// fallback 3: try any family with orig style/weight
- if (f == null) {
- return doAdjustedLookup("any", style, weight, startKey, false);
+ if (internalFontKey == null) {
+ return fuzzyFontLookup("any", style, weight, startKey, false);
}
// last resort: use default
- if (f == null) {
+ if (internalFontKey == null) {
key = Font.DEFAULT_FONT;
- f = getInternalFontKey(key);
+ internalFontKey = getInternalFontKey(key);
}
- if (f != null) {
+ if (internalFontKey != null) {
return key;
} else {
return null;
@@ -290,6 +294,13 @@ public class FontInfo {
usedFonts.put(internalName, fonts.get(internalName));
}
+ private Map/*<FontTriplet,Map<Integer,Font>>*/ getFontInstanceCache() {
+ if (fontInstanceCache == null) {
+ fontInstanceCache = new java.util.HashMap/*<FontTriplet, Map<Integer,Font>>*/();
+ }
+ return fontInstanceCache;
+ }
+
/**
* Retrieves a (possibly cached) Font instance based on a FontTriplet and a font size.
* @param triplet the font triplet designating the requested font
@@ -297,10 +308,11 @@ public class FontInfo {
* @return the requested Font instance
*/
public Font getFontInstance(FontTriplet triplet, int fontSize) {
- Map sizes = (Map)fontInstanceCache.get(triplet);
+ Map/*<Integer,Font>*/ sizes
+ = (Map/*<Integer,Font>*/)getFontInstanceCache().get(triplet);
if (sizes == null) {
- sizes = new java.util.HashMap();
- fontInstanceCache.put(triplet, sizes);
+ sizes = new java.util.HashMap/*<Integer,Font>*/();
+ getFontInstanceCache().put(triplet, sizes);
}
Integer size = new Integer(fontSize);
Font font = (Font)sizes.get(size);
@@ -350,7 +362,7 @@ public class FontInfo {
throw new IllegalArgumentException("Specify at least one font family");
}
FontTriplet triplet;
- List tmpTriplets = new ArrayList();
+ List tmpTriplets = new java.util.ArrayList();
for (int i = 0, c = families.length; i < c; i++) {
triplet = fontLookup(families[i], style, weight, (i >= families.length - 1));
if (triplet != null) {
@@ -372,12 +384,16 @@ public class FontInfo {
+ "FontTriplet on the last call. Lookup: " + sb.toString());
}
- private void notifyFontReplacement(FontTriplet replacedKey, FontTriplet newKey) {
+ private Set/*<FontTriplet>*/ getLoggedFontKeys() {
if (loggedFontKeys == null) {
- loggedFontKeys = new java.util.HashSet();
+ loggedFontKeys = new java.util.HashSet/*<FontTriplet>*/();
}
- if (!loggedFontKeys.contains(replacedKey)) {
- loggedFontKeys.add(replacedKey);
+ return loggedFontKeys;
+ }
+
+ private void notifyFontReplacement(FontTriplet replacedKey, FontTriplet newKey) {
+ if (!getLoggedFontKeys().contains(replacedKey)) {
+ getLoggedFontKeys().add(replacedKey);
if (this.eventListener != null) {
this.eventListener.fontSubstituted(this, replacedKey, newKey);
} else {
@@ -477,16 +493,16 @@ public class FontInfo {
* Gets a Map of all registered fonts.
* @return a read-only Map with font key/FontMetrics pairs
*/
- public Map getFonts() {
+ public Map/*<String,FontMetrics>*/ getFonts() {
return java.util.Collections.unmodifiableMap(this.fonts);
}
/**
* Gets a Map of all registered font triplets.
- * @return a read-only Map with FontTriplet/font key pairs
+ * @return a Map with FontTriplet/font key pairs
*/
- public Map getFontTriplets() {
- return java.util.Collections.unmodifiableMap(this.triplets);
+ public Map/*<FontTriplet,String>*/ getFontTriplets() {
+ return this.triplets;
}
/**
@@ -495,7 +511,7 @@ public class FontInfo {
* This is for embedded font or creating a list of used fonts.
* @return a read-only Map with font key/FontMetrics pairs
*/
- public Map getUsedFonts() {
+ public Map/*<String,FontMetrics>*/ getUsedFonts() {
return this.usedFonts;
}
@@ -511,20 +527,30 @@ public class FontInfo {
}
/**
- * Returns the first triplet matching the given font name.
- * As there may be multiple triplets matching the font name
- * the result set is sorted first to guarantee consistent results.
+ * Returns all font triplet matching the given font name.
* @param fontName The font name we are looking for
- * @return The first triplet for the given font name
+ * @return A list of matching font triplets
*/
- public FontTriplet getTripletFor(String fontName) {
- List foundTriplets = new ArrayList();
+ public List/*<FontTriplet>*/ getTripletsFor(String fontName) {
+ List/*<FontTriplet>*/ foundTriplets = new java.util.ArrayList();
for (Iterator iter = triplets.entrySet().iterator(); iter.hasNext();) {
Map.Entry tripletEntry = (Map.Entry) iter.next();
if (fontName.equals(((String)tripletEntry.getValue()))) {
foundTriplets.add(tripletEntry.getKey());
}
}
+ return foundTriplets;
+ }
+
+ /**
+ * Returns the first triplet matching the given font name.
+ * As there may be multiple triplets matching the font name
+ * the result set is sorted first to guarantee consistent results.
+ * @param fontName The font name we are looking for
+ * @return The first triplet for the given font name
+ */
+ public FontTriplet getTripletFor(String fontName) {
+ List/*<FontTriplet>*/ foundTriplets = getTripletsFor(fontName);
if (foundTriplets.size() > 0) {
Collections.sort(foundTriplets);
return (FontTriplet)foundTriplets.get(0);
@@ -570,17 +596,25 @@ public class FontInfo {
* Diagnostic method for logging all registered fonts to System.out.
*/
public void dumpAllTripletsToSystemOut() {
+ System.out.print(toString());
+ }
+
+ /**
+ * {@inheritDoc}
+ */
+ public String toString() {
Collection entries = new java.util.TreeSet();
Iterator iter = this.triplets.keySet().iterator();
while (iter.hasNext()) {
FontTriplet triplet = (FontTriplet)iter.next();
String key = getInternalFontKey(triplet);
FontMetrics metrics = getMetricsFor(key);
- entries.add(triplet.toString() + " -> " + key + " -> " + metrics.getFontName());
+ entries.add(triplet.toString() + " -> " + key + " -> " + metrics.getFontName() + "\n");
}
- iter = entries.iterator();
- while (iter.hasNext()) {
- System.out.println(iter.next());
+ StringBuffer stringBuffer = new StringBuffer();
+ for (iter = entries.iterator(); iter.hasNext();) {
+ stringBuffer.append(iter.next());
}
+ return stringBuffer.toString();
}
}