blob: 48d71f3c58de1d790cfab56635d4f640579b2151 [file] [log] [blame]
/*
* 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/
*/
/*
* 22 October 1999: This class added by Holger Arendt.
*/
package freemarker.core;
import java.io.IOException;
import java.io.Writer;
import java.util.ArrayList;
import java.util.List;
import freemarker.template.TemplateException;
import freemarker.template.TemplateMethodModel;
import freemarker.template.TemplateMethodModelEx;
import freemarker.template.TemplateModel;
import freemarker.template.utility.NullWriter;
/**
* A unary operator that calls a TemplateMethodModel. It associates with the
* <tt>Identifier</tt> or <tt>Dot</tt> to its left.
*/
final class MethodCall extends Expression {
private final Expression target;
private final ListLiteral arguments;
MethodCall(Expression target, ArrayList arguments) {
this(target, new ListLiteral(arguments));
}
private MethodCall(Expression target, ListLiteral arguments) {
this.target = target;
this.arguments = arguments;
}
TemplateModel _eval(Environment env) throws TemplateException
{
TemplateModel targetModel = target.eval(env);
if (targetModel instanceof TemplateMethodModel) {
TemplateMethodModel targetMethod = (TemplateMethodModel)targetModel;
List argumentStrings =
targetMethod instanceof TemplateMethodModelEx
? arguments.getModelList(env)
: arguments.getValueList(env);
Object result = targetMethod.exec(argumentStrings);
return env.getObjectWrapper().wrap(result);
}
else if (targetModel instanceof Macro) {
Macro func = (Macro) targetModel;
env.setLastReturnValue(null);
if (!func.isFunction) {
throw new _MiscTemplateException(env, "A macro cannot be called in an expression.");
}
Writer prevOut = env.getOut();
try {
env.setOut(NullWriter.INSTANCE);
env.visit(func, null, arguments.items, null, null);
} catch (IOException ioe) {
throw new InternalError("This should be impossible.");
} finally {
env.setOut(prevOut);
}
return env.getLastReturnValue();
}
else {
throw new UnexpectedTypeException(target, targetModel, "method", env);
}
}
public String getCanonicalForm() {
StringBuffer buf = new StringBuffer();
buf.append(target.getCanonicalForm());
buf.append("(");
String list = arguments.getCanonicalForm();
buf.append(list.substring(1, list.length() -1));
buf.append(")");
return buf.toString();
}
String getNodeTypeSymbol() {
return "...(...)";
}
TemplateModel getConstantValue() {
return null;
}
boolean isLiteral() {
return false;
}
protected Expression deepCloneWithIdentifierReplaced_inner(
String replacedIdentifier, Expression replacement, ReplacemenetState replacementState) {
return new MethodCall(
target.deepCloneWithIdentifierReplaced(replacedIdentifier, replacement, replacementState),
(ListLiteral)arguments.deepCloneWithIdentifierReplaced(replacedIdentifier, replacement, replacementState));
}
int getParameterCount() {
return 1 + arguments.items.size();
}
Object getParameterValue(int idx) {
if (idx == 0) {
return target;
} else if (idx < getParameterCount()) {
return arguments.items.get(idx - 1);
} else {
throw new IndexOutOfBoundsException();
}
}
ParameterRole getParameterRole(int idx) {
if (idx == 0) {
return ParameterRole.CALLEE;
} else if (idx < getParameterCount()) {
return ParameterRole.ARGUMENT_VALUE;
} else {
throw new IndexOutOfBoundsException();
}
}
}