/* * 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. */ /* $Id$ */ package org.apache.fop.fonts; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Set; import java.util.TreeSet; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.apache.xmlgraphics.fonts.Glyphs; import org.apache.fop.apps.io.InternalResourceResolver; import org.apache.fop.fonts.truetype.TTFFile.PostScriptVersion; /** * Generic SingleByte font */ public class SingleByteFont extends CustomFont { /** logger */ private static Log log = LogFactory.getLog(SingleByteFont.class); private SingleByteEncoding mapping; private boolean useNativeEncoding = false; private int[] width = null; private Map unencodedCharacters; private List additionalEncodings; private Map alternativeCodes; private PostScriptVersion ttPostScriptVersion; /** * @param resourceResolver the URI resolver for controlling file access */ public SingleByteFont(InternalResourceResolver resourceResolver) { super(resourceResolver); setEncoding(CodePointMapping.WIN_ANSI_ENCODING); } /** {@inheritDoc} */ public boolean isEmbeddable() { return (!(getEmbedFileURI() == null && getEmbedResourceName() == null)); } /** {@inheritDoc} */ public boolean isSubsetEmbedded() { return false; } /** {@inheritDoc} */ public String getEncodingName() { return this.mapping.getName(); } /** * Returns the code point mapping (encoding) of this font. * @return the code point mapping */ public SingleByteEncoding getEncoding() { return this.mapping; } /** {@inheritDoc} */ public int getWidth(int i, int size) { if (i < 256) { int idx = i - getFirstChar(); if (idx >= 0 && idx < width.length) { return size * width[idx]; } } else if (this.additionalEncodings != null) { int encodingIndex = (i / 256) - 1; SimpleSingleByteEncoding encoding = getAdditionalEncoding(encodingIndex); int codePoint = i % 256; NamedCharacter nc = encoding.getCharacterForIndex(codePoint); UnencodedCharacter uc = this.unencodedCharacters.get(Character.valueOf(nc.getSingleUnicodeValue())); return size * uc.getWidth(); } return 0; } /** {@inheritDoc} */ public int[] getWidths() { int[] arr = new int[width.length]; System.arraycopy(width, 0, arr, 0, width.length); return arr; } /** * Lookup a character using its alternative names. If found, cache it so we * can speed up lookups. * @param c the character * @return the suggested alternative character present in the font */ private char findAlternative(char c) { char d; if (alternativeCodes == null) { alternativeCodes = new java.util.HashMap(); } else { Character alternative = alternativeCodes.get(c); if (alternative != null) { return alternative; } } String charName = Glyphs.charToGlyphName(c); String[] charNameAlternatives = Glyphs.getCharNameAlternativesFor(charName); if (charNameAlternatives != null && charNameAlternatives.length > 0) { for (int i = 0; i < charNameAlternatives.length; i++) { if (log.isDebugEnabled()) { log.debug("Checking alternative for char " + c + " (charname=" + charName + "): " + charNameAlternatives[i]); } String s = Glyphs.getUnicodeSequenceForGlyphName(charNameAlternatives[i]); if (s != null) { d = lookupChar(s.charAt(0)); if (d != SingleByteEncoding.NOT_FOUND_CODE_POINT) { alternativeCodes.put(c, d); return d; } } } } return SingleByteEncoding.NOT_FOUND_CODE_POINT; } private char lookupChar(char c) { char d = mapping.mapChar(c); if (d != SingleByteEncoding.NOT_FOUND_CODE_POINT) { return d; } // Check unencoded characters which are available in the font by // character name d = mapUnencodedChar(c); return d; } /** {@inheritDoc} */ @Override public char mapChar(char c) { notifyMapOperation(); char d = lookupChar(c); if (d != SingleByteEncoding.NOT_FOUND_CODE_POINT) { return d; } else { // Check for alternative d = findAlternative(c); if (d != SingleByteEncoding.NOT_FOUND_CODE_POINT) { return d; } } this.warnMissingGlyph(c); return Typeface.NOT_FOUND; } private char mapUnencodedChar(char ch) { if (this.unencodedCharacters != null) { UnencodedCharacter unencoded = this.unencodedCharacters.get(Character.valueOf(ch)); if (unencoded != null) { if (this.additionalEncodings == null) { this.additionalEncodings = new ArrayList(); } SimpleSingleByteEncoding encoding = null; char mappedStart = 0; int additionalsCount = this.additionalEncodings.size(); for (int i = 0; i < additionalsCount; i++) { mappedStart += 256; encoding = getAdditionalEncoding(i); char alt = encoding.mapChar(ch); if (alt != 0) { return (char)(mappedStart + alt); } } if (encoding != null && encoding.isFull()) { encoding = null; } if (encoding == null) { encoding = new SimpleSingleByteEncoding( getFontName() + "EncodingSupp" + (additionalsCount + 1)); this.additionalEncodings.add(encoding); mappedStart += 256; } return (char)(mappedStart + encoding.addCharacter(unencoded.getCharacter())); } } return 0; } /** {@inheritDoc} */ @Override public boolean hasChar(char c) { char d = mapping.mapChar(c); if (d != SingleByteEncoding.NOT_FOUND_CODE_POINT) { return true; } //Check unencoded characters which are available in the font by character name d = mapUnencodedChar(c); if (d != SingleByteEncoding.NOT_FOUND_CODE_POINT) { return true; } // Check if an alternative exists d = findAlternative(c); if (d != SingleByteEncoding.NOT_FOUND_CODE_POINT) { return true; } return false; } /* ---- single byte font specific setters --- */ /** * Updates the mapping variable based on the encoding. * @param encoding the name of the encoding */ protected void updateMapping(String encoding) { try { this.mapping = CodePointMapping.getMapping(encoding); } catch (UnsupportedOperationException e) { log.error("Font '" + super.getFontName() + "': " + e.getMessage()); } } /** * Sets the encoding of the font. * @param encoding the encoding (ex. "WinAnsiEncoding" or "SymbolEncoding") */ public void setEncoding(String encoding) { updateMapping(encoding); } /** * Sets the encoding of the font. * @param encoding the encoding information */ public void setEncoding(CodePointMapping encoding) { this.mapping = encoding; } /** * Controls whether the font is configured to use its native encoding or if it * may need to be re-encoded for the target format. * @param value true indicates that the configured encoding is the font's native encoding */ public void setUseNativeEncoding(boolean value) { this.useNativeEncoding = value; } /** * Indicates whether this font is configured to use its native encoding. This * method is used to determine whether the font needs to be re-encoded. * @return true if the font uses its native encoding. */ public boolean isUsingNativeEncoding() { return this.useNativeEncoding; } /** * Sets a width for a character. * @param index index of the character * @param w the width of the character */ public void setWidth(int index, int w) { if (this.width == null) { this.width = new int[getLastChar() - getFirstChar() + 1]; } this.width[index - getFirstChar()] = w; } /** * Adds an unencoded character (one that is not supported by the primary encoding). * @param ch the named character * @param width the width of the character */ public void addUnencodedCharacter(NamedCharacter ch, int width) { if (this.unencodedCharacters == null) { this.unencodedCharacters = new HashMap(); } if (ch.hasSingleUnicodeValue()) { UnencodedCharacter uc = new UnencodedCharacter(ch, width); this.unencodedCharacters.put(Character.valueOf(ch.getSingleUnicodeValue()), uc); } else { //Cannot deal with unicode sequences, so ignore this character } } /** * Makes all unencoded characters available through additional encodings. This method * is used in cases where the fonts need to be encoded in the target format before * all text of the document is processed (for example in PostScript when resource optimization * is disabled). */ public void encodeAllUnencodedCharacters() { if (this.unencodedCharacters != null) { Set sortedKeys = new TreeSet(this.unencodedCharacters.keySet()); for (Character ch : sortedKeys) { char mapped = mapChar(ch.charValue()); assert mapped != Typeface.NOT_FOUND; } } } /** * Indicates whether the encoding has additional encodings besides the primary encoding. * @return true if there are additional encodings. */ public boolean hasAdditionalEncodings() { return (this.additionalEncodings != null) && (this.additionalEncodings.size() > 0); } /** * Returns the number of additional encodings this single-byte font maintains. * @return the number of additional encodings */ public int getAdditionalEncodingCount() { if (hasAdditionalEncodings()) { return this.additionalEncodings.size(); } else { return 0; } } /** * Returns an additional encoding. * @param index the index of the additional encoding * @return the additional encoding * @throws IndexOutOfBoundsException if the index is out of bounds */ public SimpleSingleByteEncoding getAdditionalEncoding(int index) throws IndexOutOfBoundsException { if (hasAdditionalEncodings()) { return this.additionalEncodings.get(index); } else { throw new IndexOutOfBoundsException("No additional encodings available"); } } /** * Returns an array with the widths for an additional encoding. * @param index the index of the additional encoding * @return the width array */ public int[] getAdditionalWidths(int index) { SimpleSingleByteEncoding enc = getAdditionalEncoding(index); int[] arr = new int[enc.getLastChar() - enc.getFirstChar() + 1]; for (int i = 0, c = arr.length; i < c; i++) { NamedCharacter nc = enc.getCharacterForIndex(enc.getFirstChar() + i); UnencodedCharacter uc = this.unencodedCharacters.get( Character.valueOf(nc.getSingleUnicodeValue())); arr[i] = uc.getWidth(); } return arr; } private static final class UnencodedCharacter { private final NamedCharacter character; private final int width; public UnencodedCharacter(NamedCharacter character, int width) { this.character = character; this.width = width; } public NamedCharacter getCharacter() { return this.character; } public int getWidth() { return this.width; } /** {@inheritDoc} */ @Override public String toString() { return getCharacter().toString(); } } /** * Sets the version of the PostScript table stored in the TrueType font represented by * this instance. * * @param version version of the post table */ public void setTrueTypePostScriptVersion(PostScriptVersion version) { ttPostScriptVersion = version; } /** * Returns the version of the PostScript table stored in the TrueType font represented by * this instance. * * @return the version of the post table */ public PostScriptVersion getTrueTypePostScriptVersion() { assert getFontType() == FontType.TRUETYPE; return ttPostScriptVersion; } }