/*
*
*
* Copyright 1990-2007 Sun Microsystems, Inc. All Rights Reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License version
* 2 only, as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License version 2 for more details (a copy is
* included at /legal/license.txt).
*
* You should have received a copy of the GNU General Public License
* version 2 along with this work; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA
*
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
* Clara, CA 95054 or visit www.sun.com if you need additional
* information or have any questions.
*/
package com.sun.perseus.model;
import com.sun.perseus.util.SVGConstants;
import org.w3c.dom.DOMException;
import org.w3c.dom.svg.SVGRGBColor;
import com.sun.perseus.j2d.RGB;
/**
* <code>Stop</code> class represents the <code><stop></code>
* SVG Tiny 1.2 element.
*
* @version $Id: Stop.java,v 1.5 2006/06/29 10:47:34 ln156897 Exp $
*/
public class Stop extends CompositeGraphicsNode {
/**
* offset is required on <stop>
*/
static final String[] REQUIRED_TRAITS
= {SVGConstants.SVG_OFFSET_ATTRIBUTE};
/**
* The default stop color.
*/
static final RGB DEFAULT_STOP_COLOR = RGB.black;
/**
* The default stop opacity.
*/
static final float DEFAULT_STOP_OPACITY = 1;
/**
* The stop's color
*/
RGB stopColor = DEFAULT_STOP_COLOR;
/**
* The stop's opacity
*/
float stopOpacity = DEFAULT_STOP_OPACITY;
/**
* The stop offset value.
*/
float offset;
/**
* Constructor.
*
* @param ownerDocument this element's owner <code>DocumentNode</code>
*/
public Stop(final DocumentNode ownerDocument) {
super(ownerDocument);
}
/**
* @return an adequate <code>ElementNodeProxy</code> for this node.
*/
ElementNodeProxy buildProxy() {
return new CompositeGraphicsNodeProxy(this);
}
/**
* @return the SVGConstants.SVG_STOP_TAG value
*/
public String getLocalName() {
return SVGConstants.SVG_STOP_TAG;
}
/**
* Used by <code>DocumentNode</code> to create a new instance from
* a prototype <code>Stop</code>.
*
* @param doc the <code>DocumentNode</code> for which a new node is
* should be created.
* @return a new <code>SolidColor</code> for the requested document.
*/
public ElementNode newInstance(final DocumentNode doc) {
return new Stop(doc);
}
/**
* Sets the stop-color property.
*
* @param newStopColor the new stop-color property
*/
public void setStopColor(final RGB newStopColor) {
if (stopColor.equals(newStopColor)) {
return;
}
stopColor = newStopColor;
updateGradient();
}
/**
* @return the current stopColor property.
*/
public RGB getStopColor() {
return stopColor;
}
/**
* Sets the stop-opacity property.
*
* @param newStopOpacity the new stop-opacity property
*/
public void setStopOpacity(final float newStopOpacity) {
if (newStopOpacity == stopOpacity) {
return;
}
stopOpacity = newStopOpacity;
updateGradient();
}
/**
* @return the current stopOpacity property.
*/
public float getStopOpacity() {
return stopOpacity;
}
/**
* Sets the stop offset
*
* @param newOffset the new offset
*/
public void setOffset(final float newOffset) {
if (newOffset == offset) {
return;
}
offset = newOffset;
updateGradient();
}
/**
* @return the stop's offset
*/
public float getOffset() {
return offset;
}
/**
* Should be called by the stop every time its parent gradient
* should recompute its state.
*/
void updateGradient() {
if (parent != null) {
if (parent instanceof GradientElement) {
((GradientElement) parent).onPaintChange();
}
}
}
/**
* @return the number of properties supported by this node
*/
public int getNumberOfProperties() {
return GraphicsNode.NUMBER_OF_PROPERTIES;
}
/**
* @param traitName the name of the trait which the element may support.
* @return true if this element supports the given trait in one of the
* trait accessor methods.
*/
boolean supportsTrait(final String traitName) {
if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE == traitName
||
SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == traitName
||
SVGConstants.SVG_OFFSET_ATTRIBUTE == traitName) {
return true;
} else {
return super.supportsTrait(traitName);
}
}
/**
* @return an array of traits that are required by this element.
*/
public String[] getRequiredTraits() {
return REQUIRED_TRAITS;
}
/**
* @param traitName the trait name.
*/
TraitAnim createTraitAnimImpl(final String traitName) {
if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == traitName
||
SVGConstants.SVG_OFFSET_ATTRIBUTE == traitName) {
return new FloatTraitAnim(this, traitName, TRAIT_TYPE_FLOAT);
} else if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE == traitName) {
return new FloatTraitAnim(this, traitName,
TRAIT_TYPE_SVG_RGB_COLOR);
} else {
return super.createTraitAnimImpl(traitName);
}
}
/**
* @param name the requested trait name (e.g., "ry")
* @return the trait's value, as a string.
*
* @throws DOMException with error code NOT_SUPPORTED_ERROR if the requested
* trait is not supported on this element or null.
* @throws DOMException with error code TYPE_MISMATCH_ERR if requested
* trait's computed value cannot be converted to a String (SVG Tiny only).
*/
public String getTraitImpl(final String name)
throws DOMException {
if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE == name) {
return stopColor.toString();
} else if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == name) {
return Float.toString(stopOpacity);
} else if (SVGConstants.SVG_OFFSET_ATTRIBUTE == name) {
return Float.toString(offset);
} else {
return super.getTraitImpl(name);
}
}
/**
* @param name the requested trait name (e.g., "y")
* @return the requested trait value
*
* @throws DOMException with error code NOT_SUPPORTED_ERROR if the requested
* trait is not supported on this element or null.
* @throws DOMException with error code TYPE_MISMATCH_ERR if requested
* trait's computed value cannot be converted to a float
* @throws SecurityException if the application does not have the necessary
* privilege rights to access this (SVG) content.
*/
float getFloatTraitImpl(final String name)
throws DOMException {
if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == name) {
return stopOpacity;
} else if (SVGConstants.SVG_OFFSET_ATTRIBUTE == name) {
return offset;
} else {
return super.getFloatTraitImpl(name);
}
}
/**
* @param name the requested trait's name.
* @return the requested trait's value, as an <code>SVGRGBColor</code>.
*
* @throws DOMException with error code NOT_SUPPORTED_ERROR if the requested
* trait is not supported on this element or null.
* @throws DOMException with error code TYPE_MISMATCH_ERR if requested
* trait's computed value cannot be converted to {@link
* org.w3c.dom.svg.SVGRGBColor SVGRGBColor}
* @throws SecurityException if the application does not have the necessary
* privilege rights to access this (SVG) content.
*/
SVGRGBColor getRGBColorTraitImpl(final String name)
throws DOMException {
// We use .equals here because name is not interned.
if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE.equals(name)) {
return stopColor;
} else {
return super.getRGBColorTraitImpl(name);
}
}
/**
* Set the trait value as float array.
*
* @param name the trait's name.
* @param value the trait's value.
*
* @throws DOMException with error code NOT_SUPPORTED_ERROR if the requested
* trait is not supported on this element.
* @throws DOMException with error code TYPE_MISMATCH_ERR if the requested
* trait's value cannot be specified as a float
* @throws DOMException with error code INVALID_ACCESS_ERR if the input
* value is an invalid value for the given trait.
*/
void setFloatArrayTrait(final String name, final float[][] value)
throws DOMException {
if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE == name) {
setStopColor(toRGB(name, value));
} else if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == name) {
if (value[0][0] < 0 || value[0][0] > 1) {
throw illegalTraitValue(name, Float.toString(value[0][0]));
}
setStopOpacity(value[0][0]);
} else if (SVGConstants.SVG_OFFSET_ATTRIBUTE == name) {
setOffset(value[0][0]);
} else {
super.setFloatArrayTrait(name, value);
}
}
/**
* Validates the input trait value.
*
* @param traitName the name of the trait to be validated.
* @param value the value to be validated
* @param reqNamespaceURI the namespace of the element requesting
* validation.
* @param reqLocalName the local name of the element requesting validation.
* @param reqTraitNamespace the namespace of the trait which has the values
* value on the requesting element.
* @param reqTraitName the name of the trait which has the values value on
* the requesting element.
* @throws DOMException with error code INVALID_ACCESS_ERR if the input
* value is incompatible with the given trait.
*/
public float[][] validateFloatArrayTrait(
final String traitName,
final String value,
final String reqNamespaceURI,
final String reqLocalName,
final String reqTraitNamespace,
final String reqTraitName) throws DOMException {
if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE == traitName) {
RGB color = parseColorTrait
(SVGConstants.SVG_COLOR_ATTRIBUTE, value);
if (color == null) {
throw illegalTraitValue(traitName, value);
}
return new float[][] {
{color.getRed(), color.getGreen(), color.getBlue()}
};
} else if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == traitName) {
return new float[][] {{parseFloatTrait(traitName, value)}};
} else if (SVGConstants.SVG_OFFSET_ATTRIBUTE == traitName) {
return new float[][] {{parseFloatTrait(traitName, value)}};
} else {
return super.validateFloatArrayTrait(traitName,
value,
reqNamespaceURI,
reqLocalName,
reqTraitNamespace,
reqTraitName);
}
}
/**
* CompositeGraphicsNode handles the graphics node traits.
* Other attributes are handled by the super class.
*
* Supported traits: stroke-width, stroke-miterlimit, stroke-dashoffset,
* fill-rule, stroke-linejoin, stroke-linecap, display, visibility,
* color, fill, stroke, fill-opacity, stroke-opacity, stroke-dasharray
*
* @param name the name of the trait to set.
* @param value the value of the trait to set.
*
* @throws DOMException with error code NOT_SUPPORTED_ERROR if the requested
* trait is not supported on this element or null.
* @throws DOMException with error code TYPE_MISMATCH_ERR if the requested
* trait's value cannot be specified as a String
* @throws DOMException with error code INVALID_ACCESS_ERR if the input
* value is an invalid value for the given trait or null.
* @throws DOMException with error code NO_MODIFICATION_ALLOWED_ERR: if
* attempt is made to change readonly trait.
*/
public void setTraitImpl(final String name, final String value)
throws DOMException {
if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == name) {
// ======================= stop-opacity ===================== //
if (SVGConstants.CSS_INHERIT_VALUE.equals(value)) {
setStopOpacity(DEFAULT_STOP_OPACITY);
} else {
setStopOpacity(parsePositiveFloatTrait(name, value));
}
} else if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE
.equals(name)) {
// ======================== stop-color ===================== //
if (SVGConstants.CSS_INHERIT_VALUE.equals(value)) {
setStopColor(DEFAULT_STOP_COLOR);
} else if (SVGConstants.CSS_CURRENTCOLOR_VALUE.equals(value)) {
setStopColor(getColor());
} else {
setStopColor(parseColorTrait
(SVGConstants.SVG_STOP_COLOR_ATTRIBUTE, value));
}
} else if (SVGConstants.SVG_OFFSET_ATTRIBUTE == name) {
setOffset(parseFloatTrait(name, value));
} else {
super.setTraitImpl(name, value);
}
}
/**
* Set the trait value as float.
*
* Supported float traits: stroke-width, stroke-miterlimit,
* stroke-dashoffset, fill-opacity, stroke-opacity.
*
* @param name the name of the trait to set.
* @param value the value of the trait to set.
*
* @throws DOMException with error code NOT_SUPPORTED_ERROR if the requested
* trait is not supported on this element.
* @throws DOMException with error code TYPE_MISMATCH_ERR if the requested
* trait's value cannot be specified as a float
* @throws DOMException with error code INVALID_ACCESS_ERR if the input
* value is an invalid value for the given trait.
* @throws SecurityException if the application does not have the necessary
* privilege rights to access this (SVG) content.
*/
public void setFloatTraitImpl(final String name, final float value)
throws DOMException {
if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == name) {
setStopOpacity(value);
} else if (SVGConstants.SVG_OFFSET_ATTRIBUTE == name) {
setOffset(value);
} else {
super.setFloatTraitImpl(name, value);
}
}
/**
* @param name the name of the trait to convert.
* @param value the float trait value to convert.
*/
String toStringTrait(final String name, final float[][] value) {
if (SVGConstants.SVG_STOP_OPACITY_ATTRIBUTE == name
||
SVGConstants.SVG_OFFSET_ATTRIBUTE == name) {
return Float.toString(value[0][0]);
} else if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE == name) {
return toRGBString(name, value);
} else {
return super.toStringTrait(name, value);
}
}
/**
* Set the trait value as {@link org.w3c.dom.svg.SVGRGBColor SVGRGBColor}.
*
* Supported color traits: color, fill, stroke
*
* @param name the name of the trait to set.
* @param value the value of the trait to set.
*
* @throws DOMException with error code NOT_SUPPORTED_ERROR if the requested
* trait is not supported on this element or null.
* @throws DOMException with error code TYPE_MISMATCH_ERR if the requested
* trait's value cannot be specified as an {@link
* org.w3c.dom.svg.SVGRGBColor SVGRGBColor}
* @throws DOMException with error code INVALID_ACCESS_ERR if the input
* value is null.
* @throws SecurityException if the application does not have the necessary
* privilege rights to access this (SVG) content.
*/
void setRGBColorTraitImpl(final String name, final SVGRGBColor color)
throws DOMException {
try {
if (SVGConstants.SVG_STOP_COLOR_ATTRIBUTE.equals(name)) {
setStopColor((RGB) color);
} else {
super.setRGBColorTraitImpl(name, color);
}
} catch (IllegalArgumentException iae) {
throw new DOMException(DOMException.INVALID_ACCESS_ERR,
iae.getMessage());
}
}
}
|