/* * 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.afp.fonts; import java.awt.Rectangle; /** * The IBM Font Object Content Architecture (FOCA) supports presentation * of character shapes by defining their characteristics, which include * Font-Description information for identifying the characters, Font-Metric * information for positioning the characters, and Character-Shape * information for presenting the character images. * * Presenting a graphic character on a presentation surface requires * that you communicate this information clearly to rotate and position * characters correctly on the physical or logical page. * * This class provides font metric information for a particular font * as by the orientation. * * This information is obtained directly from the AFP font files which must * be installed in the classpath under in the location specified by the path * attribute in the afp-font.xml file. *

*/ public class CharacterSetOrientation { /** * The ascender height for the character set */ private int ascender; /** * The descender depth for the character set */ private int descender; /** * The height of capital letters */ private int capHeight; /** * The character widths in the character set (indexed using Unicode codepoints) */ private IntegerKeyStore characterMetrics; /** * The height of lowercase letters */ private int xHeight; /** The character set orientation */ private final int orientation; /** space increment */ private final int spaceIncrement; /** em space increment */ private final int emSpaceIncrement; /** Nominal Character Increment */ private final int nomCharIncrement; private int underscoreWidth; private int underscorePosition; /** * Constructor for the CharacterSetOrientation, the orientation is * expressed as the degrees rotation (i.e 0, 90, 180, 270) * @param orientation the character set orientation */ public CharacterSetOrientation(int orientation, int spaceIncrement, int emSpaceIncrement, int nomCharIncrement) { this.orientation = orientation; this.spaceIncrement = spaceIncrement; this.emSpaceIncrement = emSpaceIncrement; this.nomCharIncrement = nomCharIncrement; this.characterMetrics = new IntegerKeyStore(); } /** * Ascender height is the distance from the character baseline to the * top of the character box. A negative ascender height signifies that * all of the graphic character is below the character baseline. For * a character rotation other than 0, ascender height loses its * meaning when the character is lying on its side or is upside down * with respect to normal viewing orientation. For the general case, * Ascender Height is the character's most positive y-axis value. * For bounded character boxes, for a given character having an * ascender, ascender height and baseline offset are equal. * @return the ascender value in millipoints */ public int getAscender() { return ascender; } /** * Cap height is the average height of the uppercase characters in * a font. This value is specified by the designer of a font and is * usually the height of the uppercase M. * @return the cap height value in millipoints */ public int getCapHeight() { return capHeight; } /** * Descender depth is the distance from the character baseline to * the bottom of a character box. A negative descender depth signifies * that all of the graphic character is above the character baseline. * @return the descender value in millipoints */ public int getDescender() { return descender; } /** * TODO */ public int getUnderscoreWidth() { return underscoreWidth; } /** * TODO */ public int getUnderscorePosition() { return underscorePosition; } /** * The orientation for these metrics in the character set * @return the orientation */ public int getOrientation() { return orientation; } /** * XHeight refers to the height of the lower case letters above * the baseline. * @return heightX the typical height of characters */ public int getXHeight() { return xHeight; } /** * Get the width (in 1/1000ths of a point size) of the character * identified by the parameter passed. * @param character the Unicode character to evaluate * @return the widths of the character */ public int getWidth(char character, int size) { CharacterMetrics cm = getCharacterMetrics(character); return cm == null ? -1 : size * cm.width; } private CharacterMetrics getCharacterMetrics(char character) { return characterMetrics.get((int) character); } /** * Get the character box (rectangle with dimensions in 1/1000ths of a point size) of the character * identified by the parameter passed. * @param character the Unicode character to evaluate * @return the character box */ public Rectangle getCharacterBox(char character, int size) { CharacterMetrics cm = getCharacterMetrics(character); return scale(cm == null ? getFallbackCharacterBox() : cm.characterBox, size); } private static Rectangle scale(Rectangle rectangle, int size) { if (rectangle == null) { return null; } else { return new Rectangle((int) (size * rectangle.getX()), (int) (size * rectangle.getY()), (int) (size * rectangle.getWidth()), (int) (size * rectangle.getHeight())); } } private Rectangle getFallbackCharacterBox() { // TODO replace with something sensible return new Rectangle(0, 0, 0, 0); } /** * Ascender height is the distance from the character baseline to the * top of the character box. A negative ascender height signifies that * all of the graphic character is below the character baseline. For * a character rotation other than 0, ascender height loses its * meaning when the character is lying on its side or is upside down * with respect to normal viewing orientation. For the general case, * Ascender Height is the character's most positive y-axis value. * For bounded character boxes, for a given character having an * ascender, ascender height and baseline offset are equal. * @param ascender the ascender to set */ public void setAscender(int ascender) { this.ascender = ascender; } /** * Cap height is the average height of the uppercase characters in * a font. This value is specified by the designer of a font and is * usually the height of the uppercase M. * @param capHeight the cap height to set */ public void setCapHeight(int capHeight) { this.capHeight = capHeight; } /** * Descender depth is the distance from the character baseline to * the bottom of a character box. A negative descender depth signifies * that all of the graphic character is above the character baseline. * @param descender the descender value in millipoints */ public void setDescender(int descender) { this.descender = descender; } /** * TODO * @param underscoreWidth the underscore width value in millipoints */ public void setUnderscoreWidth(int underscoreWidth) { this.underscoreWidth = underscoreWidth; } /** * TODO * @param underscorePosition the underscore position value in millipoints */ public void setUnderscorePosition(int underscorePosition) { this.underscorePosition = underscorePosition; } /** * Set the width (in 1/1000ths of a point size) of the character * identified by the parameter passed. * @param character the Unicode character for which the width is being set * @param width the widths of the character */ public void setCharacterMetrics(char character, int width, Rectangle characterBox) { characterMetrics.put((int) character, new CharacterMetrics(width, characterBox)); } /** * XHeight refers to the height of the lower case letters above * the baseline. * @param xHeight the typical height of characters */ public void setXHeight(int xHeight) { this.xHeight = xHeight; } /** * Returns the space increment. * @return the space increment */ public int getSpaceIncrement() { return this.spaceIncrement; } /** * Returns the em space increment. * @return the em space increment */ public int getEmSpaceIncrement() { return this.emSpaceIncrement; } /** * Returns the nominal character increment. * @return the nominal character increment */ public int getNominalCharIncrement() { return this.nomCharIncrement; } private static class CharacterMetrics { public final int width; public final Rectangle characterBox; public CharacterMetrics(int width, Rectangle characterBox) { this.width = width; this.characterBox = characterBox; } } }