/* * 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.area; import java.awt.Color; import java.io.Serializable; import org.apache.xmlgraphics.image.loader.ImageInfo; import org.apache.fop.fonts.FontTriplet; import org.apache.fop.traits.BorderProps; import org.apache.fop.traits.Direction; import org.apache.fop.traits.WritingMode; import org.apache.fop.util.ColorUtil; import static org.apache.fop.fo.Constants.EN_NOREPEAT; import static org.apache.fop.fo.Constants.EN_REPEAT; import static org.apache.fop.fo.Constants.EN_REPEATX; import static org.apache.fop.fo.Constants.EN_REPEATY; // properties should be serialized by the holder /** * Area traits used for rendering. * This class represents an area trait that specifies a value for rendering. */ public final class Trait implements Serializable { private static final long serialVersionUID = 3234280285391611437L; private Trait() { } /** Id reference line, not resolved. (not sure if this is needed.) */ //public static final Integer ID_LINK = new Integer(0); /** * Internal link trait. * Contains the PageViewport key and the PROD_ID of the target area */ public static final Integer INTERNAL_LINK = 1; /** * External link. A URL link to an external resource. */ public static final Integer EXTERNAL_LINK = 2; /** The font triplet for the current font. */ public static final Integer FONT = 3; /** Font size for the current font. */ public static final Integer FONT_SIZE = 4; /** The current color. */ public static final Integer COLOR = 7; /** The ID of the FO that produced an area. */ public static final Integer PROD_ID = 8; /** Background trait for an area. */ public static final Integer BACKGROUND = 9; /** Underline trait used when rendering inline parent. */ public static final Integer UNDERLINE = 10; /** Overline trait used when rendering inline parent. */ public static final Integer OVERLINE = 11; /** Linethrough trait used when rendering inline parent. */ public static final Integer LINETHROUGH = 12; /** Shadow offset. */ //public static final Integer OFFSET = new Integer(13); /** The shadow for text. */ //public static final Integer SHADOW = new Integer(14); /** The border start. */ public static final Integer BORDER_START = 15; /** The border end. */ public static final Integer BORDER_END = 16; /** The border before. */ public static final Integer BORDER_BEFORE = 17; /** The border after. */ public static final Integer BORDER_AFTER = 18; /** The padding start. */ public static final Integer PADDING_START = 19; /** The padding end. */ public static final Integer PADDING_END = 20; /** The padding before. */ public static final Integer PADDING_BEFORE = 21; /** The padding after. */ public static final Integer PADDING_AFTER = 22; /** The space start. */ public static final Integer SPACE_START = 23; /** The space end. */ public static final Integer SPACE_END = 24; /** break before */ //public static final Integer BREAK_BEFORE = new Integer(25); /** break after */ //public static final Integer BREAK_AFTER = new Integer(26); /** The start-indent trait. */ public static final Integer START_INDENT = 27; /** The end-indent trait. */ public static final Integer END_INDENT = 28; /** The space-before trait. */ public static final Integer SPACE_BEFORE = 29; /** The space-after trait. */ public static final Integer SPACE_AFTER = 30; /** The is-reference-area trait. */ public static final Integer IS_REFERENCE_AREA = 31; /** The is-viewport-area trait. */ public static final Integer IS_VIEWPORT_AREA = 32; /** Blinking trait used when rendering inline parent. */ public static final Integer BLINK = 33; /** Trait for color of underline decorations when rendering inline parent. */ public static final Integer UNDERLINE_COLOR = 34; /** Trait for color of overline decorations when rendering inline parent. */ public static final Integer OVERLINE_COLOR = 35; /** Trait for color of linethrough decorations when rendering inline parent. */ public static final Integer LINETHROUGH_COLOR = 36; /** For navigation in the document structure. */ public static final Integer STRUCTURE_TREE_ELEMENT = 37; /** writing mode trait */ public static final Integer WRITING_MODE = 38; /** inline progression direction trait */ public static final Integer INLINE_PROGRESSION_DIRECTION = 39; /** block progression direction trait */ public static final Integer BLOCK_PROGRESSION_DIRECTION = 40; /** column progression direction trait */ public static final Integer COLUMN_PROGRESSION_DIRECTION = 41; /** shift direction trait */ public static final Integer SHIFT_DIRECTION = 42; /** Maximum value used by trait keys */ public static final int MAX_TRAIT_KEY = 42; private static final TraitInfo[] TRAIT_INFO = new TraitInfo[MAX_TRAIT_KEY + 1]; private static class TraitInfo { private String name; private Class clazz; // Class of trait data public TraitInfo(String name, Class clazz) { this.name = name; this.clazz = clazz; } public String getName() { return this.name; } public Class getClazz() { return this.clazz; } } private static void put(Integer key, TraitInfo info) { TRAIT_INFO[key] = info; } static { // Create a hashmap mapping trait code to name for external representation //put(ID_LINK, new TraitInfo("id-link", String.class)); put(STRUCTURE_TREE_ELEMENT, new TraitInfo("structure-tree-element", String.class)); put(INTERNAL_LINK, new TraitInfo("internal-link", InternalLink.class)); put(EXTERNAL_LINK, new TraitInfo("external-link", ExternalLink.class)); put(FONT, new TraitInfo("font", FontTriplet.class)); put(FONT_SIZE, new TraitInfo("font-size", Integer.class)); put(COLOR, new TraitInfo("color", Color.class)); put(PROD_ID, new TraitInfo("prod-id", String.class)); put(BACKGROUND, new TraitInfo("background", Background.class)); put(UNDERLINE, new TraitInfo("underline-score", Boolean.class)); put(UNDERLINE_COLOR, new TraitInfo("underline-score-color", Color.class)); put(OVERLINE, new TraitInfo("overline-score", Boolean.class)); put(OVERLINE_COLOR, new TraitInfo("overline-score-color", Color.class)); put(LINETHROUGH, new TraitInfo("through-score", Boolean.class)); put(LINETHROUGH_COLOR, new TraitInfo("through-score-color", Color.class)); put(BLINK, new TraitInfo("blink", Boolean.class)); //put(OFFSET, new TraitInfo("offset", Integer.class)); //put(SHADOW, new TraitInfo("shadow", Integer.class)); put(BORDER_START, new TraitInfo("border-start", BorderProps.class)); put(BORDER_END, new TraitInfo("border-end", BorderProps.class)); put(BORDER_BEFORE, new TraitInfo("border-before", BorderProps.class)); put(BORDER_AFTER, new TraitInfo("border-after", BorderProps.class)); put(PADDING_START, new TraitInfo("padding-start", Integer.class)); put(PADDING_END, new TraitInfo("padding-end", Integer.class)); put(PADDING_BEFORE, new TraitInfo("padding-before", Integer.class)); put(PADDING_AFTER, new TraitInfo("padding-after", Integer.class)); put(SPACE_START, new TraitInfo("space-start", Integer.class)); put(SPACE_END, new TraitInfo("space-end", Integer.class)); //put(BREAK_BEFORE, new TraitInfo("break-before", Integer.class)); //put(BREAK_AFTER, new TraitInfo("break-after", Integer.class)); put(START_INDENT, new TraitInfo("start-indent", Integer.class)); put(END_INDENT, new TraitInfo("end-indent", Integer.class)); put(SPACE_BEFORE, new TraitInfo("space-before", Integer.class)); put(SPACE_AFTER, new TraitInfo("space-after", Integer.class)); put(IS_REFERENCE_AREA, new TraitInfo("is-reference-area", Boolean.class)); put(IS_VIEWPORT_AREA, new TraitInfo("is-viewport-area", Boolean.class)); put(WRITING_MODE, new TraitInfo("writing-mode", WritingMode.class)); put(INLINE_PROGRESSION_DIRECTION, new TraitInfo("inline-progression-direction", Direction.class)); put(BLOCK_PROGRESSION_DIRECTION, new TraitInfo("block-progression-direction", Direction.class)); put(SHIFT_DIRECTION, new TraitInfo("shift-direction", Direction.class)); } /** * Get the trait name for a trait code. * * @param traitCode the trait code to get the name for * @return the trait name */ public static String getTraitName(Object traitCode) { return TRAIT_INFO[(Integer)traitCode].getName(); } /** * Get the data storage class for the trait. * * @param traitCode the trait code to lookup * @return the class type for the trait */ public static Class getTraitClass(Object traitCode) { return TRAIT_INFO[(Integer)traitCode].getClazz(); } /** * Class for internal link traits. * Stores PageViewport key and producer ID */ public static class InternalLink implements Serializable { private static final long serialVersionUID = -8993505060996723039L; /** The unique key of the PageViewport. */ private String pvKey; /** The PROD_ID of the link target */ private String idRef; /** * Create an InternalLink to the given PageViewport and target ID * * @param pvKey the PageViewport key * @param idRef the target ID */ public InternalLink(String pvKey, String idRef) { setPVKey(pvKey); setIDRef(idRef); } /** * Create an InternalLink based on the given XML attribute value. * This is typically called when data are read from an XML area tree. * * @param attrValue attribute value to be parsed by InternalLink.parseXMLAttribute */ public InternalLink(String attrValue) { String[] values = parseXMLAttribute(attrValue); setPVKey(values[0]); setIDRef(values[1]); } /** * Sets the key of the targeted PageViewport. * * @param pvKey the PageViewport key */ public void setPVKey(String pvKey) { this.pvKey = pvKey; } /** * Returns the key of the targeted PageViewport. * * @return the PageViewport key */ public String getPVKey() { return pvKey; } /** * Sets the target ID. * * @param idRef the target ID */ public void setIDRef(String idRef) { this.idRef = idRef; } /** * Returns the target ID. * * @return the target ID */ public String getIDRef() { return idRef; } /** * Returns the attribute value for this object as * used in the area tree XML. * * @return a string of the type "(thisPVKey,thisIDRef)" */ public String xmlAttribute() { return makeXMLAttribute(pvKey, idRef); } /** * Returns the XML attribute value for the given PV key and ID ref. * This value is used in the area tree XML. * * @param pvKey the PageViewport key of the link target * @param idRef the ID of the link target * @return a string of the type "(thisPVKey,thisIDRef)" */ public static String makeXMLAttribute(String pvKey, String idRef) { return "(" + (pvKey == null ? "" : pvKey) + "," + (idRef == null ? "" : idRef) + ")"; } /** * Parses XML attribute value from the area tree into * PageViewport key + IDRef strings. If the attribute value is * formatted like "(s1,s2)", then s1 and s2 are considered to be * the PV key and the IDRef, respectively. * Otherwise, the entire string is the PV key and the IDRef is null. * * @param attrValue the atribute value (PV key and possibly IDRef) * @return a 2-String array containing the PV key and the IDRef. * Both may be null. */ public static String[] parseXMLAttribute(String attrValue) { String[] result = {null, null}; if (attrValue != null) { int len = attrValue.length(); if (len >= 2 && attrValue.charAt(0) == '(' && attrValue.charAt(len - 1) == ')' && attrValue.indexOf(',') != -1) { String value = attrValue.substring(1, len - 1); // remove brackets int delimIndex = value.indexOf(','); result[0] = value.substring(0, delimIndex).trim(); // PV key result[1] = value.substring(delimIndex + 1, value.length()).trim(); // IDRef } else { // PV key only, e.g. from old area tree XML: result[0] = attrValue; } } return result; } /** * Return the human-friendly string for debugging. * {@inheritDoc} */ @Override public String toString() { StringBuffer sb = new StringBuffer(); sb.append("pvKey=").append(pvKey); sb.append(",idRef=").append(idRef); return sb.toString(); } } /** * External Link trait structure */ public static class ExternalLink implements Serializable { private static final long serialVersionUID = -3720707599232620946L; private String destination; private boolean newWindow; /** * Constructs an ExternalLink object with the given destination * * @param destination target of the link * @param newWindow true if the target should be opened in a new window */ public ExternalLink(String destination, boolean newWindow) { this.destination = destination; this.newWindow = newWindow; } /** * Create an ExternalLink from a trait value/attribute value in the * area tree * @param traitValue the value to use (should match the result of {@link #toString()} * @return an ExternalLink instance corresponding to the given value */ protected static ExternalLink makeFromTraitValue(String traitValue) { String dest = null; boolean newWindow = false; String[] values = traitValue.split(","); for (int i = 0, c = values.length; i < c; i++) { String v = values[i]; if (v.startsWith("dest=")) { dest = v.substring(5); } else if (v.startsWith("newWindow=")) { newWindow = Boolean.valueOf(v.substring(10)); } else { throw new IllegalArgumentException( "Malformed trait value for Trait.ExternalLink: " + traitValue); } } return new ExternalLink(dest, newWindow); } /** * Get the target/destination of the link * @return the destination of the link */ public String getDestination() { return this.destination; } /** * Check if the target has to be displayed in a new window * @return true if the target has to be displayed in a new window */ public boolean newWindow() { return this.newWindow; } /** * Return a String representation of the object. * @return a String of the form * "org.apache.fop.area.Trait.ExternalLink[dest=someURL,newWindow=false]" */ @Override public String toString() { StringBuffer sb = new StringBuffer(64); sb.append("newWindow=").append(newWindow); sb.append(",dest=").append(this.destination); return sb.toString(); } } /** * Background trait structure. * Used for storing back trait information which are related. */ public static class Background implements Serializable { private static final long serialVersionUID = 8452078676273242870L; /** The background color if any. */ private Color color = null; /** The background image url if any. */ private String url = null; /** The background image if any. */ private ImageInfo imageInfo = null; /** Background repeat enum for images. */ private int repeat; /** Background horizontal offset for images. */ private int horiz; /** Background vertical offset for images. */ private int vertical; /** * Returns the background color. * @return background color, null if n/a */ public Color getColor() { return color; } /** * Returns the horizontal offset for images. * @return the horizontal offset */ public int getHoriz() { return horiz; } /** * Returns the image repetition behaviour for images. * @return the image repetition behaviour */ public int getRepeat() { return repeat; } /** * Returns the URL to the background image * @return URL to the background image, null if n/a */ public String getURL() { return url; } /** * Returns the ImageInfo object representing the background image * @return the background image, null if n/a */ public ImageInfo getImageInfo() { return imageInfo; } /** * Returns the vertical offset for images. * @return the vertical offset */ public int getVertical() { return vertical; } /** * Sets the color. * @param color The color to set */ public void setColor(Color color) { this.color = color; } /** * Sets the horizontal offset. * @param horiz The horizontal offset to set */ public void setHoriz(int horiz) { this.horiz = horiz; } /** * Sets the image repetition behaviour for images. * @param repeat The image repetition behaviour to set */ public void setRepeat(int repeat) { this.repeat = repeat; } /** * Sets the image repetition behaviour for images. * @param repeat The image repetition behaviour to set */ public void setRepeat(String repeat) { setRepeat(getConstantForRepeat(repeat)); } /** * Sets the URL to the background image. * @param url The URL to set */ public void setURL(String url) { this.url = url; } /** * Sets the ImageInfo of the image to use as the background image. * @param info The background image's info object */ public void setImageInfo(ImageInfo info) { this.imageInfo = info; } /** * Sets the vertical offset for images. * @param vertical The vertical offset to set */ public void setVertical(int vertical) { this.vertical = vertical; } private String getRepeatString() { switch (getRepeat()) { case EN_REPEAT: return "repeat"; case EN_REPEATX: return "repeat-x"; case EN_REPEATY: return "repeat-y"; case EN_NOREPEAT: return "no-repeat"; default: throw new IllegalStateException("Illegal repeat style: " + getRepeat()); } } private static int getConstantForRepeat(String repeat) { if ("repeat".equalsIgnoreCase(repeat)) { return EN_REPEAT; } else if ("repeat-x".equalsIgnoreCase(repeat)) { return EN_REPEATX; } else if ("repeat-y".equalsIgnoreCase(repeat)) { return EN_REPEATY; } else if ("no-repeat".equalsIgnoreCase(repeat)) { return EN_NOREPEAT; } else { throw new IllegalStateException("Illegal repeat style: " + repeat); } } /** * Return the string for debugging. * {@inheritDoc} */ @Override public String toString() { StringBuffer sb = new StringBuffer(); if (color != null) { sb.append("color=").append(ColorUtil.colorToString(color)); } if (url != null) { if (color != null) { sb.append(","); } sb.append("url=").append(url); sb.append(",repeat=").append(getRepeatString()); sb.append(",horiz=").append(horiz); sb.append(",vertical=").append(vertical); } return sb.toString(); } } }