📄 builtin.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.core;
import freemarker.template.*;
import java.io.UnsupportedEncodingException;
import java.text.DateFormat;
import java.text.NumberFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import freemarker.template.utility.ClassUtil;
import freemarker.template.utility.StringUtil;
import freemarker.core.NodeBuiltins.*;
import freemarker.core.NumericalBuiltins.*;
import freemarker.core.SequenceBuiltins.*;
import freemarker.core.StringBuiltins.*;
/**
* The ? operator used to get the
* functionality of built-in unary operators
* @author <a href="mailto:jon@revusky.com">Jonathan Revusky</a>
*/
abstract class BuiltIn extends Expression implements Cloneable {
Expression target;
String key;
static final HashMap builtins = new HashMap();
static {
// These are the only ones we have now.
// We throw a parse exception if it's not one of these.
builtins.put("ancestors", new ancestorsBI());
builtins.put("byte", new byteBI());
builtins.put("c", new cBI());
builtins.put("cap_first", new cap_firstBI());
builtins.put("capitalize", new capitalizeBI());
builtins.put("children", new childrenBI());
builtins.put("chop_linebreak", new chop_linebreakBI());
builtins.put("contains", new containsBI());
builtins.put("date", new dateBI(TemplateDateModel.DATE));
builtins.put("datetime", new dateBI(TemplateDateModel.DATETIME));
builtins.put("default", new defaultBI());
builtins.put("double", new doubleBI());
builtins.put("ends_with", new ends_withBI());
builtins.put("eval", new evalBI());
builtins.put("exists", new existsBI());
builtins.put("first", new firstBI());
builtins.put("float", new floatBI());
builtins.put("chunk", new chunkBI());
builtins.put("has_content", new has_contentBI());
builtins.put("html", new htmlBI());
builtins.put("if_exists", new if_existsBI());
builtins.put("index_of", new index_ofBI());
builtins.put("int", new intBI());
builtins.put("interpret", new Interpret());
builtins.put("is_boolean", new is_booleanBI());
builtins.put("is_collection", new is_collectionBI());
builtins.put("is_date", new is_dateBI());
builtins.put("is_directive", new is_directiveBI());
builtins.put("is_enumerable", new is_enumerableBI());
builtins.put("is_hash_ex", new is_hash_exBI());
builtins.put("is_hash", new is_hashBI());
builtins.put("is_indexable", new is_indexableBI());
builtins.put("is_macro", new is_macroBI());
builtins.put("is_method", new is_methodBI());
builtins.put("is_node", new is_nodeBI());
builtins.put("is_number", new is_numberBI());
builtins.put("is_sequence", new is_sequenceBI());
builtins.put("is_string", new is_stringBI());
builtins.put("is_transform", new is_transformBI());
builtins.put("j_string", new j_stringBI());
builtins.put("js_string", new js_stringBI());
builtins.put("keys", new keysBI());
builtins.put("last_index_of", new last_index_ofBI());
builtins.put("last", new lastBI());
builtins.put("left_pad", new left_padBI());
builtins.put("length", new lengthBI());
builtins.put("long", new longBI());
builtins.put("lower_case", new lower_caseBI());
builtins.put("namespace", new namespaceBI());
builtins.put("new", new NewBI());
builtins.put("node_name", new node_nameBI());
builtins.put("node_namespace", new node_namespaceBI());
builtins.put("node_type", new node_typeBI());
builtins.put("number", new numberBI());
builtins.put("parent", new parentBI());
builtins.put("replace", new replaceBI());
builtins.put("reverse", new reverseBI());
builtins.put("right_pad", new right_padBI());
builtins.put("root", new rootBI());
builtins.put("rtf", new rtfBI());
builtins.put("seq_contains", new seq_containsBI());
builtins.put("seq_index_of", new seq_index_ofBI(1));
builtins.put("seq_last_index_of", new seq_index_ofBI(-1));
builtins.put("short", new shortBI());
builtins.put("size", new sizeBI());
builtins.put("sort_by", new sort_byBI());
builtins.put("sort", new sortBI());
builtins.put("split", new splitBI());
builtins.put("starts_with", new starts_withBI());
builtins.put("string", new stringBI());
builtins.put("substring", new substringBI());
builtins.put("time", new dateBI(TemplateDateModel.TIME));
builtins.put("trim", new trimBI());
builtins.put("uncap_first", new uncap_firstBI());
builtins.put("upper_case", new upper_caseBI());
builtins.put("url", new urlBI());
builtins.put("values", new valuesBI());
builtins.put("web_safe", builtins.get("html")); // deprecated; use ?html instead
builtins.put("word_list", new word_listBI());
builtins.put("xml", new xmlBI());
try {
Class.forName("java.util.regex.Pattern");
builtins.put("matches", instantiate("freemarker.core.RegexBuiltins$matchesBI"));
builtins.put("groups", instantiate("freemarker.core.RegexBuiltins$groupsBI"));
builtins.put("replace", instantiate("freemarker.core.RegexBuiltins$replace_reBI"));
builtins.put("split", instantiate("freemarker.core.RegexBuiltins$split_reBI"));
} catch (Exception e) {}
}
private static Object instantiate(String className) throws Exception
{
return ClassUtil.forName(className).newInstance();
}
static BuiltIn newBuiltIn(Expression target, String key, Token tok, String templateName) throws ParseException {
BuiltIn bi = (BuiltIn) builtins.get(key);
if (bi == null) {
String locationInfo = "Error on line " + tok.beginLine + ", column " + tok.beginColumn + ", in template " + templateName + "\n";
StringBuffer buf = new StringBuffer("Found " + key + ", expecting one of: ");
for (Iterator it= builtins.keySet().iterator(); it.hasNext();) {
if (it.hasNext()) {
buf.append(" ");
} else {
buf.append( " or ");
}
buf.append(it.next());
if (it.hasNext()) {
buf.append(", ");
}
}
throw new ParseException(locationInfo + buf, target);
}
try {
bi = (BuiltIn) bi.clone();
}
catch (CloneNotSupportedException e) {
throw new InternalError();
}
bi.target = target;
bi.key = key;
return bi;
}
public String getCanonicalForm() {
return target.getCanonicalForm() + "?" + key;
}
boolean isLiteral() {
return false; // be on the safe side.
}
Expression _deepClone(String name, Expression subst) {
try {
BuiltIn clone = (BuiltIn)clone();
clone.target = target.deepClone(name, subst);
return clone;
}
catch (CloneNotSupportedException e) {
throw new InternalError();
}
}
static class lengthBI extends BuiltIn {
TemplateModel _getAsTemplateModel(Environment env)
throws TemplateException
{
return new SimpleNumber(target.getStringValue(env).length());
}
}
static class dateBI extends BuiltIn {
private final int dateType;
dateBI(int dateType) {
this.dateType = dateType;
}
TemplateModel _getAsTemplateModel(Environment env)
throws TemplateException
{
TemplateModel model = target.getAsTemplateModel(env);
if (model instanceof TemplateDateModel) {
TemplateDateModel dmodel = (TemplateDateModel)model;
int dtype = dmodel.getDateType();
// Any date model can be coerced into its own type
if(dateType == dtype) {
return model;
}
// unknown and datetime can be coerced into any date type
if(dtype == TemplateDateModel.UNKNOWN || dtype == TemplateDateModel.DATETIME) {
return new SimpleDate(dmodel.getAsDate(), dateType);
}
throw new TemplateException(
"Cannot convert " + TemplateDateModel.TYPE_NAMES.get(dtype)
+ " into " + TemplateDateModel.TYPE_NAMES.get(dateType), env);
}
// Otherwise, interpret as a string and attempt
// to parse it into a date.
String s = target.getStringValue(env);
return new DateParser(s, env);
}
private class DateParser
implements
TemplateDateModel,
TemplateMethodModel,
TemplateHashModel
{
private final String text;
private final Environment env;
private final DateFormat defaultFormat;
private Date cachedValue;
DateParser(String text, Environment env)
throws
TemplateModelException
{
this.text = text;
this.env = env;
this.defaultFormat = env.getDateFormatObject(dateType);
}
public Date getAsDate() throws TemplateModelException {
if(cachedValue == null) {
cachedValue = parse(defaultFormat);
}
return cachedValue;
}
public int getDateType() {
return dateType;
}
public TemplateModel get(String pattern) throws TemplateModelException {
return new SimpleDate(
parse(env.getDateFormatObject(dateType, pattern)),
dateType);
}
public Object exec(List arguments)
throws TemplateModelException {
if (arguments.size() != 1) {
throw new TemplateModelException(
"string?" + key + "(...) requires exactly 1 argument.");
}
return get((String) arguments.get(0));
}
public boolean isEmpty()
{
return false;
}
private Date parse(DateFormat df)
throws
TemplateModelException
{
try {
return df.parse(text);
}
catch(java.text.ParseException e) {
String mess = "Error: " + getStartLocation()
+ "\nExpecting a date here, found: " + text;
throw new TemplateModelException(mess);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -