📄 tagtransformmodel.java
字号:
/*
* Copyright (c) 2003 The Visigoth Software Society. All rights
* reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* 3. The end-user documentation included with the redistribution, if
* any, must include the following acknowledgement:
* "This product includes software developed by the
* Visigoth Software Society (http://www.visigoths.org/)."
* Alternately, this acknowledgement may appear in the software itself,
* if and wherever such third-party acknowledgements normally appear.
*
* 4. Neither the name "FreeMarker", "Visigoth", nor any of the names of the
* project contributors may be used to endorse or promote products derived
* from this software without prior written permission. For written
* permission, please contact visigoths@visigoths.org.
*
* 5. Products derived from this software may not be called "FreeMarker" or "Visigoth"
* nor may "FreeMarker" or "Visigoth" appear in their names
* without prior written permission of the Visigoth Software Society.
*
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE VISIGOTH SOFTWARE SOCIETY OR
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Visigoth Software Society. For more
* information on the Visigoth Software Society, please see
* http://www.visigoths.org/
*/
package freemarker.ext.jsp;
import freemarker.ext.beans.BeansWrapper;
import freemarker.log.Logger;
import freemarker.template.ObjectWrapper;
import freemarker.template.TemplateModel;
import freemarker.template.TemplateModelException;
import freemarker.template.TemplateTransformModel;
import freemarker.template.TransformControl;
import freemarker.template.utility.SecurityUtilities;
import freemarker.template.utility.StringUtil;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.tagext.BodyContent;
import javax.servlet.jsp.tagext.BodyTag;
import javax.servlet.jsp.tagext.IterationTag;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TryCatchFinally;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.io.CharArrayReader;
import java.io.CharArrayWriter;
import java.io.IOException;
import java.io.Reader;
import java.io.Writer;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
/**
* @version $Id: TagTransformModel.java,v 1.17.2.2 2006/07/08 14:45:34 ddekany Exp $
* @author Attila Szegedi
*/
class TagTransformModel implements TemplateTransformModel
{
private static final char[] NEWLINE = SecurityUtilities.getSystemProperty("line.separator").toCharArray();
private static final Logger logger = Logger.getLogger("freemarker.servlet");
private final Class tagClass;
private final Method dynaSetter;
private final Map propertySetters = new HashMap();
private final boolean isBodyTag;
private final boolean isIterationTag;
private final boolean isTryCatchFinally;
public TagTransformModel(Class tagClass)
throws
IntrospectionException
{
if(!Tag.class.isAssignableFrom(tagClass)) {
throw new IllegalArgumentException(tagClass.getName() + " does not implement the " + Tag.class.getName() + " interface.");
}
isIterationTag = IterationTag.class.isAssignableFrom(tagClass);
isBodyTag = isIterationTag && BodyTag.class.isAssignableFrom(tagClass);
isTryCatchFinally = TryCatchFinally.class.isAssignableFrom(tagClass);
this.tagClass = tagClass;
BeanInfo bi = Introspector.getBeanInfo(tagClass);
PropertyDescriptor[] pda = bi.getPropertyDescriptors();
for (int i = 0; i < pda.length; i++)
{
PropertyDescriptor pd = pda[i];
Method m = pd.getWriteMethod();
if(m != null)
{
propertySetters.put(pd.getName(), m);
}
}
// Check to see if the tag implements the JSP2.0 DynamicAttributes
// interface, to allow setting of arbitrary attributes
Method dynaSetter;
try {
dynaSetter = tagClass.getMethod("setDynamicAttribute",
new Class[] {String.class, String.class, Object.class});
}
catch (NoSuchMethodException nsme) {
dynaSetter = null;
}
this.dynaSetter = dynaSetter;
}
public Writer getWriter(Writer out, Map args) throws TemplateModelException
{
try {
Tag tag = getTagInstance();
FreeMarkerPageContext pageContext = PageContextFactory.getCurrentPageContext();
Tag parentTag = pageContext.peekTopTag();
tag.setParent(parentTag);
tag.setPageContext(pageContext);
setupTag(tag, args, pageContext.getObjectWrapper());
// If the parent of this writer is not a JspWriter itself, use
// a little Writer-to-JspWriter adapter...
boolean usesAdapter;
if(out instanceof JspWriter) {
// This is just a sanity check. If it were JDK 1.4-only,
// we'd use an assert.
if(out != pageContext.getOut()) {
throw new TemplateModelException(
"out != pageContext.getOut(). Out is " +
out + " pageContext.getOut() is " +
pageContext.getOut());
}
usesAdapter = false;
}
else {
out = new JspWriterAdapter(out);
pageContext.pushWriter((JspWriter)out);
usesAdapter = true;
}
JspWriter w = new TagWriter(out, tag, pageContext, usesAdapter);
pageContext.pushTopTag(tag);
pageContext.pushWriter(w);
return w;
}
catch(TemplateModelException e) {
throw e;
}
catch(Exception e) {
throw new TemplateModelException(e);
}
}
private Tag getTagInstance()
throws
IllegalAccessException,
InstantiationException
{
return (Tag)tagClass.newInstance();
}
private void setupTag(Object tag, Map args, ObjectWrapper wrapper)
throws
TemplateModelException,
InvocationTargetException,
IllegalAccessException
{
BeansWrapper bwrapper =
wrapper instanceof BeansWrapper
? (BeansWrapper)wrapper
: BeansWrapper.getDefaultInstance();
if(args != null && !args.isEmpty())
{
Object[] aarg = new Object[1];
for (Iterator iter = args.entrySet().iterator(); iter.hasNext();)
{
Map.Entry entry = (Map.Entry) iter.next();
aarg[0] = bwrapper.unwrap((TemplateModel)entry.getValue());
Method m = (Method) propertySetters.get(entry.getKey());
if (m == null) {
if (dynaSetter == null) {
throw new TemplateModelException("Unknown property "
+ StringUtil.jQuote(entry.getKey().toString())
+ " on instance of " + tagClass.getName());
}
else {
dynaSetter.invoke(tag, new Object[] {null, entry.getKey(), aarg[0]});
}
}
else {
BeansWrapper.coerceBigDecimals(m, aarg);
m.invoke(tag, aarg);
}
}
}
}
/**
* An implementation of BodyContent that buffers it's input to a char[].
*/
static class BodyContentImpl extends BodyContent {
private CharArrayWriter buf;
BodyContentImpl(JspWriter out, boolean buffer) {
super(out);
if (buffer) initBuffer();
}
void initBuffer() {
buf = new CharArrayWriter();
}
public void flush() throws IOException {
if(buf == null) {
getEnclosingWriter().flush();
}
}
public void clear() throws IOException {
if(buf != null) {
buf = new CharArrayWriter();
}
else {
throw new IOException("Can't clear");
}
}
public void clearBuffer() throws IOException {
if(buf != null) {
buf = new CharArrayWriter();
}
else {
throw new IOException("Can't clear");
}
}
public int getRemaining() {
return Integer.MAX_VALUE;
}
public void newLine() throws IOException {
write(NEWLINE);
}
public void close() throws IOException {
}
public void print(boolean arg0) throws IOException {
write(arg0 ? Boolean.TRUE.toString() : Boolean.FALSE.toString());
}
public void print(char arg0) throws IOException
{
write(arg0);
}
public void print(char[] arg0) throws IOException
{
write(arg0);
}
public void print(double arg0) throws IOException
{
write(Double.toString(arg0));
}
public void print(float arg0) throws IOException
{
write(Float.toString(arg0));
}
public void print(int arg0) throws IOException
{
write(Integer.toString(arg0));
}
public void print(long arg0) throws IOException
{
write(Long.toString(arg0));
}
public void print(Object arg0) throws IOException
{
write(arg0 == null ? "null" : arg0.toString());
}
public void print(String arg0) throws IOException
{
write(arg0);
}
public void println() throws IOException
{
newLine();
}
public void println(boolean arg0) throws IOException
{
print(arg0);
newLine();
}
public void println(char arg0) throws IOException
{
print(arg0);
newLine();
}
public void println(char[] arg0) throws IOException
{
print(arg0);
newLine();
}
public void println(double arg0) throws IOException
{
print(arg0);
newLine();
}
public void println(float arg0) throws IOException
{
print(arg0);
newLine();
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -