/* * Copyright 1999-2006 The Apache Software Foundation. * * Licensed 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 org.apache.fop.datatypes.ColorType; import org.apache.fop.fo.Constants; import org.apache.fop.fonts.FontTriplet; import org.apache.fop.image.FopImage; import org.apache.fop.traits.BorderProps; import java.io.Serializable; // 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 class Trait implements Serializable { /** * Id reference line, not resolved. * not sure if this is needed. */ //public static final Integer ID_LINK = new Integer(0); /** * Internal link trait. * This is resolved and provides a link to an internal area. */ public static final Integer INTERNAL_LINK = new Integer(1); //resolved /** * External link. A URL link to an external resource. */ public static final Integer EXTERNAL_LINK = new Integer(2); /** * The font triplet for the current font. */ public static final Integer FONT = new Integer(3); /** * Font size for the current font. */ public static final Integer FONT_SIZE = new Integer(4); /** * The current colour. */ public static final Integer COLOR = new Integer(7); /** * The ID of the FO that produced an area. */ public static final Integer PROD_ID = new Integer(8); /** * Background trait for an area. */ public static final Integer BACKGROUND = new Integer(9); /** * Underline trait used when rendering inline parent. */ public static final Integer UNDERLINE = new Integer(10); /** * Overline trait used when rendering inline parent. */ public static final Integer OVERLINE = new Integer(11); /** * Linethrough trait used when rendering inline parent. */ public static final Integer LINETHROUGH = new Integer(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 = new Integer(15); /** * The border end. */ public static final Integer BORDER_END = new Integer(16); /** * The border before. */ public static final Integer BORDER_BEFORE = new Integer(17); /** * The border after. */ public static final Integer BORDER_AFTER = new Integer(18); /** * The padding start. */ public static final Integer PADDING_START = new Integer(19); /** * The padding end. */ public static final Integer PADDING_END = new Integer(20); /** * The padding before. */ public static final Integer PADDING_BEFORE = new Integer(21); /** * The padding after. */ public static final Integer PADDING_AFTER = new Integer(22); /** * The space start. */ public static final Integer SPACE_START = new Integer(23); /** * The space end. */ public static final Integer SPACE_END = new Integer(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 = new Integer(27); /** * The end-indent trait. */ public static final Integer END_INDENT = new Integer(28); /** The space-before trait. */ public static final Integer SPACE_BEFORE = new Integer(29); /** The space-after trait. */ public static final Integer SPACE_AFTER = new Integer(30); /** The is-reference-area trait. */ public static final Integer IS_REFERENCE_AREA = new Integer(31); /** The is-viewport-area trait. */ public static final Integer IS_VIEWPORT_AREA = new Integer(32); /** Blinking trait used when rendering inline parent. */ public static final Integer BLINK = new Integer(33); /** Trait for color of underline decorations when rendering inline parent. */ public static final Integer UNDERLINE_COLOR = new Integer(34); /** Trait for color of overline decorations when rendering inline parent. */ public static final Integer OVERLINE_COLOR = new Integer(35); /** Trait for color of linethrough decorations when rendering inline parent. */ public static final Integer LINETHROUGH_COLOR = new Integer(36); /** Maximum value used by trait keys */ public static final int MAX_TRAIT_KEY = 36; 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.intValue()] = info; } static { // Create a hashmap mapping trait code to name for external representation //put(ID_LINK, new TraitInfo("id-link", String.class)); put(INTERNAL_LINK, new TraitInfo("internal-link", String.class)); put(EXTERNAL_LINK, new TraitInfo("external-link", String.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)); } /** * 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).intValue()].getName(); } /** * Get the trait code for a trait name. * * @param sTraitName the name of the trait to find * @return the trait code object */ /* public static Object getTraitCode(String sTraitName) { Iterator iter = TRAIT_INFO.entrySet().iterator(); while (iter.hasNext()) { Map.Entry entry = (Map.Entry) iter.next(); TraitInfo ti = (TraitInfo) entry.getValue(); if (ti != null && ti.getName().equals(sTraitName)) { return entry.getKey(); } } return null; }*/ /** * 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).intValue()].getClazz(); } /** * The type of trait for an area. */ private Object propType; /** * The data value of the trait. */ private Object data; /** * Create a new empty trait. */ public Trait() { this.propType = null; this.data = null; } /** * Create a trait with the value and type. * * @param propType the type of trait * @param data the data value */ public Trait(Object propType, Object data) { this.propType = propType; this.data = data; } /** * Returns the trait data value. * @return the trait data value */ public Object getData() { return this.data; } /** * Returns the property type. * @return the property type */ public Object getPropType() { return this.propType; } /** * Return the string for debugging. * @see java.lang.Object#toString() */ public String toString() { return data.toString(); } /** * Make a trait value. * * @param oCode trait code * @param sTraitValue trait value as String * @return the trait value as object */ /* public static Object makeTraitValue(Object oCode, String sTraitValue) { // Get the code from the name // See what type of object it is // Convert string value to an object of that type Class tclass = getTraitClass(oCode); if (tclass == null) { return null; } if (tclass.equals(String.class)) { return sTraitValue; } if (tclass.equals(Integer.class)) { return new Integer(sTraitValue); } // See if the class has a constructor from string or can read from a string try { Object o = tclass.newInstance(); //return o.fromString(sTraitValue); } catch (IllegalAccessException e1) { log.error("Can't create instance of " + tclass.getName()); return null; } catch (InstantiationException e2) { log.error("Can't create instance of " + tclass.getName()); return null; } return null; }*/ /** * Serializable ColorType implementation for the area tree. * @TODO Think about switching to java.awt.Color entirely! */ public static class Color implements ColorType, Serializable { private float red; private float green; private float blue; private float alpha; /** * Creates a new Color instance * @param r the red component * @param g the green component * @param b the blue component * @param a the alpha component */ public Color(float r, float g, float b, float a) { this.red = r; this.green = g; this.blue = b; this.alpha = a; } /** * Copy constructor * @param col the ColorType instance which shall be duplicated */ public Color(ColorType col) { this(col.getRed(), col.getGreen(), col.getBlue(), col.getAlpha()); } /** @see org.apache.fop.datatypes.ColorType#getRed() */ public float getRed() { return this.red; } /** @see org.apache.fop.datatypes.ColorType#getGreen() */ public float getGreen() { return this.green; } /** @see org.apache.fop.datatypes.ColorType#getBlue() */ public float getBlue() { return this.blue; } /** @see org.apache.fop.datatypes.ColorType#getAlpha() */ public float getAlpha() { return this.alpha; } /** @see org.apache.fop.datatypes.ColorType#getAWTColor() */ public java.awt.Color getAWTColor() { return new java.awt.Color(red, green, blue, alpha); } /** * Converts a given color to a serializable instance if necessary. * @param col the color * @return the serializable color value. */ public static ColorType makeSerializable(ColorType col) { if (col == null) { return null; } else if (col instanceof Serializable) { return col; } else { return new Color(col); } } /** @see java.lang.Object#hashCode() */ public int hashCode() { return toString().hashCode(); } /** @see java.lang.Object#equals(java.lang.Object) */ public boolean equals(Object obj) { if (obj == null) { return false; } else if (obj == this) { return true; } else { if (obj instanceof ColorType) { ColorType other = (ColorType)obj; return getRed() == other.getRed() && getGreen() == other.getGreen() && getBlue() == other.getBlue() && getAlpha() == other.getAlpha(); } } return false; } /** * Returns a Color represtation of a string of the format "#RRGGBB". * @param s the string * @return the Color value */ public static Color valueOf(String s) { if (s == null) { return null; } if (!s.startsWith("#")) { throw new IllegalArgumentException("Color must start with '#'"); } int r = Integer.parseInt(s.substring(1, 3), 16); int g = Integer.parseInt(s.substring(3, 5), 16); int b = Integer.parseInt(s.substring(5, 7), 16); int a = 255; if (s.length() > 7) { a = Integer.parseInt(s.substring(7, 9), 16); } return new Color(r / 255.0f, g / 255.0f, b / 255.0f, a / 255.0f); } /** @see java.lang.Object#toString() */ public String toString() { StringBuffer sbuf = new StringBuffer(8); sbuf.append('#'); String s = Integer.toHexString((int)(red * 255.0)); if (s.length() == 1) { sbuf.append('0'); } sbuf.append(s); s = Integer.toHexString((int)(green * 255.0)); if (s.length() == 1) { sbuf.append('0'); } sbuf.append(s); s = Integer.toHexString((int)(blue * 255.0)); if (s.length() == 1) { sbuf.append('0'); } sbuf.append(s); if (alpha != 1) { s = Integer.toHexString((int)(alpha * 255.0)); if (s.length() == 1) { sbuf.append('0'); } sbuf.append(s); } return sbuf.toString(); } } /** * Background trait structure. * Used for storing back trait information which are related. */ public static class Background implements Serializable { /** The background color if any. */ private ColorType color = null; /** The background image url if any. */ private String url = null; /** The background image if any. */ private FopImage fopimage = 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 ColorType 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 FopImage representing the background image * @return the background image, null if n/a */ public FopImage getFopImage() { return fopimage; } /** * 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(ColorType color) { this.color = Color.makeSerializable(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 FopImage to use as the background image. * @param fopimage The FopImage to use */ public void setFopImage(FopImage fopimage) { this.fopimage = fopimage; } /** * 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 Constants.EN_REPEAT: return "repeat"; case Constants.EN_REPEATX: return "repeat-x"; case Constants.EN_REPEATY: return "repeat-y"; case Constants.EN_NOREPEAT: return "no-repeat"; default: throw new IllegalStateException("Illegal repeat style: " + getRepeat()); } } private static int getConstantForRepeat(String repeat) { if ("repeat".equalsIgnoreCase(repeat)) { return Constants.EN_REPEAT; } else if ("repeat-x".equalsIgnoreCase(repeat)) { return Constants.EN_REPEATX; } else if ("repeat-y".equalsIgnoreCase(repeat)) { return Constants.EN_REPEATY; } else if ("no-repeat".equalsIgnoreCase(repeat)) { return Constants.EN_NOREPEAT; } else { throw new IllegalStateException("Illegal repeat style: " + repeat); } } /** * Return the string for debugging. * @see java.lang.Object#toString() */ public String toString() { StringBuffer sb = new StringBuffer(); sb.append("color=").append(color); if (url != null) { sb.append(",url=").append(url); sb.append(",repeat=").append(getRepeatString()); sb.append(",horiz=").append(horiz); sb.append(",vertical=").append(vertical); } return sb.toString(); } } }