| /* |
| * 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 java.math.BigDecimal; |
| import java.math.BigInteger; |
| import java.util.Date; |
| |
| import freemarker.template.SimpleDate; |
| import freemarker.template.SimpleNumber; |
| import freemarker.template.TemplateBooleanModel; |
| import freemarker.template.TemplateDateModel; |
| import freemarker.template.TemplateException; |
| import freemarker.template.TemplateModel; |
| import freemarker.template.TemplateModelException; |
| import freemarker.template.TemplateNumberModel; |
| import freemarker.template.utility.NumberUtil; |
| |
| /** |
| * A holder for builtins that operate exclusively on number left-hand value. |
| */ |
| class NumericalBuiltins { |
| |
| // Can't be instantiated |
| private NumericalBuiltins() { } |
| |
| private abstract static class NumberBuiltIn extends BuiltIn { |
| TemplateModel _eval(Environment env) |
| throws TemplateException |
| { |
| TemplateModel model = target.eval(env); |
| return calculateResult(target.modelToNumber(model, env), model); |
| } |
| |
| abstract TemplateModel calculateResult(Number num, TemplateModel model) |
| throws TemplateModelException; |
| } |
| |
| static class byteBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| if (num instanceof Byte) { |
| return model; |
| } |
| return new SimpleNumber(new Byte(num.byteValue())); |
| } |
| } |
| |
| static class shortBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| if (num instanceof Short) { |
| return model; |
| } |
| return new SimpleNumber(new Short(num.shortValue())); |
| } |
| } |
| |
| static class intBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| if (num instanceof Integer) { |
| return model; |
| } |
| return new SimpleNumber(num.intValue()); |
| } |
| } |
| |
| // Does both someNumber?long and someDate?long, thus it doesn't extend NumberBuiltIn |
| static class longBI extends BuiltIn { |
| TemplateModel _eval(Environment env) |
| throws TemplateException |
| { |
| TemplateModel model = target.eval(env); |
| if (!(model instanceof TemplateNumberModel) |
| && model instanceof TemplateDateModel) { |
| Date date = EvalUtil.modelToDate((TemplateDateModel) model, target); |
| return new SimpleNumber(date.getTime()); |
| } else { |
| Number num = target.modelToNumber(model, env); |
| if (num instanceof Long) { |
| return model; |
| } |
| return new SimpleNumber(num.longValue()); |
| } |
| } |
| } |
| |
| static class floatBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| if (num instanceof Float) { |
| return model; |
| } |
| return new SimpleNumber(num.floatValue()); |
| } |
| } |
| |
| static class doubleBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| if (num instanceof Double) { |
| return model; |
| } |
| return new SimpleNumber(num.doubleValue()); |
| } |
| } |
| |
| |
| private static final BigDecimal BIG_DECIMAL_ONE = new BigDecimal("1"); |
| |
| static class floorBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| return new SimpleNumber(new BigDecimal(num.doubleValue()).divide(BIG_DECIMAL_ONE, 0, BigDecimal.ROUND_FLOOR)); |
| } |
| } |
| |
| static class ceilingBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| return new SimpleNumber(new BigDecimal(num.doubleValue()).divide(BIG_DECIMAL_ONE, 0, BigDecimal.ROUND_CEILING)); |
| } |
| } |
| |
| static class roundBI extends NumberBuiltIn { |
| private static final BigDecimal half = new BigDecimal("0.5"); |
| TemplateModel calculateResult(Number num, TemplateModel model) { |
| return new SimpleNumber(new BigDecimal(num.doubleValue()).add(half).divide(BIG_DECIMAL_ONE, 0, BigDecimal.ROUND_FLOOR)); |
| } |
| } |
| |
| static class absBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateModelException { |
| if (num instanceof Integer) { |
| int n = ((Integer) num).intValue(); |
| if (n < 0) { |
| return new SimpleNumber(-n); |
| } else { |
| return model; |
| } |
| } else if (num instanceof BigDecimal) { |
| BigDecimal n = (BigDecimal) num; |
| if (n.signum() < 0) { |
| return new SimpleNumber(n.negate()); |
| } else { |
| return model; |
| } |
| } else if (num instanceof Double) { |
| double n = ((Double) num).doubleValue(); |
| if (n < 0) { |
| return new SimpleNumber(-n); |
| } else { |
| return model; |
| } |
| } else if (num instanceof Float) { |
| float n = ((Float) num).floatValue(); |
| if (n < 0) { |
| return new SimpleNumber(-n); |
| } else { |
| return model; |
| } |
| } else if (num instanceof Long) { |
| long n = ((Long) num).longValue(); |
| if (n < 0) { |
| return new SimpleNumber(-n); |
| } else { |
| return model; |
| } |
| } else if (num instanceof Short) { |
| short n = ((Short) num).shortValue(); |
| if (n < 0) { |
| return new SimpleNumber(-n); |
| } else { |
| return model; |
| } |
| } else if (num instanceof Byte) { |
| byte n = ((Byte) num).byteValue(); |
| if (n < 0) { |
| return new SimpleNumber(-n); |
| } else { |
| return model; |
| } |
| } else if (num instanceof BigInteger) { |
| BigInteger n = (BigInteger) num; |
| if (n.signum() < 0) { |
| return new SimpleNumber(n.negate()); |
| } else { |
| return model; |
| } |
| } else { |
| throw new _TemplateModelException(new Object[] { |
| "Unsupported number class: ", num.getClass() }); |
| } |
| } |
| } |
| |
| static class is_nanBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateModelException { |
| return NumberUtil.isNaN(num) ? TemplateBooleanModel.TRUE : TemplateBooleanModel.FALSE; |
| } |
| } |
| |
| static class is_infiniteBI extends NumberBuiltIn { |
| TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateModelException { |
| return NumberUtil.isInfinite(num) ? TemplateBooleanModel.TRUE : TemplateBooleanModel.FALSE; |
| } |
| } |
| |
| private static final BigDecimal BIG_DECIMAL_LONG_MIN = BigDecimal.valueOf(Long.MIN_VALUE); |
| private static final BigDecimal BIG_DECIMAL_LONG_MAX = BigDecimal.valueOf(Long.MAX_VALUE); |
| private static final BigInteger BIG_INTEGER_LONG_MIN = BigInteger.valueOf(Long.MIN_VALUE); |
| private static final BigInteger BIG_INTEGER_LONG_MAX = BigInteger.valueOf(Long.MAX_VALUE); |
| |
| private static final long safeToLong(Number num) throws TemplateModelException { |
| if (num instanceof Double) { |
| double d = Math.round(((Double) num).doubleValue()); |
| if (d > Long.MAX_VALUE || d < Long.MIN_VALUE) { |
| throw new _TemplateModelException(new Object[] { |
| "Number doesn't fit into a 64 bit signed integer (long): ", new Double(d) }); |
| } else { |
| return (long) d; |
| } |
| } else if (num instanceof Float) { |
| float f = Math.round(((Float) num).floatValue()); |
| if (f > Long.MAX_VALUE || f < Long.MIN_VALUE) { |
| throw new _TemplateModelException(new Object[] { |
| "Number doesn't fit into a 64 bit signed integer (long): ", new Float(f) }); |
| } else { |
| return (long) f; |
| } |
| } else if (num instanceof BigDecimal) { |
| BigDecimal bd = ((BigDecimal) num).setScale(0, BigDecimal.ROUND_HALF_UP); |
| if (bd.compareTo(BIG_DECIMAL_LONG_MAX) > 0 || bd.compareTo(BIG_DECIMAL_LONG_MIN) < 0) { |
| throw new _TemplateModelException(new Object[] { |
| "Number doesn't fit into a 64 bit signed integer (long): ", bd }); |
| } else { |
| return bd.longValue(); |
| } |
| } else if (num instanceof BigInteger) { |
| BigInteger bi = (BigInteger) num; |
| if (bi.compareTo(BIG_INTEGER_LONG_MAX) > 0 || bi.compareTo(BIG_INTEGER_LONG_MIN) < 0) { |
| throw new _TemplateModelException(new Object[] { |
| "Number doesn't fit into a 64 bit signed integer (long): ", bi }); |
| } else { |
| return bi.longValue(); |
| } |
| } else if (num instanceof Long || num instanceof Integer || num instanceof Byte || num instanceof Short) { |
| return num.longValue(); |
| } else { |
| // Should add Atomic* types in 2.4... |
| throw new _TemplateModelException(new Object[] { "Unsupported number type: ", num.getClass() }); |
| } |
| } |
| |
| static class number_to_dateBI extends NumberBuiltIn { |
| |
| private final int dateType; |
| |
| number_to_dateBI(int dateType) { |
| this.dateType = dateType; |
| } |
| |
| TemplateModel calculateResult(Number num, TemplateModel model) |
| throws TemplateModelException { |
| return new SimpleDate(new Date(safeToLong(num)), dateType); |
| } |
| } |
| |
| } |