All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.batik.transcoder.SVGAbstractTranscoder Maven / Gradle / Ivy

/*

   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.transcoder;

import java.awt.Dimension;
import java.awt.geom.AffineTransform;
import java.awt.geom.Dimension2D;
import java.awt.geom.Rectangle2D;
import java.util.LinkedList;
import java.util.List;
import java.util.StringTokenizer;

import org.apache.batik.anim.dom.SAXSVGDocumentFactory;
import org.apache.batik.anim.dom.SVGDOMImplementation;
import org.apache.batik.anim.dom.SVGOMDocument;
import org.apache.batik.bridge.BaseScriptingEnvironment;
import org.apache.batik.bridge.BridgeContext;
import org.apache.batik.bridge.BridgeException;
import org.apache.batik.bridge.DefaultScriptSecurity;
import org.apache.batik.bridge.GVTBuilder;
import org.apache.batik.bridge.NoLoadScriptSecurity;
import org.apache.batik.bridge.RelaxedScriptSecurity;
import org.apache.batik.bridge.SVGUtilities;
import org.apache.batik.bridge.ScriptSecurity;
import org.apache.batik.bridge.UserAgent;
import org.apache.batik.bridge.UserAgentAdapter;
import org.apache.batik.bridge.ViewBox;
import org.apache.batik.bridge.svg12.SVG12BridgeContext;
import org.apache.batik.dom.util.DOMUtilities;
import org.apache.batik.dom.util.DocumentFactory;
import org.apache.batik.gvt.CanvasGraphicsNode;
import org.apache.batik.gvt.CompositeGraphicsNode;
import org.apache.batik.gvt.GraphicsNode;
import org.apache.batik.transcoder.keys.BooleanKey;
import org.apache.batik.transcoder.keys.FloatKey;
import org.apache.batik.transcoder.keys.LengthKey;
import org.apache.batik.transcoder.keys.Rectangle2DKey;
import org.apache.batik.transcoder.keys.StringKey;
import org.apache.batik.util.ParsedURL;
import org.apache.batik.util.SVGConstants;
import org.w3c.dom.DOMImplementation;
import org.w3c.dom.Document;
import org.w3c.dom.svg.SVGSVGElement;

/**
 * This class may be the base class of all transcoders which take an
 * SVG document as input and which need to build a DOM tree. The
 * SVGAbstractTranscoder uses several different hints that
 * guide it's behaviour:
* *
    *
  • KEY_WIDTH, KEY_HEIGHT can be used to specify how to scale the * SVG image
  • *
* * @author Thierry Kormann * @version $Id: SVGAbstractTranscoder.java 1805902 2017-08-23 14:54:08Z ssteiner $ */ public abstract class SVGAbstractTranscoder extends XMLAbstractTranscoder { /** * Value used as a default for the default font-family hint */ public static final String DEFAULT_DEFAULT_FONT_FAMILY = "Arial, Helvetica, sans-serif"; /** * Current area of interest. */ protected Rectangle2D curAOI; /** * Transform needed to render the current area of interest */ protected AffineTransform curTxf; /** * Current GVT Tree, i.e., the GVT tree representing the page * being printed currently. */ protected GraphicsNode root; /** * Current bridge context */ protected BridgeContext ctx; /** * Current gvt builder */ protected GVTBuilder builder; /** * Image's width and height (init to 400x400). */ protected float width=400, height=400; /** The user agent dedicated to an SVG Transcoder. */ protected UserAgent userAgent; protected SVGAbstractTranscoder() { userAgent = createUserAgent(); hints.put(KEY_DOCUMENT_ELEMENT_NAMESPACE_URI, SVGConstants.SVG_NAMESPACE_URI); hints.put(KEY_DOCUMENT_ELEMENT, SVGConstants.SVG_SVG_TAG); hints.put(KEY_DOM_IMPLEMENTATION, SVGDOMImplementation.getDOMImplementation()); hints.put(KEY_MEDIA, "screen"); hints.put(KEY_DEFAULT_FONT_FAMILY, DEFAULT_DEFAULT_FONT_FAMILY); hints.put(KEY_EXECUTE_ONLOAD, Boolean.FALSE); hints.put(KEY_ALLOWED_SCRIPT_TYPES, DEFAULT_ALLOWED_SCRIPT_TYPES); } protected UserAgent createUserAgent() { return new SVGAbstractTranscoderUserAgent(); } /** * Creates a DocumentFactory that is used to create an SVG DOM * tree. The specified DOM Implementation is ignored and the Batik * SVG DOM Implementation is automatically used. * * @param domImpl the DOM Implementation (not used) * @param parserClassname the XML parser classname */ protected DocumentFactory createDocumentFactory(DOMImplementation domImpl, String parserClassname) { return new SAXSVGDocumentFactory(parserClassname); } public void transcode(TranscoderInput input, TranscoderOutput output) throws TranscoderException { super.transcode(input, output); if (ctx != null) ctx.dispose(); } /** * Transcodes the specified Document as an image in the specified output. * * @param document the document to transcode * @param uri the uri of the document or null if any * @param output the ouput where to transcode * @exception TranscoderException if an error occured while transcoding */ protected void transcode(Document document, String uri, TranscoderOutput output) throws TranscoderException { if ((document != null) && !(document.getImplementation() instanceof SVGDOMImplementation)) { DOMImplementation impl; impl = (DOMImplementation)hints.get(KEY_DOM_IMPLEMENTATION); // impl = SVGDOMImplementation.getDOMImplementation(); document = DOMUtilities.deepCloneDocument(document, impl); if (uri != null) { ParsedURL url = new ParsedURL(uri); ((SVGOMDocument)document).setParsedURL(url); } } if (hints.containsKey(KEY_WIDTH)) width = (Float) hints.get(KEY_WIDTH); if (hints.containsKey(KEY_HEIGHT)) height = (Float) hints.get(KEY_HEIGHT); SVGOMDocument svgDoc = (SVGOMDocument)document; SVGSVGElement root = svgDoc.getRootElement(); ctx = createBridgeContext(svgDoc); // build the GVT tree builder = new GVTBuilder(); // flag that indicates if the document is dynamic boolean isDynamic = hints.containsKey(KEY_EXECUTE_ONLOAD) && (Boolean) hints.get(KEY_EXECUTE_ONLOAD); GraphicsNode gvtRoot; try { if (isDynamic) ctx.setDynamicState(BridgeContext.DYNAMIC); gvtRoot = builder.build(ctx, svgDoc); // dispatch an 'onload' event if needed if (ctx.isDynamic()) { BaseScriptingEnvironment se; se = new BaseScriptingEnvironment(ctx); se.loadScripts(); se.dispatchSVGLoadEvent(); if (hints.containsKey(KEY_SNAPSHOT_TIME)) { float t = (Float) hints.get(KEY_SNAPSHOT_TIME); ctx.getAnimationEngine().setCurrentTime(t); } else if (ctx.isSVG12()) { float t = SVGUtilities.convertSnapshotTime(root, null); ctx.getAnimationEngine().setCurrentTime(t); } } } catch (BridgeException ex) { throw new TranscoderException(ex); } // get the 'width' and 'height' attributes of the SVG document float docWidth = (float)ctx.getDocumentSize().getWidth(); float docHeight = (float)ctx.getDocumentSize().getHeight(); setImageSize(docWidth, docHeight); // compute the preserveAspectRatio matrix AffineTransform Px; // take the AOI into account if any if (hints.containsKey(KEY_AOI)) { Rectangle2D aoi = (Rectangle2D)hints.get(KEY_AOI); // transform the AOI into the image's coordinate system Px = new AffineTransform(); double sx = width / aoi.getWidth(); double sy = height / aoi.getHeight(); double scale = Math.min(sx,sy); Px.scale(scale, scale); double tx = -aoi.getX() + (width/scale - aoi.getWidth())/2; double ty = -aoi.getY() + (height/scale -aoi.getHeight())/2; Px.translate(tx, ty); // take the AOI transformation matrix into account // we apply first the preserveAspectRatio matrix curAOI = aoi; } else { String ref = new ParsedURL(uri).getRef(); // XXX Update this to use the animated value of 'viewBox' and // 'preserveAspectRatio'. String viewBox = root.getAttributeNS (null, SVGConstants.SVG_VIEW_BOX_ATTRIBUTE); if ((ref != null) && (ref.length() != 0)) { Px = ViewBox.getViewTransform(ref, root, width, height, ctx); } else if ((viewBox != null) && (viewBox.length() != 0)) { String aspectRatio = root.getAttributeNS (null, SVGConstants.SVG_PRESERVE_ASPECT_RATIO_ATTRIBUTE); Px = ViewBox.getPreserveAspectRatioTransform (root, viewBox, aspectRatio, width, height, ctx); } else { // no viewBox has been specified, create a scale transform float xscale, yscale; xscale = width/docWidth; yscale = height/docHeight; float scale = Math.min(xscale,yscale); Px = AffineTransform.getScaleInstance(scale, scale); } curAOI = new Rectangle2D.Float(0, 0, width, height); } CanvasGraphicsNode cgn = getCanvasGraphicsNode(gvtRoot); if (cgn != null) { cgn.setViewingTransform(Px); curTxf = new AffineTransform(); } else { curTxf = Px; } this.root = gvtRoot; } protected CanvasGraphicsNode getCanvasGraphicsNode(GraphicsNode gn) { if (!(gn instanceof CompositeGraphicsNode)) return null; CompositeGraphicsNode cgn = (CompositeGraphicsNode)gn; List children = cgn.getChildren(); if (children.size() == 0) return null; gn = (GraphicsNode)children.get(0); if (!(gn instanceof CanvasGraphicsNode)) return null; return (CanvasGraphicsNode)gn; } /** * Factory method for constructing an configuring a * BridgeContext so subclasses can insert new/modified * bridges in the context. * @param doc the SVG document to create the BridgeContext for * @return the newly instantiated BridgeContext */ protected BridgeContext createBridgeContext(SVGOMDocument doc) { return createBridgeContext(doc.isSVG12() ? "1.2" : "1.x"); } /** * Creates the default SVG 1.0/1.1 BridgeContext. Subclass this method to provide * customized bridges. This method is provided for historical reasons. New applications * should use {@link #createBridgeContext(String)} instead. * @return the newly instantiated BridgeContext * @see #createBridgeContext(String) */ protected BridgeContext createBridgeContext() { return createBridgeContext("1.x"); } /** * Creates the BridgeContext. Subclass this method to provide customized bridges. For example, * Apache FOP uses this method to register special bridges for optimized text painting. * @param svgVersion the SVG version in use (ex. "1.0", "1.x" or "1.2") * @return the newly instantiated BridgeContext */ protected BridgeContext createBridgeContext(String svgVersion) { if ("1.2".equals(svgVersion)) { return new SVG12BridgeContext(userAgent); } else { return new BridgeContext(userAgent); } } /** * Sets document size according to the hints. * Global variables width and height are modified. * * @param docWidth Width of the document. * @param docHeight Height of the document. */ protected void setImageSize(float docWidth, float docHeight) { // Compute the image's width and height according the hints float imgWidth = -1; if (hints.containsKey(KEY_WIDTH)) { imgWidth = (Float) hints.get(KEY_WIDTH); } float imgHeight = -1; if (hints.containsKey(KEY_HEIGHT)) { imgHeight = (Float) hints.get(KEY_HEIGHT); } if (imgWidth > 0 && imgHeight > 0) { width = imgWidth; height = imgHeight; } else if (imgHeight > 0) { width = (docWidth * imgHeight) / docHeight; height = imgHeight; } else if (imgWidth > 0) { width = imgWidth; height = (docHeight * imgWidth) / docWidth; } else { width = docWidth; height = docHeight; } // Limit image size according to the maximuxm size hints. float imgMaxWidth = -1; if (hints.containsKey(KEY_MAX_WIDTH)) { imgMaxWidth = (Float) hints.get(KEY_MAX_WIDTH); } float imgMaxHeight = -1; if (hints.containsKey(KEY_MAX_HEIGHT)) { imgMaxHeight = (Float) hints.get(KEY_MAX_HEIGHT); } if ((imgMaxHeight > 0) && (height > imgMaxHeight)) { width = (docWidth * imgMaxHeight) / docHeight; height = imgMaxHeight; } if ((imgMaxWidth > 0) && (width > imgMaxWidth)) { width = imgMaxWidth; height = (docHeight * imgMaxWidth) / docWidth; } } // -------------------------------------------------------------------- // Keys definition // -------------------------------------------------------------------- /** * The image width key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_WIDTH
Value:float
Default:The width of the topmost svg element
Required:No
Description:Specify the width of the image to create.
*/ public static final TranscodingHints.Key KEY_WIDTH = new LengthKey(); /** * The image height key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_HEIGHT
Value:Float
Default:The height of the topmost svg element
Required:No
Description:Specify the height of the image to create.
*/ public static final TranscodingHints.Key KEY_HEIGHT = new LengthKey(); /** * The maximum width of the image key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_MAX_WIDTH
Value:Float
Default:The width of the topmost svg element
Required:No
Description:Specify the maximum width of the image to create. * The value will set the maximum width of the image even when a * bigger width is specified in a document or set with KEY_WIDTH.
*/ public static final TranscodingHints.Key KEY_MAX_WIDTH = new LengthKey(); /** * The maximux height of the image key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_MAX_HEIGHT
Value:Float
Default:The height of the topmost svg element
Required:No
Description:Specify the maximum height of the image to create. * The value will set the maximum height of the image even when * bigger height is specified in a document or set with KEY_HEIGHT.
*/ public static final TranscodingHints.Key KEY_MAX_HEIGHT = new LengthKey(); /** * The area of interest key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_AOI
Value:Rectangle2D
Default:The document's size
Required:No
Description:Specify the area of interest to render. The * rectangle coordinates must be specified in pixels and in the * document coordinates system.
*/ public static final TranscodingHints.Key KEY_AOI = new Rectangle2DKey(); /** * The language key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_LANGUAGE
Value:String
Default:"en"
Required:No
Description:Specify the preferred language of the document.
*/ public static final TranscodingHints.Key KEY_LANGUAGE = new StringKey(); /** * The media key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_MEDIA
Value:String
Default:"screen"
Required:No
Description:Specify the media to use with CSS.
*/ public static final TranscodingHints.Key KEY_MEDIA = new StringKey(); /** * The default font-family key. * * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_DEFAULT_FONT_FAMILY
Value:String
Default:"Arial, Helvetica, sans-serif"
Required:No
Description:Controls the default * value used by the CSS engine for the font-family property * when that property is unspecified.
*/ public static final TranscodingHints.Key KEY_DEFAULT_FONT_FAMILY = new StringKey(); /** * The alternate stylesheet key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_ALTERNATE_STYLESHEET
Value:String
Default:null
Required:No
Description:Specify the alternate style sheet title.
*/ public static final TranscodingHints.Key KEY_ALTERNATE_STYLESHEET = new StringKey(); /** * The user stylesheet URI key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_USER_STYLESHEET_URI
Value:String
Default:null
Required:No
Description:Specify the user style sheet.
*/ public static final TranscodingHints.Key KEY_USER_STYLESHEET_URI = new StringKey(); /** * The number of millimeters in each pixel key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_PIXEL_UNIT_TO_MILLIMETER
Value:Float
Default:0.264583
Required:No
Description:Specify the size of a px CSS unit in millimeters.
*/ public static final TranscodingHints.Key KEY_PIXEL_UNIT_TO_MILLIMETER = new FloatKey(); /** * The pixel to millimeter conversion factor key. * @deprecated As of Batik Version 1.5b3 * @see #KEY_PIXEL_UNIT_TO_MILLIMETER * * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_PIXEL_TO_MM
Value:Float
Default:0.264583
Required:No
Description:Specify the size of a px CSS unit in millimeters.
*/ public static final TranscodingHints.Key KEY_PIXEL_TO_MM = KEY_PIXEL_UNIT_TO_MILLIMETER; /** * The 'onload' execution key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_EXECUTE_ONLOAD
Value:Boolean
Default:false
Required:No
Description:Specify if scripts added on the 'onload' event * attribute must be invoked.
*/ public static final TranscodingHints.Key KEY_EXECUTE_ONLOAD = new BooleanKey(); /** * The snapshot time key. * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_SNAPSHOT_TIME
Value:Float
Default:0
Required:No
Description:Specifies the document time to seek to before * rasterization. Only applies if {@link #KEY_EXECUTE_ONLOAD} is * set to true.
*/ public static final TranscodingHints.Key KEY_SNAPSHOT_TIME = new FloatKey(); /** * The set of supported script languages (i.e., the set of possible * values for the <script> tag's type attribute). * * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_ALLOWED_SCRIPT_TYPES
Value:String (Comma separated values)
Default:text/ecmascript, application/java-archive
Required:No
Description:Specifies the allowed values for the type attribute * in the <script> element. This is a comma separated list. The * special value '*' means that all script types are allowed.
*/ public static final TranscodingHints.Key KEY_ALLOWED_SCRIPT_TYPES = new StringKey(); /** * Default value for the KEY_ALLOWED_SCRIPT_TYPES key */ public static final String DEFAULT_ALLOWED_SCRIPT_TYPES = SVGConstants.SVG_SCRIPT_TYPE_ECMASCRIPT + ", " + SVGConstants.SVG_SCRIPT_TYPE_APPLICATION_ECMASCRIPT + ", " + SVGConstants.SVG_SCRIPT_TYPE_JAVASCRIPT + ", " + SVGConstants.SVG_SCRIPT_TYPE_APPLICATION_JAVASCRIPT + ", " + SVGConstants.SVG_SCRIPT_TYPE_JAVA; /** * Controls whether or not scripts can only be loaded from the * same location as the document which references them. * * * * * * * * * * * * * * * * * * * * * * *
Key:KEY_CONSTRAIN_SCRIPT_ORIGIN
Value:Boolean
Default:true
Required:No
Description:When set to true, script elements referencing * files from a different origin (server) than the document containing * the script element will not be loaded. When set to true, script elements * may reference script files from any origin.
*/ public static final TranscodingHints.Key KEY_CONSTRAIN_SCRIPT_ORIGIN = new BooleanKey(); /** * A user agent implementation for PrintTranscoder. */ protected class SVGAbstractTranscoderUserAgent extends UserAgentAdapter { /** * Vector containing the allowed script types */ protected List scripts; public SVGAbstractTranscoderUserAgent() { addStdFeatures(); } /** * Return the rendering transform. */ public AffineTransform getTransform() { return SVGAbstractTranscoder.this.curTxf; } /** * Return the rendering transform. */ public void setTransform(AffineTransform at) { SVGAbstractTranscoder.this.curTxf = at; } /** * Returns the default size of this user agent (400x400). */ public Dimension2D getViewportSize() { return new Dimension((int)SVGAbstractTranscoder.this.width, (int)SVGAbstractTranscoder.this.height); } /** * Displays the specified error message using the ErrorHandler. */ public void displayError(String message) { try { SVGAbstractTranscoder.this.handler.error (new TranscoderException(message)); } catch (TranscoderException ex) { throw new RuntimeException( ex.getMessage() ); } } /** * Displays the specified error using the ErrorHandler. */ public void displayError(Exception e) { try { e.printStackTrace(); SVGAbstractTranscoder.this.handler.error (new TranscoderException(e)); } catch (TranscoderException ex) { throw new RuntimeException( ex.getMessage() ); } } /** * Displays the specified message using the ErrorHandler. */ public void displayMessage(String message) { try { SVGAbstractTranscoder.this.handler.warning (new TranscoderException(message)); } catch (TranscoderException ex) { throw new RuntimeException( ex.getMessage() ); } } /** * Returns the pixel to millimeter conversion factor specified in the * TranscodingHints or 0.26458333 if not specified. */ public float getPixelUnitToMillimeter() { Object obj = SVGAbstractTranscoder.this.hints.get (KEY_PIXEL_UNIT_TO_MILLIMETER); if (obj != null) { return (Float) obj; } return super.getPixelUnitToMillimeter(); } /** * Returns the user language specified in the * TranscodingHints or "en" (english) if any. */ public String getLanguages() { if (SVGAbstractTranscoder.this.hints.containsKey(KEY_LANGUAGE)) { return (String)SVGAbstractTranscoder.this.hints.get (KEY_LANGUAGE); } return super.getLanguages(); } /** * Returns this user agent's CSS media. */ public String getMedia() { String s = (String)hints.get(KEY_MEDIA); if (s != null) return s; return super.getMedia(); } /** * Returns the default font family. */ public String getDefaultFontFamily() { String s = (String)hints.get(KEY_DEFAULT_FONT_FAMILY); if (s != null) return s; return super.getDefaultFontFamily(); } /** * Returns this user agent's alternate style-sheet title. */ public String getAlternateStyleSheet() { String s = (String)hints.get(KEY_ALTERNATE_STYLESHEET); if (s != null) return s; return super.getAlternateStyleSheet(); } /** * Returns the user stylesheet specified in the * TranscodingHints or null if any. */ public String getUserStyleSheetURI() { String s = (String)SVGAbstractTranscoder.this.hints.get (KEY_USER_STYLESHEET_URI); if (s != null) return s; return super.getUserStyleSheetURI(); } /** * Returns the XML parser to use from the TranscodingHints. */ public String getXMLParserClassName() { String s = (String)SVGAbstractTranscoder.this.hints.get (KEY_XML_PARSER_CLASSNAME); if (s != null) return s; return super.getXMLParserClassName(); } /** * Returns true if the XML parser must be in validation mode, false * otherwise. */ public boolean isXMLParserValidating() { Boolean b = (Boolean)SVGAbstractTranscoder.this.hints.get (KEY_XML_PARSER_VALIDATING); if (b != null) return b; return super.isXMLParserValidating(); } /** * Returns the security settings for the given script * type, script url and document url * * @param scriptType type of script, as found in the * type attribute of the <script> element. * @param scriptPURL url for the script, as defined in * the script's xlink:href attribute. If that * attribute was empty, then this parameter should * be null * @param docPURL url for the document into which the * script was found. */ public ScriptSecurity getScriptSecurity(String scriptType, ParsedURL scriptPURL, ParsedURL docPURL){ if (scripts == null){ computeAllowedScripts(); } if (!scripts.contains(scriptType)) { return new NoLoadScriptSecurity(scriptType); } boolean constrainOrigin = true; if (SVGAbstractTranscoder.this.hints.containsKey (KEY_CONSTRAIN_SCRIPT_ORIGIN)) { constrainOrigin = (Boolean) SVGAbstractTranscoder.this.hints.get (KEY_CONSTRAIN_SCRIPT_ORIGIN); } if (constrainOrigin) { return new DefaultScriptSecurity (scriptType,scriptPURL,docPURL); } else { return new RelaxedScriptSecurity (scriptType,scriptPURL,docPURL); } } /** * Helper method. Builds a Vector containing the allowed * values for the <script> element's type attribute. */ protected void computeAllowedScripts(){ scripts = new LinkedList(); if (!SVGAbstractTranscoder.this.hints.containsKey (KEY_ALLOWED_SCRIPT_TYPES)) { return; } String allowedScripts = (String)SVGAbstractTranscoder.this.hints.get (KEY_ALLOWED_SCRIPT_TYPES); StringTokenizer st = new StringTokenizer(allowedScripts, ","); while (st.hasMoreTokens()) { scripts.add(st.nextToken()); } } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy