/*

   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.

 */
package org.apache.batik.bridge;

import java.awt.Color;
import java.awt.geom.Rectangle2D;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

import org.apache.batik.anim.dom.SVGOMDocument;
import org.apache.batik.dom.util.XLinkSupport;
import org.apache.batik.ext.awt.image.PadMode;
import org.apache.batik.ext.awt.image.renderable.Filter;
import org.apache.batik.ext.awt.image.renderable.FilterChainRable;
import org.apache.batik.ext.awt.image.renderable.FilterChainRable8Bit;
import org.apache.batik.ext.awt.image.renderable.FloodRable8Bit;
import org.apache.batik.ext.awt.image.renderable.PadRable8Bit;
import org.apache.batik.gvt.GraphicsNode;
import org.apache.batik.util.ParsedURL;
import org.w3c.dom.Element;
import org.w3c.dom.Node;

Bridge class for the <filter> element.
Author:Thierry Kormann
Version:$Id: SVGFilterElementBridge.java 1805408 2017-08-18 12:21:52Z ssteiner $
/** * Bridge class for the &lt;filter&gt; element. * * @author <a href="mailto:tkormann@apache.org">Thierry Kormann</a> * @version $Id: SVGFilterElementBridge.java 1805408 2017-08-18 12:21:52Z ssteiner $ */
public class SVGFilterElementBridge extends AnimatableGenericSVGBridge implements FilterBridge, ErrorConstants {
Transparent black color.
/** * Transparent black color. */
protected static final Color TRANSPARENT_BLACK = new Color(0, true);
Constructs a new bridge for the <filter> element.
/** * Constructs a new bridge for the &lt;filter&gt; element. */
public SVGFilterElementBridge() {}
Returns 'filter'.
/** * Returns 'filter'. */
public String getLocalName() { return SVG_FILTER_TAG; }
Creates a Filter according to the specified parameters.
Params:
  • ctx – the bridge context to use
  • filterElement – the element that defines the filter
  • filteredElement – the element that references the filter element
  • filteredNode – the graphics node to filter
/** * Creates a <code>Filter</code> according to the specified parameters. * * @param ctx the bridge context to use * @param filterElement the element that defines the filter * @param filteredElement the element that references the filter element * @param filteredNode the graphics node to filter */
public Filter createFilter(BridgeContext ctx, Element filterElement, Element filteredElement, GraphicsNode filteredNode) { // get filter chain region Rectangle2D filterRegion = SVGUtilities.convertFilterChainRegion (filterElement, filteredElement, filteredNode, ctx); if (filterRegion == null) { return null; } // make the initial source as a RenderableImage Filter sourceGraphic = filteredNode.getGraphicsNodeRable(true); // Pad out to filterRegion sourceGraphic = new PadRable8Bit(sourceGraphic, filterRegion, PadMode.ZERO_PAD); // build a FilterChainRable8Bit FilterChainRable filterChain = new FilterChainRable8Bit(sourceGraphic, filterRegion); // 'filterRes' attribute - default is implementation specific float [] filterRes = SVGUtilities.convertFilterRes(filterElement, ctx); filterChain.setFilterResolutionX((int)filterRes[0]); filterChain.setFilterResolutionY((int)filterRes[1]); // Create a map for filter nodes to advertise themselves as // named source Map filterNodeMap = new HashMap(11); filterNodeMap.put(SVG_SOURCE_GRAPHIC_VALUE, sourceGraphic); Filter in = buildFilterPrimitives(filterElement, filterRegion, filteredElement, filteredNode, sourceGraphic, filterNodeMap, ctx); if (in == null) { // error in one of the primitives, disable the filter return null; } else if (in == sourceGraphic) { // no filter primitive found, so output transparent black in = createEmptyFilter(filterElement, filterRegion, filteredElement, filteredNode, ctx); } filterChain.setSource(in); return filterChain; }
Creates a new returns a new filter that fills its output with transparent black. This is used when a <filter> element has no filter primitive children.
/** * Creates a new returns a new filter that fills its output with * transparent black. This is used when a &lt;filter&gt; element * has no filter primitive children. */
protected static Filter createEmptyFilter(Element filterElement, Rectangle2D filterRegion, Element filteredElement, GraphicsNode filteredNode, BridgeContext ctx) { Rectangle2D primitiveRegion = SVGUtilities.convertFilterPrimitiveRegion(null, filterElement, filteredElement, filteredNode, filterRegion, filterRegion, ctx); return new FloodRable8Bit(primitiveRegion, TRANSPARENT_BLACK); }
Builds the filter primitives of filter chain of the specified filter element and returns the last filter primitive created. Filter primitives can be children of the filter or defined on one of its 'ancestor' (linked with the xlink:href attribute).
Params:
  • filterElement – the filter element
  • filterRegion – the filter chain region
  • filteredElement – the filtered element
  • filteredNode – the filtered node
  • in – the input Filter
  • filterNodeMap – the map used by named filter primitives
  • ctx – the bridge context
Returns:the last filter primitive created
/** * Builds the filter primitives of filter chain of the specified * filter element and returns the last filter primitive * created. Filter primitives can be children of the filter or * defined on one of its 'ancestor' (linked with the xlink:href * attribute). * * @param filterElement the filter element * @param filterRegion the filter chain region * @param filteredElement the filtered element * @param filteredNode the filtered node * @param in the input Filter * @param filterNodeMap the map used by named filter primitives * @param ctx the bridge context * @return the last filter primitive created */
protected static Filter buildFilterPrimitives(Element filterElement, Rectangle2D filterRegion, Element filteredElement, GraphicsNode filteredNode, Filter in, Map filterNodeMap, BridgeContext ctx) { List refs = new LinkedList(); for (;;) { Filter newIn = buildLocalFilterPrimitives(filterElement, filterRegion, filteredElement, filteredNode, in, filterNodeMap, ctx); if (newIn != in) { return newIn; // filter primitives found, exit } String uri = XLinkSupport.getXLinkHref(filterElement); if (uri.length() == 0) { return in; // no xlink:href found, exit } // check if there is circular dependencies SVGOMDocument doc = (SVGOMDocument)filterElement.getOwnerDocument(); ParsedURL url = new ParsedURL(doc.getURLObject(), uri); if (refs.contains(url)) { throw new BridgeException(ctx, filterElement, ERR_XLINK_HREF_CIRCULAR_DEPENDENCIES, new Object[] {uri}); } refs.add(url); filterElement = ctx.getReferencedElement(filterElement, uri); } }
Builds the filter primitives of filter chain of the specified filter element and returns the last filter primitive created or 'in' if no filter primitive has been specified.
Params:
  • filterElement – the filter element
  • filterRegion – the filter chain region
  • filteredElement – the filtered element
  • filteredNode – the filtered node
  • in – the input Filter
  • filterNodeMap – the map used by named filter primitives
  • ctx – the bridge context
Returns:the last filter primitive created or 'in'
/** * Builds the filter primitives of filter chain of the specified * filter element and returns the last filter primitive * created or 'in' if no filter primitive has been specified. * * @param filterElement the filter element * @param filterRegion the filter chain region * @param filteredElement the filtered element * @param filteredNode the filtered node * @param in the input Filter * @param filterNodeMap the map used by named filter primitives * @param ctx the bridge context * @return the last filter primitive created or 'in' */
protected static Filter buildLocalFilterPrimitives(Element filterElement, Rectangle2D filterRegion, Element filteredElement, GraphicsNode filteredNode, Filter in, Map filterNodeMap, BridgeContext ctx) { for (Node n = filterElement.getFirstChild(); n != null; n = n.getNextSibling()) { if (n.getNodeType() != Node.ELEMENT_NODE) { continue; // skip node that is not an Element } Element e = (Element)n; Bridge bridge = ctx.getBridge(e); if (bridge == null || !(bridge instanceof FilterPrimitiveBridge)) { continue; } FilterPrimitiveBridge filterBridge = (FilterPrimitiveBridge)bridge; Filter filterNode = filterBridge.createFilter(ctx, e, filteredElement, filteredNode, in, filterRegion, filterNodeMap); if (filterNode == null) { return null; // disable the filter if a primitive is null } else { in = filterNode; } } return in; } }