📄 abstractxsltview.java
字号:
/*
* Copyright 2002-2005 the original author or authors.
*
* 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.
*/
package org.springframework.web.servlet.view.xslt;
import java.io.BufferedOutputStream;
import java.io.IOException;
import java.net.URL;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.Map;
import java.util.Properties;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.xml.transform.ErrorListener;
import javax.xml.transform.OutputKeys;
import javax.xml.transform.Result;
import javax.xml.transform.Source;
import javax.xml.transform.Templates;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerConfigurationException;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.URIResolver;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import javax.xml.transform.stream.StreamSource;
import org.w3c.dom.Node;
import org.springframework.context.ApplicationContextException;
import org.springframework.core.io.Resource;
import org.springframework.util.xml.SimpleTransformErrorListener;
import org.springframework.web.servlet.view.AbstractView;
import org.springframework.web.util.NestedServletException;
/**
* Convenient superclass for views rendered using an XSLT stylesheet.
* Subclasses must <b>either</b> provide the XML W3C Document or Node to
* transform by overriding <code>createDomNode()</code>, <b>or</b> provide the
* <code>Source</code> to transform by overriding <code>createXsltSource()</code>.
*
* <p>Note that <code>createXsltSource()</code> is the preferred method which all
* new subclasses should override since Spring 1.2. <code>createDomNode()</code>
* has been deprecated and may be removed in a future version.
*
* <p>Subclasses do not need to concern themselves with XSLT other than providing
* a valid stylesheet location.
*
* <p>Properties:
* <ul>
* <li>stylesheetLocation: a Spring <code>Resource</code> pointing to the
* XSLT stylesheet
* <li>root: name of the root element, defaults to "DocRoot"
* <li>uriResolver: URIResolver used in the transform
* <li>errorListener (optional): ErrorListener implementation for custom
* handling of warnings and errors during TransformerFactory operations
* <li>indent (optional, default=true): whether additional whitespace
* may be added when outputting the result true
* <li>cache (optional, default=true): debug setting only
* </ul>
*
* <p>Note that setting "cache" to "false" will cause the template objects
* to be reloaded for each rendering. This is useful during development,
* but will seriously affect performance in production and isn't thread-safe.
*
* @author Rod Johnson
* @author Darren Davison
* @author Juergen Hoeller
*/
public abstract class AbstractXsltView extends AbstractView {
public static final String DEFAULT_ROOT = "DocRoot";
private Resource stylesheetLocation;
private String root = DEFAULT_ROOT;
private boolean useSingleModelNameAsRoot = true;
private URIResolver uriResolver;
private ErrorListener errorListener = new SimpleTransformErrorListener(logger);
private boolean indent = true;
private Properties outputProperties;
private boolean cache = true;
private TransformerFactory transformerFactory;
private Templates templates;
/**
* Set the location of the XSLT stylesheet.
* @param stylesheetLocation the location of the XSLT stylesheet
* @see org.springframework.context.ApplicationContext#getResource
*/
public void setStylesheetLocation(Resource stylesheetLocation) {
this.stylesheetLocation = stylesheetLocation;
// Re-cache templates if transformer factory already initialized.
if (this.transformerFactory != null) {
cacheTemplates();
}
}
/**
* Document root element name. Default is "DocRoot".
* Only used if we're not passed a single Node as model.
* @param root the document root element name
* @see #DEFAULT_ROOT
*/
public void setRoot(String root) {
this.root = root;
}
/**
* Set whether to use the name of a given single model object
* as document root element name.
* <p>Default is "true": If you pass in a model with a single object
* named "myElement", then the document root will be named "myElement"
* as well. Set this flag to "false" if you want to pass in a single
* model object while still using the root element name configured
* through the "root" property.
* @see #setRoot
*/
public void setUseSingleModelNameAsRoot(boolean useSingleModelNameAsRoot) {
this.useSingleModelNameAsRoot = useSingleModelNameAsRoot;
}
/**
* Set the URIResolver used in the transform. The URIResolver
* handles calls to the XSLT document() function.
* This method can be used by subclasses or as a bean property.
* @param uriResolver URIResolver to set. No URIResolver
* will be set if this is null (this is the default).
*/
public void setUriResolver(URIResolver uriResolver) {
this.uriResolver = uriResolver;
}
/**
* Set an implementation of the <code>javax.xml.transform.ErrorListener</code>
* interface for custom handling of transformation errors and warnings.
* <p>If not set, a default SimpleTransformErrorListener is used that simply
* logs warnings using the logger instance of the view class,
* and rethrows errors to discontinue the XML transformation.
* @see org.springframework.util.xml.SimpleTransformErrorListener
*/
public void setErrorListener(ErrorListener errorListener) {
this.errorListener = errorListener;
}
/**
* Set whether the XSLT transformer may add additional whitespace when
* outputting the result tree. Default is on; turn this off to not
* specify an "indent" key, leaving the choice up to the stylesheet.
* @see javax.xml.transform.OutputKeys#INDENT
*/
public void setIndent(boolean indent) {
this.indent = indent;
}
/**
* Set arbitrary transformer output properties to be applied to the stylesheet.
* <p>Any values specified here will override defaults that this view sets
* programmatically.
* @param outputProperties output properties to apply to the transformation process
* @see javax.xml.transform.Transformer#setOutputProperty
*/
public void setOutputProperties(Properties outputProperties) {
this.outputProperties = outputProperties;
}
/**
* Set whether to activate the cache. Default is on.
*/
public void setCache(boolean cache) {
this.cache = cache;
}
/**
* Here we load our template, as we need the ApplicationContext to do it.
*/
protected final void initApplicationContext() throws ApplicationContextException {
this.transformerFactory = TransformerFactory.newInstance();
this.transformerFactory.setErrorListener(this.errorListener);
if (this.uriResolver != null) {
if (logger.isInfoEnabled()) {
logger.info("Using custom URIResolver [" + this.uriResolver + "] in XSLT view with name '" +
getBeanName() + "'");
}
this.transformerFactory.setURIResolver(this.uriResolver);
}
if (logger.isDebugEnabled()) {
logger.debug("URL in view is " + this.stylesheetLocation);
}
cacheTemplates();
}
private synchronized void cacheTemplates() throws ApplicationContextException {
if (this.stylesheetLocation != null) {
try {
this.templates = this.transformerFactory.newTemplates(getStylesheetSource(this.stylesheetLocation));
if (logger.isDebugEnabled()) {
logger.debug("Loaded templates [" + this.templates + "] in XSLT view '" + getBeanName() + "'");
}
}
catch (TransformerConfigurationException ex) {
throw new ApplicationContextException("Can't load stylesheet from " + this.stylesheetLocation +
" in XSLT view '" + getBeanName() + "'", ex);
}
}
}
/**
* Load the stylesheet. Subclasses can override this.
*/
protected Source getStylesheetSource(Resource stylesheetLocation) throws ApplicationContextException {
if (logger.isDebugEnabled()) {
logger.debug("Loading XSLT stylesheet from " + stylesheetLocation);
}
try {
URL url = stylesheetLocation.getURL();
String urlPath = url.toString();
String systemId = urlPath.substring(0, urlPath.lastIndexOf('/') + 1);
return new StreamSource(url.openStream(), systemId);
}
catch (IOException ex) {
throw new ApplicationContextException("Can't load XSLT stylesheet from " + stylesheetLocation, ex);
}
}
protected final void renderMergedOutputModel(
Map model, HttpServletRequest request, HttpServletResponse response) throws Exception {
if (!this.cache) {
logger.warn("DEBUG SETTING: NOT THREADSAFE AND WILL IMPAIR PERFORMANCE: template will be refreshed");
cacheTemplates();
}
if (this.templates == null) {
if (this.transformerFactory == null) {
throw new ServletException("XLST view is incorrectly configured. Templates AND TransformerFactory are null");
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -