Package mondrian.olap.fun.vba

Source Code of mondrian.olap.fun.vba.Vba

/*
// This software is subject to the terms of the Eclipse Public License v1.0
// Agreement, available at the following URL:
// http://www.eclipse.org/legal/epl-v10.html.
// Copyright (C) 2007-2010 Julian Hyde
// All Rights Reserved.
// You must accept the terms of that agreement to use this software.
*/
package mondrian.olap.fun.vba;

import mondrian.olap.Util;
import mondrian.olap.InvalidArgumentException;
import static mondrian.olap.fun.JavaFunDef.*;

import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.text.*;

/**
* Implementations of functions in the Visual Basic for Applications (VBA)
* specification.
*
* <p>The functions are defined in
* <a href="http://msdn.microsoft.com/en-us/library/32s6akha(VS.80).aspx">MSDN
* </a>.
*
* @author jhyde
* @version $Id: //open/mondrian-release/3.2/src/main/mondrian/olap/fun/vba/Vba.java#2 $
* @since Dec 31, 2007
*/
public class Vba {
    private static final long MILLIS_IN_A_DAY = 24 * 60 * 60 * 1000;

    private static final DateFormatSymbols DATE_FORMAT_SYMBOLS =
        new DateFormatSymbols(Locale.getDefault());

    // Conversion

    @FunctionName("CBool")
    @Signature("CBool(expression)")
    @Description(
        "Returns an expression that has been converted to a Variant of subtype "
        + "Boolean.")
    public static boolean cBool(Object expression) {
        if (expression instanceof Boolean) {
            return (Boolean) expression;
        } else {
            int i = cInt(expression);
            return i != 0;
        }
    }

    // Conversion functions

    @FunctionName("CByte")
    @Signature("CByte(expression)")
    @Description(
        "Returns an expression that has been converted to a Variant of subtype "
        + "Byte.")
    public static byte cByte(Object expression) {
        if (expression instanceof Byte) {
            return (Byte) expression;
        } else {
            int i = cInt(expression);
            return (byte) i;
        }
    }

    // public Currency cCur(Object expression)

    @FunctionName("CDate")
    @Signature("CDate(date)")
    @Description(
        "Returns an expression that has been converted to a Variant of subtype "
        + "Date.")
    public static Date cDate(Object expression) {
        String str = String.valueOf(expression);
        if (expression instanceof Date) {
            return (Date) expression;
        } else if (expression == null) {
            return null;
        } else {
            // note that this currently only supports a limited set of dates and
            // times
            // "October 19, 1962"
            // "4:35:47 PM"
            try {
                return DateFormat.getTimeInstance().parse(str);
            } catch (ParseException ex0) {
                try {
                    return DateFormat.getDateTimeInstance().parse(str);
                } catch (ParseException ex1) {
                    try {
                        return DateFormat.getDateInstance().parse(str);
                    } catch (ParseException ex2) {
                        throw new InvalidArgumentException(
                            "Invalid parameter. "
                            + "expression parameter of CDate function must be "
                            + "formatted correctly ("
                            + String.valueOf(expression) + ")");
                    }
                }
            }
        }
    }

    @FunctionName("CDbl")
    @Signature("CDbl(expression)")
    @Description(
        "Returns an expression that has been converted to a Variant of subtype "
        + "Double.")
    public static double cDbl(Object expression) {
        if (expression instanceof Number) {
            Number number = (Number) expression;
            return number.doubleValue();
        } else {
            final String s = String.valueOf(expression);
            return new Double(s).intValue();
        }
    }

    @FunctionName("CInt")
    @Signature("CInt(expression)")
    @Description(
        "Returns an expression that has been converted to a Variant of subtype "
        + "Integer.")
    public static int cInt(Object expression) {
        if (expression instanceof Number) {
            Number number = (Number) expression;
            final int intValue = number.intValue();
            if (number instanceof Float || number instanceof Double) {
                final double doubleValue = number.doubleValue();
                if (doubleValue == (double) intValue) {
                    // Number is already an integer
                    return intValue;
                }
                final double doubleDouble = doubleValue * 2d;
                if (doubleDouble == Math.floor(doubleDouble)) {
                    // Number ends in .5 - round towards even required
                    return (int) Math.round(doubleValue / 2d) * 2;
                }
                return (int) Math.round(doubleValue);
            }
            return intValue;
        } else {
            // Try to parse as integer before parsing as double. More
            // efficient, and avoids loss of precision.
            final String s = String.valueOf(expression);
            try {
                return Integer.parseInt(s);
            } catch (NumberFormatException e) {
                return new Double(s).intValue();
            }
        }
    }

    // public int cLng(Object expression)
    // public float cSng(Object expression)
    // public String cStr(Object expression)
    // public Object cVDate(Object expression)
    // public Object cVErr(Object expression)
    // public Object cVar(Object expression)
    // public String error$(Object errorNumber)
    // public Object error(Object errorNumber)

    @FunctionName("Fix")
    @Signature("Fix(number)")
    @Description(
        "Returns the integer portion of a number. If negative, returns the "
        + "negative number greater than or equal to the number.")
    public static int fix(Object number) {
        if (number instanceof Number) {
            int v = ((Number) number).intValue();
            double dv = ((Number) number).doubleValue();
            if (v < 0 && v < dv) {
                v++;
            }
            return v;
        } else {
            throw new InvalidArgumentException(
                "Invalid parameter. "
                + "number parameter " + number
                + " of Int function must be " + "of type number");
        }
    }

    @FunctionName("Hex")
    @Signature("Hex(number)")
    @Description(
        "Returns a String representing the hexadecimal value of a number.")
    public static String hex(Object number) {
        if (number instanceof Number) {
            return Integer.toHexString(((Number) number).intValue())
                    .toUpperCase();
        } else {
            throw new InvalidArgumentException(
                "Invalid parameter. "
                + "number parameter " + number
                + " of Hex function must be " + "of type number");
        }
    }

    @FunctionName("Int")
    @Signature("Int(number)")
    @Description(
        "Returns the integer portion of a number. If negative, returns the "
        + "negative number less than or equal to the number.")
    public static int int_(Object number) {
        if (number instanceof Number) {
            int v = ((Number) number).intValue();
            double dv = ((Number) number).doubleValue();
            if (v < 0 && v > dv) {
                v--;
            }
            return v;
        } else {
            throw new InvalidArgumentException(
                "Invalid parameter. "
                + "number parameter " + number
                + " of Int function must be " + "of type number");
        }
    }

    /**
     * Equivalent of the {@link #int_} function on the native 'double' type.
     * Not an MDX function.
     *
     * @param dv Double value
     * @return Value rounded towards negative infinity
     */
    static int intNative(double dv) {
        int v = (int) dv;
        if (v < 0 && v > dv) {
            v--;
        }
        return v;
    }

    // public String oct$(Object number)

    @FunctionName("Oct")
    @Signature("Oct(number)")
    @Description(
        "Returns a Variant (String) representing the octal value of a number.")
    public static String oct(Object number) {
        if (number instanceof Number) {
            return Integer.toOctalString(((Number) number).intValue());
        } else {
            throw new InvalidArgumentException(
                "Invalid parameter. "
                + "number parameter " + number
                + " of Oct function must be " + "of type number");
        }
    }

    // public String str$(Object number)

    @FunctionName("Str")
    @Signature("Str(number)")
    @Description("Returns a Variant (String) representation of a number.")
    public static String str(Object number) {
        // When numbers are converted to strings, a leading space is always
        // reserved for the sign of number. If number is positive, the returned
        // string contains a leading space and the plus sign is implied.
        //
        // Use the Format function to convert numeric values you want formatted
        // as dates, times, or currency or in other user-defined formats.
        // Unlike Str, the Format function doesn't include a leading space for
        // the sign of number.
        //
        // Note The Str function recognizes only the period (.) as a valid
        // decimal separator. When different decimal separators may be used
        // (for example, in international applications), use CStr to convert a
        // number to a string.
        if (number instanceof Number) {
            if (((Number) number).doubleValue() >= 0) {
                return " " + number.toString();
            } else {
                return number.toString();
            }
        } else {
            throw new InvalidArgumentException(
                "Invalid parameter. "
                + "number parameter " + number
                + " of Str function must be " + "of type number");
        }
    }

    @FunctionName("Val")
    @Signature("Val(string)")
    @Description(
        "Returns the numbers contained in a string as a numeric value of "
        + "appropriate type.")
    public static double val(String string) {
        // The Val function stops reading the string at the first character it
        // can't recognize as part of a number. Symbols and characters that are
        // often considered parts of numeric values, such as dollar signs and
        // commas, are not recognized. However, the function recognizes the
        // radix prefixes &O (for octal) and &H (for hexadecimal). Blanks,
        // tabs, and linefeed characters are stripped from the argument.
        //
        // The following returns the value 1615198:
        //
        // Val(" 1615 198th Street N.E.")
        // In the code below, Val returns the decimal value -1 for the
        // hexadecimal value shown:
        //
        // Val("&HFFFF")
        // Note The Val function recognizes only the period (.) as a valid
        // decimal separator. When different decimal separators are used, as in
        // international applications, use CDbl instead to convert a string to
        // a number.

        string = string.replaceAll("\\s", ""); // remove all whitespace
        if (string.startsWith("&H")) {
            string = string.substring(2);
            Pattern p = Pattern.compile("[0-9a-fA-F]*");
            Matcher m = p.matcher(string);
            m.find();
            return Integer.parseInt(m.group(), 16);
        } else if (string.startsWith("&O")) {
            string = string.substring(2);
            Pattern p = Pattern.compile("[0-7]*");
            Matcher m = p.matcher(string);
            m.find();
            return Integer.parseInt(m.group(), 8);
        } else {
            // find the first number
            Pattern p = Pattern.compile("-?[0-9]*[.]?[0-9]*");
            Matcher m = p.matcher(string);
            m.find();
            return Double.parseDouble(m.group());
        }
    }

    // DateTime

    // public Calendar calendar()
    // public void calendar(Calendar val)
    // public String date$()
    // public void date$(String val)

    @FunctionName("DateAdd")
    @Signature("DateAdd(interval, number, date)")
    @Description(
        "Returns a Variant (Date) containing a date to which a specified time "
        + "interval has been added.")
    public static Date dateAdd(String intervalName, double number, Date date) {
        Interval interval = Interval.valueOf(intervalName);
        final double floor = Math.floor(number);

        // We use the local calendar here. This method will therefore return
        // different results in different locales: it depends whether the
        // initial date and the final date are in DST.
        Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        if (floor != number) {
            final double ceil = Math.ceil(number);
            interval.add(calendar, (int) ceil);
            final long ceilMillis = calendar.getTimeInMillis();

            calendar.setTime(date);
            interval.add(calendar, (int) floor);
            final long floorMillis = calendar.getTimeInMillis();

            final long amount =
                (long)
                    (((double) (ceilMillis - floorMillis)) * (number - floor));
            calendar.add(
                Calendar.DAY_OF_YEAR,
                (int) (amount / MILLIS_IN_A_DAY));
            calendar.add(
                Calendar.MILLISECOND, (int)
                (amount % MILLIS_IN_A_DAY));
        } else {
            interval.add(calendar, (int) floor);
        }
        return calendar.getTime();
    }

    @FunctionName("DateDiff")
    @Signature(
        "DateDiff(interval, date1, date2[, firstdayofweek[, firstweekofyear]])")
    @Description(
        "Returns a Variant (Long) specifying the number of time intervals "
        + "between two specified dates.")
    public static long dateDiff(String interval, Date date1, Date date2) {
        return _dateDiff(
            interval, date1, date2, Calendar.SUNDAY,
            FirstWeekOfYear.vbFirstJan1);
    }

    @FunctionName("DateDiff")
    @Signature(
        "DateDiff(interval, date1, date2[, firstdayofweek[, firstweekofyear]])")
    @Description(
        "Returns a Variant (Long) specifying the number of time intervals "
        + "between two specified dates.")
    public static long dateDiff(
        String interval, Date date1, Date date2, int firstDayOfWeek)
    {
        return _dateDiff(
            interval, date1, date2, firstDayOfWeek,
            FirstWeekOfYear.vbFirstJan1);
    }

    @FunctionName("DateDiff")
    @Signature(
        "DateDiff(interval, date1, date2[, firstdayofweek[, firstweekofyear]])")
    @Description(
        "Returns a Variant (Long) specifying the number of time intervals "
        + "between two specified dates.")
    public static long dateDiff(
        String interval, Date date1, Date date2,
        int firstDayOfWeek, int firstWeekOfYear)
    {
        return _dateDiff(
            interval, date1, date2, firstDayOfWeek,
            FirstWeekOfYear.values()[firstWeekOfYear]);
    }

    private static long _dateDiff(
        String intervalName, Date date1, Date date2,
        int firstDayOfWeek, FirstWeekOfYear firstWeekOfYear)
    {
        Interval interval = Interval.valueOf(intervalName);
        Calendar calendar1 = Calendar.getInstance();
        firstWeekOfYear.apply(calendar1);
        calendar1.setTime(date1);
        Calendar calendar2 = Calendar.getInstance();
        firstWeekOfYear.apply(calendar2);
        calendar2.setTime(date2);
        return interval.diff(calendar1, calendar2, firstDayOfWeek);
    }

    @FunctionName("DatePart")
    @Signature("DatePart(interval, date[,firstdayofweek[, firstweekofyear]])")
    @Description(
        "Returns a Variant (Integer) containing the specified part of a given "
        + "date.")
    public static int datePart(String interval, Date date) {
        return _datePart(
            interval, date, Calendar.SUNDAY,
            FirstWeekOfYear.vbFirstJan1);
    }

    @FunctionName("DatePart")
    @Signature("DatePart(interval, date[,firstdayofweek[, firstweekofyear]])")
    @Description(
        "Returns a Variant (Integer) containing the specified part of a given "
        + "date.")
    public static int datePart(String interval, Date date, int firstDayOfWeek) {
        return _datePart(
            interval, date, firstDayOfWeek,
            FirstWeekOfYear.vbFirstJan1);
    }

    @FunctionName("DatePart")
    @Signature("DatePart(interval, date[,firstdayofweek[, firstweekofyear]])")
    @Description(
        "Returns a Variant (Integer) containing the specified part of a given "
        + "date.")
    public static int datePart(
        String interval, Date date, int firstDayOfWeek,
        int firstWeekOfYear)
    {
        return _datePart(
            interval, date, firstDayOfWeek,
            FirstWeekOfYear.values()[firstWeekOfYear]);
    }

    private static int _datePart(
        String intervalName,
        Date date,
        int firstDayOfWeek,
        FirstWeekOfYear firstWeekOfYear)
    {
        Interval interval = Interval.valueOf(intervalName);
        Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        switch (interval) {
        case w:
        case ww:
            // firstWeekOfYear and firstDayOfWeek only matter for 'w' and 'ww'
            firstWeekOfYear.apply(calendar);
            calendar.setFirstDayOfWeek(firstDayOfWeek);
            break;
        }
        return interval.datePart(calendar);
    }

    @FunctionName("Date")
    @Signature("Date")
    @Description("Returns a Variant (Date) containing the current system date.")
    public static Date date() {
        Calendar calendar = Calendar.getInstance();
        calendar.clear();
        calendar.set(Calendar.HOUR_OF_DAY, 0);
        calendar.set(Calendar.MINUTE, 0);
        calendar.set(Calendar.SECOND, 0);
        calendar.set(Calendar.MILLISECOND, 0);
        return calendar.getTime();
    }

    // public void date(Object val)

    @FunctionName("DateSerial")
    @Signature("DateSerial(year, month, day)")
    @Description(
        "Returns a Variant (Date) for a specified year, month, and day.")
    public static Date dateSerial(int year, int month, int day) {
        Calendar calendar = Calendar.getInstance();
        calendar.clear();
        calendar.set(year, month - 1, day);
        return calendar.getTime();
    }

    @FunctionName("DateValue")
    @Signature("DateValue(date)")
    @Description("Returns a Variant (Date).")
    public static Date dateValue(Date date) {
        final Calendar calendar = Calendar.getInstance();
        calendar.clear();
        calendar.setTime(date);
        calendar.set(Calendar.HOUR_OF_DAY, 0);
        calendar.set(Calendar.MINUTE, 0);
        calendar.set(Calendar.SECOND, 0);
        calendar.set(Calendar.MILLISECOND, 0);
        return calendar.getTime();
    }

    @FunctionName("Day")
    @Signature("Day(date)")
    @Description(
        "Returns a Variant (Integer) specifying a whole number between 1 and "
        + "31, inclusive, representing the day of the month.")
    public static int day(Date date) {
        final Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        return calendar.get(Calendar.DAY_OF_MONTH);
    }

    @FunctionName("Hour")
    @Signature("Hour(time)")
    @Description(
        "Returns a Variant (Integer) specifying a whole number between 0 and "
        + "23, inclusive, representing the hour of the day.")
    public static int hour(Date time) {
        final Calendar calendar = Calendar.getInstance();
        calendar.setTime(time);
        return calendar.get(Calendar.HOUR_OF_DAY);
    }

    @FunctionName("Minute")
    @Signature("Minute(time)")
    @Description(
        "Returns a Variant (Integer) specifying a whole number between 0 and "
        + "59, inclusive, representing the minute of the hour.")
    public static int minute(Date time) {
        final Calendar calendar = Calendar.getInstance();
        calendar.setTime(time);
        return calendar.get(Calendar.MINUTE);
    }

    @FunctionName("Month")
    @Signature("Month(date)")
    @Description(
        "Returns a Variant (Integer) specifying a whole number between 1 and "
        + "12, inclusive, representing the month of the year.")
    public static int month(Date date) {
        final Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        final int month = calendar.get(Calendar.MONTH);
        return month + 1; // convert from 0- to 1-based
    }

    @FunctionName("Now")
    @Signature("Now()")
    @Description(
        "Returns a Variant (Date) specifying the current date and time "
        + "according your computer's system date and time.")
    public static Date now() {
        return new Date();
    }

    @FunctionName("Second")
    @Signature("Second(time)")
    @Description(
        "Returns a Variant (Integer) specifying a whole number between 0 and "
        + "59, inclusive, representing the second of the minute.")
    public static int second(Date time) {
        final Calendar calendar = Calendar.getInstance();
        calendar.setTime(time);
        return calendar.get(Calendar.SECOND);
    }

    // public String time$()
    // public void time$(String val)

    @FunctionName("Time")
    @Signature("Time()")
    @Description("Returns a Variant (Date) indicating the current system time.")
    public static Date time() {
        return new Date();
    }

    // public void time(Object val)

    @FunctionName("TimeSerial")
    @Signature("TimeSerial(hour, minute, second)")
    @Description(
        "Returns a Variant (Date) containing the time for a specific hour, "
        + "minute, and second.")
    public static Date timeSerial(int hour, int minute, int second) {
        final Calendar calendar = Calendar.getInstance();
        calendar.clear();
        calendar.set(Calendar.HOUR_OF_DAY, hour);
        calendar.set(Calendar.MINUTE, minute);
        calendar.set(Calendar.SECOND, second);
        return calendar.getTime();
    }

    @FunctionName("TimeValue")
    @Signature("TimeValue(time)")
    @Description("Returns a Variant (Date) containing the time.")
    public static Date timeValue(Date time) {
        final Calendar calendar = Calendar.getInstance();
        calendar.clear();
        calendar.setTime(time);
        calendar.set(1970, 0, 1);
        return calendar.getTime();
    }

    @FunctionName("Timer")
    @Signature("Timer()")
    @Description(
        "Returns a Single representing the number of seconds elapsed since "
        + "midnight.")
    public static float timer() {
        final Calendar calendar = Calendar.getInstance();
        final long now = calendar.getTimeInMillis();
        calendar.set(Calendar.HOUR_OF_DAY, 0);
        calendar.set(Calendar.MINUTE, 0);
        calendar.set(Calendar.SECOND, 0);
        calendar.set(Calendar.MILLISECOND, 0);
        final long midnight = calendar.getTimeInMillis();
        return ((float) (now - midnight)) / 1000f;
    }

    @FunctionName("Weekday")
    @Signature("Weekday(date[, firstDayOfWeek])")
    @Description(
        "Returns a Variant (Integer) containing a whole number representing "
        + "the day of the week.")
    public static int weekday(Date date) {
        return weekday(date, Calendar.SUNDAY);
    }

    @FunctionName("Weekday")
    @Signature("Weekday(date[, firstDayOfWeek])")
    @Description(
        "Returns a Variant (Integer) containing a whole number representing "
        + "the day of the week.")
    public static int weekday(Date date, int firstDayOfWeek) {
        final Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        int weekday = calendar.get(Calendar.DAY_OF_WEEK);
        // adjust for start of week
        weekday -= (firstDayOfWeek - 1);
        // bring into range 1..7
        weekday = (weekday + 6) % 7 + 1;
        return weekday;
    }

    @FunctionName("Year")
    @Signature("Year(date)")
    @Description(
        "Returns a Variant (Integer) containing a whole number representing "
        + "the year.")
    public static int year(Date date) {
        final Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        return calendar.get(Calendar.YEAR);
    }

    //
    // /* FileSystem */
    // public void chDir(String path)
    // public void chDrive(String drive)
    // public String curDir$(Object drive)
    // public Object curDir(Object drive)
    // public String dir(Object pathName, FileAttribute attributes /* default
    // FileAttribute.Normal */)
    // public boolean EOF(int fileNumber)
    // public int fileAttr(int fileNumber, int returnType /* default 1 */)
    // public void fileCopy(String source, String destination)
    // public Object fileDateTime(String pathName)
    // public int fileLen(String pathName)
    // public int freeFile(Object rangeNumber)
    // public FileAttribute getAttr(String pathName)
    // public void kill(Object pathName)
    // public int LOF(int fileNumber)
    // public int loc(int fileNumber)
    // public void mkDir(String path)
    // public void reset()
    // public void rmDir(String path)
    // public int seek(int fileNumber)
    // public void setAttr(String pathName, FileAttribute attributes)
    //
    // Financial

    @FunctionName("DDB")
    @Signature("DDB(cost, salvage, life, period[, factor])")
    @Description(
        "Returns a Double specifying the depreciation of an asset for a "
        + "specific time period using the double-declining balance method or "
        + "some other method you specify.")
    public static double dDB(
        double cost,
        double salvage,
        double life,
        double period)
    {
        return dDB(cost, salvage, life, period, 2.0);
    }

    @FunctionName("DDB")
    @Signature("DDB(cost, salvage, life, period[, factor])")
    @Description(
        "Returns a Double specifying the depreciation of an asset for a "
        + "specific time period using the double-declining balance method or "
        + "some other method you specify.")
    public static double dDB(
        double cost,
        double salvage,
        double life,
        double period,
        double factor)
    {
        return (((cost - salvage) * factor) / life) * period;
    }

    @FunctionName("FV")
    @Signature("FV(rate, nper, pmt[, pv[, type]])")
    @Description(
        "Returns a Double specifying the future value of an annuity based on "
        + "periodic, fixed payments and a fixed interest rate.")
    public static double fV(double rate, double nPer, double pmt) {
        return fV(rate, nPer, pmt, 0d, false);
    }

    @FunctionName("FV")
    @Signature("FV(rate, nper, pmt[, pv[, type]])")
    @Description(
        "Returns a Double specifying the future value of an annuity based on "
        + "periodic, fixed payments and a fixed interest rate.")
    public static double fV(double rate, double nPer, double pmt, double pv) {
        return fV(rate, nPer, pmt, pv, false);
    }

    @FunctionName("FV")
    @Signature("FV(rate, nper, pmt[, pv[, type]])")
    @Description(
        "Returns a Double specifying the future value of an annuity based on "
        + "periodic, fixed payments and a fixed interest rate.")
    public static double fV(
        double rate,
        double nPer,
        double pmt,
        double pv,
        boolean type)
    {
        if (rate == 0) {
            return -(pv + (nPer * pmt));
        } else {
            double r1 = rate + 1;
            return ((1 - Math.pow(r1, nPer)) * (type ? r1 : 1) * pmt) / rate
                    - pv * Math.pow(r1, nPer);
        }
    }

    @FunctionName("IPmt")
    @Signature("IPmt(rate, per, nper, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the interest payment for a given period "
        + "of an annuity based on periodic, fixed payments and a fixed "
        + "interest rate.")
    public static double iPmt(double rate, double per, double nPer, double PV) {
        return iPmt(rate, per, nPer, PV, 0);
    }


    @FunctionName("IPmt")
    @Signature("IPmt(rate, per, nper, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the interest payment for a given period "
        + "of an annuity based on periodic, fixed payments and a fixed "
        + "interest rate.")
    public static double iPmt(
        double rate,
        double per,
        double nPer,
        double PV,
        double fv)
    {
        return iPmt(rate, per, nPer, PV, fv, false);
    }


    @FunctionName("IPmt")
    @Signature("IPmt(rate, per, nper, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the interest payment for a given period "
        + "of an annuity based on periodic, fixed payments and a fixed "
        + "interest rate.")
    public static double iPmt(
        double rate,
        double per,
        double nPer,
        double PV,
        double fv,
        boolean due)
    {
        double pmtVal = pmt(rate, nPer, PV, fv, due);
        double pValm1 = PV - pV(rate, per - 1, pmtVal, fv, due);
        return - pValm1 * rate;
    }

    @FunctionName("IRR")
    @Signature("IRR(values()[, guess])")
    @Description(
        "Returns a Double specifying the internal rate of return for a series "
        + "of periodic cash flows (payments and receipts).")
    public static double IRR(double[] valueArray) {
        return IRR(valueArray, 0.10);
    }


    @FunctionName("IRR")
    @Signature("IRR(values()[, guess])")
    @Description(
        "Returns a Double specifying the internal rate of return for a series "
        + "of periodic cash flows (payments and receipts).")
    public static double IRR(double[] valueArray, double guess) {
        // calc pV of stream (sum of pV's for valueArray) ((1 + guess) ^ index)
        double minGuess = 0.0;
        double maxGuess = 1.0;

        // i'm not certain
        int r = 1;
        if (valueArray[0] > 0) {
            r = -1;
        }

        for (int i = 0; i < 30; i++) {
            // first calculate overall return based on guess
            double totalPv = 0;
            for (int j = 0; j < valueArray.length; j++) {
                totalPv += valueArray[j] / Math.pow(1.0 + guess, j);
            }
            if ((maxGuess - minGuess) < 0.0000001) {
                return guess;
            } else if (totalPv * r < 0) {
                maxGuess = guess;
            } else {
                minGuess = guess;
            }
            // avg max min to determine next step
            guess = (maxGuess + minGuess) / 2;
        }
        // unable to find a match
        return -1;
    }

    @FunctionName("MIRR")
    @Signature("MIRR(values(), finance_rate, reinvest_rate)")
    @Description(
        "Returns a Double specifying the modified internal rate of return for "
        + "a series of periodic cash flows (payments and receipts).")
    public static double MIRR(
        double valueArray[],
        double financeRate,
        double reinvestRate)
    {
        // based on
        // http://en.wikipedia.org/wiki/Modified_Internal_Rate_of_Return
        double reNPV = 0.0;
        double fiNPV = 0.0;
        for (int j = 0; j < valueArray.length; j++) {
            if (valueArray[j] > 0) {
                reNPV += valueArray[j] / Math.pow(1.0 + reinvestRate, j);
            } else {
                fiNPV += valueArray[j] / Math.pow(1.0 + financeRate, j);
            }
        }

        double ratio =
            (- reNPV * Math.pow(1 + reinvestRate, valueArray.length))
            / (fiNPV * (1 + financeRate));

        return Math.pow(ratio, 1.0 / (valueArray.length - 1)) - 1.0;
    }

    @FunctionName("NPer")
    @Signature("NPer(rate, pmt, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the number of periods for an annuity "
        + "based on periodic, fixed payments and a fixed interest rate.")
    public static double nPer(
        double rate,
        double pmt,
        double pv,
        double fv,
        boolean due)
    {
        if (rate == 0) {
            return -(fv + pv) / pmt;
        } else {
            double r1 = rate + 1;
            double ryr = (due ? r1 : 1) * pmt / rate;
            double a1 =
                ((ryr - fv) < 0)
                ? Math.log(fv - ryr)
                : Math.log(ryr - fv);
            double a2 =
                ((ryr - fv) < 0)
                ? Math.log(-pv - ryr)
                : Math.log(pv + ryr);
            double a3 = Math.log(r1);
            return (a1 - a2) / a3;
        }
    }

    @FunctionName("NPV")
    @Signature("NPV(rate, values())")
    @Description(
        "Returns a Double specifying the net present value of an investment "
        + "based on a series of periodic cash flows (payments and receipts) "
        + "and a discount rate.")
    public static double nPV(double r, double[] cfs) {
        double npv = 0;
        double r1 = r + 1;
        double trate = r1;
        for (int i = 0, iSize = cfs.length; i < iSize; i++) {
            npv += cfs[i] / trate;
            trate *= r1;
        }
        return npv;
    }

    @FunctionName("PPmt")
    @Signature("PPmt(rate, per, nper, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the principal payment for a given period "
        + "of an annuity based on periodic, fixed payments and a fixed "
        + "interest rate.")
    public static double pPmt(double rate, double per, double nPer, double PV) {
        return pPmt(rate, per, nPer, PV, 0);
    }

    @FunctionName("PPmt")
    @Signature("PPmt(rate, per, nper, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the principal payment for a given period "
        + "of an annuity based on periodic, fixed payments and a fixed "
        + "interest rate.")
    public static double pPmt(
        double rate,
        double per,
        double nPer,
        double PV,
        double fv)
    {
        return pPmt(rate, per, nPer, PV, fv, false);
    }

    @FunctionName("PPmt")
    @Signature("PPmt(rate, per, nper, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the principal payment for a given period "
        + "of an annuity based on periodic, fixed payments and a fixed "
        + "interest rate.")
    public static double pPmt(
        double rate,
        double per,
        double nPer,
        double PV,
        double fv,
        boolean due)
    {
        return pmt(rate, nPer, PV, fv, due)
            - iPmt(rate, per, nPer, PV, fv, due);
    }

    @FunctionName("Pmt")
    @Signature("Pmt(rate, nper, pv[, fv[, type]])")
    @Description(
        "Returns a Double specifying the payment for an annuity based on "
        + "periodic, fixed payments and a fixed interest rate.")
    public static double pmt(
        double rate,
        double nPer,
        double pv,
        double fv,
        boolean due)
    {
        if (rate == 0) {
            return -(fv + pv) / nPer;
        } else {
            double r1 = rate + 1;
            return
                (fv + pv * Math.pow(r1, nPer))
                * rate
                / ((due ? r1 : 1) * (1 - Math.pow(r1, nPer)));
        }
    }

    @FunctionName("PV")
    @Signature("PV(rate, nper, pmt[, fv[, type]])")
    @Description(
        "Returns a Double specifying the present value of an annuity based on "
        + "periodic, fixed payments to be paid in the future and a fixed "
        + "interest rate.")
    public static double pV(
        double rate,
        double nper,
        double pmt,
        double fv,
        boolean due)
    {
        if (rate == 0) {
            return -((nper * pmt) + fv);
        } else {
            double r1 = rate + 1;
            return
                (((1 - Math.pow(r1, nper)) / rate) * (due ? r1 : 1) * pmt - fv)
                    / Math.pow(r1, nper);
        }
    }

    @FunctionName("Rate")
    @Signature("Rate(nper, pmt, pv[, fv[, type[, guess]]])")
    @Description(
        "Returns a Double specifying the interest rate per period for an "
        + "annuity.")
    public static double rate(
        double nPer,
        double pmt,
        double PV)
    {
        return rate(nPer, pmt, PV, 0);
    }

    @FunctionName("Rate")
    @Signature("Rate(nper, pmt, pv[, fv[, type[, guess]]])")
    @Description(
        "Returns a Double specifying the interest rate per period for an "
        + "annuity.")
    public static double rate(
        double nPer,
        double pmt,
        double PV,
        double fv)
    {
        return rate(nPer, pmt, PV, fv, false);
    }

    @FunctionName("Rate")
    @Signature("Rate(nper, pmt, pv[, fv[, type[, guess]]])")
    @Description(
        "Returns a Double specifying the interest rate per period for an "
        + "annuity.")
    public static double rate(
        double nPer,
        double pmt,
        double PV,
        double fv,
        boolean type)
    {
        return rate(nPer, pmt, PV, fv, type, 0.1);
    }

    @FunctionName("Rate")
    @Signature("Rate(nper, pmt, pv[, fv[, type[, guess]]])")
    @Description(
        "Returns a Double specifying the interest rate per period for an "
        + "annuity.")
    public static double rate(
        double nPer, // specifies the number of payment periods
        double pmt, // payment per period of annuity
        double PV, // the present value of the annuity (0 if a loan)
        double fv, // the future value of the annuity ($ if savings)
        boolean due,
        double guess)
    {
        if (nPer <= 0) {
            throw new InvalidArgumentException(
                "number of payment periods must be larger than 0");
        }
        double minGuess = 0.0;
        double maxGuess = 1.0;

        // converge on the correct answer should use Newton's Method
        // for now use a binary search
        int r = 1;
        if (PV < fv) {
            r = -1;
        }

        // the vb method uses 20 iterations, but they also probably use newton's
        // method,
        // so i've bumped it up to 30 iterations.
        for (int n = 0; n < 30; n++) {
            double gFV = fV(guess, nPer, pmt, PV, due);
            double diff = gFV - fv;
            if ((maxGuess - minGuess) < 0.0000001) {
                return guess;
            } else {
                if (diff * r < 0) {
                    maxGuess = guess;
                } else {
                    minGuess = guess;
                }
                guess = (maxGuess + minGuess) / 2;
            }
        }
        // fail, not sure how VB fails
        return -1;
    }

    @FunctionName("SLN")
    @Signature("SLN(cost, salvage, life)")
    @Description(
        "Returns a Double specifying the straight-line depreciation of an "
        + "asset for a single period.")
    public static double sLN(double cost, double salvage, double life) {
        return (cost - salvage) / life;
    }

    @FunctionName("SYD")
    @Signature("SYD(cost, salvage, life, period)")
    @Description(
        "Returns a Double specifying the sum-of-years' digits depreciation of "
        + "an asset for a specified period.")
    public static double sYD(
        double cost,
        double salvage,
        double life,
        double period)
    {
        return (cost - salvage) * (life / (period * (period + 1) / 2));
    }

    // Information

    // public Throwable err()
    // public Object iMEStatus()

    @FunctionName("IsArray")
    @Signature("IsArray(varname)")
    @Description(
        "Returns a Boolean value indicating whether a variable is an array.")
    public boolean isArray(Object varName) {
        // arrays are not supported at present
        return false;
    }

    @FunctionName("IsDate")
    @Signature("IsDate(varname)")
    @Description(
        "Returns a Boolean value indicating whether an expression can be "
        + "converted to a date.")
    public static boolean isDate(Object expression) {
        // IsDate returns True if Expression represents a valid date, a valid
        // time, or a valid date and time.
        try {
            Date val = cDate(expression);
            return (val != null);
        } catch (InvalidArgumentException e) {
            return false;
        }
    }

    // use mondrian's implementation of IsEmpty
    // public boolean isEmpty(Object expression)

    @FunctionName("IsError")
    @Signature("IsError(varname)")
    @Description(
        "Returns a Boolean value indicating whether an expression is an error "
        + "value.")
    public boolean isError(Object expression) {
        return expression instanceof Throwable;
    }

    @FunctionName("IsMissing")
    @Signature("IsMissing(varname)")
    @Description(
        "Returns a Boolean value indicating whether an optional Variant "
        + "argument has been passed to a procedure.")
    public boolean isMissing(Object argName) {
        // We have no way to detect missing arguments.
        return false;
    }

    @FunctionName("IsNull")
    @Signature("IsNull(varname)")
    @Description(
        "Returns a Boolean value that indicates whether an expression "
        + "contains no valid data (Null).")
    public boolean isNull(Object expression) {
        return expression == null;
    }

    @FunctionName("IsNumeric")
    @Signature("IsNumeric(varname)")
    @Description(
        "Returns a Boolean value indicating whether an expression can be "
        + "evaluated as a number.")
    public boolean isNumeric(Object expression) {
        return expression instanceof Number;
    }

    @FunctionName("IsObject")
    @Signature("IsObject(varname)")
    @Description(
        "Returns a Boolean value indicating whether an identifier represents "
        + "an object variable.")
    public boolean isObject(Object expression) {
        return false;
    }

    // public int qBColor(int color)
    // public int RGB(int red, int green, int blue)

    @FunctionName("TypeName")
    @Signature("TypeName(varname)")
    @Description("Returns a String that provides information about a variable.")
    public static String typeName(Object varName) {
        // The string returned by TypeName can be any one of the following:
        //
        // String returned Variable
        // object type An object whose type is objecttype
        // Byte Byte value
        // Integer Integer
        // Long Long integer
        // Single Single-precision floating-point number
        // Double Double-precision floating-point number
        // Currency Currency value
        // Decimal Decimal value
        // Date Date value
        // String String
        // Boolean Boolean value
        // Error An error value
        // Empty Uninitialized
        // Null No valid data
        // Object An object
        // Unknown An object whose type is unknown
        // Nothing Object variable that doesn't refer to an object

        if (varName == null) {
            return "NULL";
        } else {
            // strip off the package information
            String name = varName.getClass().getName();
            if (name.lastIndexOf(".") >= 0) {
                name = name.substring(name.lastIndexOf(".") + 1);
            }
            return name;
        }
    }

    // public VarType varType(Object varName)

    // Interaction

    // public void appActivate(Object title, Object wait)
    // public void beep()
    // public Object callByName(Object object, String procName, CallType
    // callType, Object args, int lcid)
    // public Object choose(float index, Object choice)
    // public String command$()
    // public Object command()
    // public Object createObject(String Class, String serverName)
    // public int doEvents()
    // public String environ$(Object expression)
    // public Object environ(Object expression)
    // public Object getAllSettings(String appName, String section)
    // public Object getObject(Object pathName, Object Class)
    // public String getSetting(String appName, String section, String key,
    // Object Default)
    // public Object iIf(Object expression, Object truePart, Object falsePart)
    // public String inputBox(Object prompt, Object title, Object Default,
    // Object xPos, Object yPos, Object helpFile, Object context)
    // public String macScript(String script)
    // public MsgBoxResult msgBox(Object prompt, MsgBoxStyle buttons /* default
    // MsgBoxStyle.OKOnly */, Object title, Object helpFile, Object context)
    // public Object partition(Object number, Object start, Object stop, Object
    // interval)
    // public void saveSetting(String appName, String section, String key,
    // String setting)
    // public void sendKeys(String string, Object wait)
    // public double shell(Object pathName, AppWinStyle windowStyle /* default
    // AppWinStyle.MinimizedFocus */)
    // public Object Switch(Object varExpr)

    // Mathematical

    @FunctionName("Abs")
    @Signature("Abs(number)")
    @Description(
        "Returns a value of the same type that is passed to it specifying the "
        + "absolute value of a number.")
    public static double abs(double number) {
        return Math.abs(number);
    }

    @FunctionName("Atn")
    @Signature("Atn(number)")
    @Description("Returns a Double specifying the arctangent of a number.")
    public static double atn(double number) {
        return Math.atan(number);
    }

    @FunctionName("Cos")
    @Signature("Cos(number)")
    @Description("Returns a Double specifying the cosine of an angle.")
    public static double cos(double number) {
        return Math.cos(number);
    }

    @FunctionName("Exp")
    @Signature("Exp(number)")
    @Description(
        "Returns a Double specifying e (the base of natural logarithms) "
        + "raised to a power.")
    public static double exp(double number) {
        return Math.exp(number);
    }

    @FunctionName("Log")
    @Signature("Log(number)")
    @Description(
        "Returns a Double specifying the natural logarithm of a number.")
    public static double log(double number) {
        return Math.log(number);
    }

    // Cannot implement randomize and rnd - we require context to hold the
    // seed

    // public void randomize(Object number)
    // public float rnd(Object number)

    @FunctionName("Round")
    @Signature("Round(number[, numDigitsAfterDecimal])")
    @Description(
        "Returns a number rounded to a specified number of decimal places.")
    public static double round(double number) {
        return Math.round(number);
    }

    @FunctionName("Round")
    @Signature("Round(number[, numDigitsAfterDecimal])")
    @Description(
        "Returns a number rounded to a specified number of decimal places.")
    public static double round(double number, int numDigitsAfterDecimal) {
        if (numDigitsAfterDecimal == 0) {
            return Math.round(number);
        }
        final double shift = Math.pow(10d, numDigitsAfterDecimal);
        double numberScaled = number * shift;
        double resultScaled = Math.round(numberScaled);
        return resultScaled / shift;
    }

    @FunctionName("Sgn")
    @Signature("Sgn(number)")
    @Description("Returns a Variant (Integer) indicating the sign of a number.")
    public static int sgn(double number) {
        // We could use Math.signum(double) from JDK 1.5 onwards.
        return number < 0.0d ? -1 : number > 0.0d ? 1 : 0;
    }

    @FunctionName("Sin")
    @Signature("Sin(number)")
    @Description("Returns a Double specifying the sine of an angle.")
    public static double sin(double number) {
        return Math.sin(number);
    }

    @FunctionName("Sqr")
    @Signature("Sqr(number)")
    @Description("Returns a Double specifying the square root of a number.")
    public static double sqr(double number) {
        return Math.sqrt(number);
    }

    @FunctionName("Tan")
    @Signature("Tan(number)")
    @Description("Returns a Double specifying the tangent of an angle.")
    public static double tan(double number) {
        return Math.tan(number);
    }

    // Strings

    @FunctionName("Asc")
    @Signature("Asc(string)")
    @Description(
        "Returns an Integer representing the character code corresponding to "
        + "the first letter in a string.")
    public static int asc(String string) {
        return string.charAt(0);
    }

    @FunctionName("AscB")
    @Signature("AscB(string)")
    @Description("See Asc.")
    public static int ascB(String string) {
        return (byte) string.charAt(0);
    }

    @FunctionName("AscW")
    @Signature("AscW(string)")
    @Description("See Asc.")
    public static int ascW(String string) {
        return asc(string);
    }

    // public String chr$(int charCode)
    // public String chrB$(int charCode)

    @FunctionName("Chr")
    @Signature("Chr(charcode)")
    @Description(
        "Returns a String containing the character associated with the "
        + "specified character code.")
    public static String chr(int charCode) {
        return new String(new char[] { (char) charCode });
    }

    @FunctionName("ChrB")
    @Signature("ChrB(charcode)")
    @Description("See Chr.")
    public static String chrB(int charCode) {
        return new String(new byte[] { (byte) charCode });
    }

    // public String chrW$(int charCode)

    @FunctionName("ChrW")
    @Signature("ChrW(charcode)")
    @Description("See Chr.")
    public static String chrW(int charCode) {
        return new String(new char[] { (char) charCode });
    }

    // public Object filter(Object sourceArray, String match, boolean include /*
    // default 1 */, int compare /* default BinaryCompare */)
    // public String format$(Object expression, Object format, int
    // firstDayOfWeek /* default Sunday */, int firstWeekOfYear /* default
    // FirstJan1 */)

    @FunctionName("FormatCurrency")
    @Signature(
        "FormatCurrency(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a currency value using the "
        + "currency symbol defined in the system control panel.")
    public static String formatCurrency(Object expression) {
        return formatCurrency(expression, -1, -2, -2, -2);
    }

    @FunctionName("FormatCurrency")
    @Signature(
        "FormatCurrency(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a currency value using the "
        + "currency symbol defined in the system control panel.")
    public static String formatCurrency(
        Object expression,
        int numDigitsAfterDecimal)
    {
        return formatCurrency(expression, numDigitsAfterDecimal, -2, -2, -2);
    }

    @FunctionName("FormatCurrency")
    @Signature(
        "FormatCurrency(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a currency value using the "
        + "currency symbol defined in the system control panel.")
    public static String formatCurrency(
        Object expression,
        int numDigitsAfterDecimal,
        int includeLeadingDigit)
    {
        return formatCurrency(
            expression, numDigitsAfterDecimal,
            includeLeadingDigit, -2, -2);
    }

    @FunctionName("FormatCurrency")
    @Signature(
        "FormatCurrency(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a currency value using the "
        + "currency symbol defined in the system control panel.")
    public static String formatCurrency(
        Object expression,
        int numDigitsAfterDecimal,
        int includeLeadingDigit,
        int useParensForNegativeNumbers)
    {
        return formatCurrency(
            expression,
            numDigitsAfterDecimal,
            includeLeadingDigit,
            useParensForNegativeNumbers,
            -2);
    }

    @FunctionName("FormatCurrency")
    @Signature(
        "FormatCurrency(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a currency value using the "
        + "currency symbol defined in the system control panel.")
    public static String formatCurrency(
        Object expression,
        int numDigitsAfterDecimal,
        int includeLeadingDigit,
        int useParensForNegativeNumbers,
        int groupDigits)
    {
        DecimalFormat format =
            (DecimalFormat) NumberFormat.getCurrencyInstance();
        if (numDigitsAfterDecimal != -1) {
            format.setMaximumFractionDigits(numDigitsAfterDecimal);
            format.setMinimumFractionDigits(numDigitsAfterDecimal);
        }
        if (includeLeadingDigit != -2) {
            if (includeLeadingDigit != 0) {
                format.setMinimumIntegerDigits(1);
            } else {
                format.setMinimumIntegerDigits(0);
            }
        }
        if (useParensForNegativeNumbers != -2) {
            // todo: implement.
            // This will require tweaking of the currency expression
        }
        if (groupDigits != -2) {
            if (groupDigits != 0) {
                format.setGroupingUsed(false);
            } else {
                format.setGroupingUsed(true);
            }
        }
        return format.format(expression);
    }

    @FunctionName("FormatDateTime")
    @Signature("FormatDateTime(Date[,NamedFormat])")
    @Description("Returns an expression formatted as a date or time.")
    public static String formatDateTime(Date date) {
        return formatDateTime(date, 0);
    }

    @FunctionName("FormatDateTime")
    @Signature("FormatDateTime(Date[,NamedFormat])")
    @Description("Returns an expression formatted as a date or time.")
    public static String formatDateTime(
        Date date,
        int namedFormat /* default 0, GeneralDate */)
    {
        // todo: test
        // todo: how do we support VB Constants? Strings or Ints?
        switch (namedFormat) {
            // vbLongDate, 1
            // Display a date using the long date format specified in your
            // computer's regional settings.

        case 1:
            return DateFormat.getDateInstance(DateFormat.LONG).format(date);

            // vbShortDate, 2
            // Display a date using the short date format specified in your
            // computer's regional settings.
        case 2:
            return DateFormat.getDateInstance(DateFormat.SHORT).format(date);

            // vbLongTime, 3
            // Display a time using the time format specified in your computer's
            // regional settings.
        case 3:
            return DateFormat.getTimeInstance(DateFormat.LONG).format(date);

            // vbShortTime, 4
            // Display a time using the 24-hour format (hh:mm).
        case 4:
            return DateFormat.getTimeInstance(DateFormat.SHORT).format(date);

            // vbGeneralDate, 0
            // Display a date and/or time. If there is a date part,
            // display it as a short date. If there is a time part,
            // display it as a long time. If present, both parts are
            // displayed.
            //
            // todo: how do we determine if there is a "time part" in java?
        case 0:
        default:
            return DateFormat.getDateTimeInstance().format(date);
        }
    }

    // Format is implemented with FormatFunDef, third and fourth params are not
    // supported
    // @FunctionName("Format")
    // @Signature("Format(expression[, format[, firstdayofweek[,
    // firstweekofyear]]])")
    // @Description("Returns a Variant (String) containing an expression
    // formatted according to instructions contained in a format expression.")


    @FunctionName("FormatNumber")
    @Signature(
        "FormatNumber(Expression[,NumDigitsAfterDecimal [,IncludeLeadingDigit "
        + "[,UseParensForNegativeNumbers [,GroupDigits]]]])")
    @Description("Returns an expression formatted as a number.")
    public static String formatNumber(Object expression) {
        return formatNumber(expression, -1);
    }

    @FunctionName("FormatNumber")
    @Signature(
        "FormatNumber(Expression[,NumDigitsAfterDecimal [,IncludeLeadingDigit "
        + "[,UseParensForNegativeNumbers [,GroupDigits]]]])")
    @Description("Returns an expression formatted as a number.")
    public static String formatNumber(
        Object expression,
        int numDigitsAfterDecimal)
    {
        return formatNumber(expression, numDigitsAfterDecimal, -1);
    }

    @FunctionName("FormatNumber")
    @Signature(
        "FormatNumber(Expression[,NumDigitsAfterDecimal [,IncludeLeadingDigit "
        + "[,UseParensForNegativeNumbers [,GroupDigits]]]])")
    @Description("Returns an expression formatted as a number.")
    public static String formatNumber(
        Object expression,
        int numDigitsAfterDecimal,
        int includeLeadingDigit)
    {
        return formatNumber(
            expression,
            numDigitsAfterDecimal,
            includeLeadingDigit,
            -1);
    }

    @FunctionName("FormatNumber")
    @Signature(
        "FormatNumber(Expression[,NumDigitsAfterDecimal [,IncludeLeadingDigit "
        + "[,UseParensForNegativeNumbers [,GroupDigits]]]])")
    @Description("Returns an expression formatted as a number.")
    public static String formatNumber(
        Object expression,
        int numDigitsAfterDecimal,
        int includeLeadingDigit,
        int useParensForNegativeNumbers)
    {
        return formatNumber(
            expression,
            numDigitsAfterDecimal,
            includeLeadingDigit,
            useParensForNegativeNumbers, -1);
    }

    @FunctionName("FormatNumber")
    @Signature(
        "FormatNumber(Expression[,NumDigitsAfterDecimal [,IncludeLeadingDigit "
        + "[,UseParensForNegativeNumbers [,GroupDigits]]]])")
    @Description("Returns an expression formatted as a number.")
    public static String formatNumber(
            Object expression,
            int numDigitsAfterDecimal /* default -1 */,
            int includeLeadingDigit /* default usedefault */,
            int useParensForNegativeNumbers /* default UseDefault */,
            int groupDigits /* default UseDefault */)
    {
        NumberFormat format = NumberFormat.getNumberInstance();
        if (numDigitsAfterDecimal != -1) {
            format.setMaximumFractionDigits(numDigitsAfterDecimal);
            format.setMinimumFractionDigits(numDigitsAfterDecimal);
        }

        if (includeLeadingDigit != -1) {
            if (includeLeadingDigit != 0) {
                // true
                format.setMinimumIntegerDigits(1);
            } else {
                format.setMinimumIntegerDigits(0);
            }
        }

        if (useParensForNegativeNumbers != -1) {
            if (useParensForNegativeNumbers != 0) {
                DecimalFormat dformat = (DecimalFormat)format;
                dformat.setNegativePrefix("(");
                dformat.setNegativeSuffix(")");
            } else {
                DecimalFormat dformat = (DecimalFormat)format;
                dformat.setNegativePrefix(
                        "" + dformat.getDecimalFormatSymbols().getMinusSign());
                dformat.setNegativeSuffix("");
            }
        }

        if (groupDigits != -1) {
            format.setGroupingUsed(groupDigits != 0);
        }

        return format.format(expression);
    }

    @FunctionName("FormatPercent")
    @Signature(
        "FormatPercent(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a percentage (multipled by 100) "
        + "with a trailing % character.")
    public static String formatPercent(Object expression) {
        return formatPercent(expression, -1);
    }

    @FunctionName("FormatPercent")
    @Signature(
        "FormatPercent(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a percentage (multipled by 100) "
        + "with a trailing % character.")
    public static String formatPercent(
        // todo: impl & test
        Object expression, int numDigitsAfterDecimal /* default -1 */)
    {
        return formatPercent(expression, numDigitsAfterDecimal, -1);
    }

    @FunctionName("FormatPercent")
    @Signature(
        "FormatPercent(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a percentage (multipled by 100) "
        + "with a trailing % character.")
    public static String formatPercent(
        // todo: impl & test
        Object expression,
        int numDigitsAfterDecimal /* default -1 */,
        int includeLeadingDigit /* default UseDefault */)
    {
        return formatPercent(
            expression,
            numDigitsAfterDecimal,
            includeLeadingDigit,
            -1);
    }

    @FunctionName("FormatPercent")
    @Signature(
        "FormatPercent(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a percentage (multipled by 100) "
        + "with a trailing % character.")
    public static String formatPercent(
        // todo: impl & test
        Object expression,
        int numDigitsAfterDecimal /* default -1 */,
        int includeLeadingDigit /* default UseDefault */,
        int useParensForNegativeNumbers /* default UseDefault */)
    {
        return formatPercent(
            expression, numDigitsAfterDecimal,
            includeLeadingDigit, useParensForNegativeNumbers, -1);
    }

    @FunctionName("FormatPercent")
    @Signature(
        "FormatPercent(Expression[,NumDigitsAfterDecimal "
        + "[,IncludeLeadingDigit [,UseParensForNegativeNumbers "
        + "[,GroupDigits]]]])")
    @Description(
        "Returns an expression formatted as a percentage (multipled by 100) "
        + "with a trailing % character.")
    public static String formatPercent(
        Object expression,
        int numDigitsAfterDecimal /* default -1 */,
        int includeLeadingDigit /* default UseDefault */,
        int useParensForNegativeNumbers /* default UseDefault */,
        int groupDigits /* default UseDefault */)
    {
        NumberFormat format = NumberFormat.getPercentInstance();
        if (numDigitsAfterDecimal != -1) {
            format.setMaximumFractionDigits(numDigitsAfterDecimal);
            format.setMinimumFractionDigits(numDigitsAfterDecimal);
        }

        if (includeLeadingDigit != -1) {
            if (includeLeadingDigit != 0) {
                // true
                format.setMinimumIntegerDigits(1);
            } else {
                format.setMinimumIntegerDigits(0);
            }
        }

        if (useParensForNegativeNumbers != -1) {
            if (useParensForNegativeNumbers != 0) {
                DecimalFormat dformat = (DecimalFormat)format;
                dformat.setNegativePrefix("(");
                dformat.setNegativeSuffix(
                    "" + dformat.getDecimalFormatSymbols().getPercent() ")");
            } else {
                DecimalFormat dformat = (DecimalFormat)format;
                dformat.setNegativePrefix(
                        "" + dformat.getDecimalFormatSymbols().getMinusSign());
                dformat.setNegativeSuffix(
                        "" + dformat.getDecimalFormatSymbols().getPercent());
            }
        }

        if (groupDigits != -1) {
            format.setGroupingUsed(groupDigits != 0);
        }

        return format.format(expression);
    }

    // public Object inStrB(Object start, Object string1, Object string2, int
    // compare /* default BinaryCompare */)

    @FunctionName("InStr")
    @Signature("InStr([start, ]stringcheck, stringmatch[, compare])")
    @Description(
        "Returns a Variant (Long) specifying the position of the first "
        + "occurrence of one string within another.")
    public static int inStr(String stringCheck, String stringMatch) {
        return inStr(1, stringCheck, stringMatch, 0);
    }

    @FunctionName("InStr")
    @Signature("InStr([start, ]stringcheck, stringmatch[, compare])")
    @Description(
        "Returns the position of an occurrence of one string within "
        + "another.")
    public static int inStr(
        int start /* default 1 */,
        String stringCheck,
        String stringMatch)
    {
        return inStr(start, stringCheck, stringMatch, 0);
    }

    @FunctionName("InStr")
    @Signature("InStr([start, ]stringcheck, stringmatch[, compare])")
    @Description(
        "Returns the position of an occurrence of one string within "
        + "another.")
    public static int inStr(
        int start /* default 1 */,
        String stringCheck,
        String stringMatch,
        int compare /* default BinaryCompare */)
    {
        // todo: implement binary vs. text compare
        if (start == 0 || start < -1) {
            throw new InvalidArgumentException(
                    "start must be -1 or a location in the string to start");
        }
        if (start != -1) {
            return stringCheck.indexOf(stringMatch, start - 1) + 1;
        } else {
            return stringCheck.indexOf(stringMatch) + 1;
        }
    }

    @FunctionName("InStrRev")
    @Signature("InStrRev(stringcheck, stringmatch[, start[, compare]])")
    @Description(
        "Returns the position of an occurrence of one string within another, "
        + "from the end of string.")
    public static int inStrRev(String stringCheck, String stringMatch) {
        return inStrRev(stringCheck, stringMatch, -1);
    }

    @FunctionName("InStrRev")
    @Signature("InStrRev(stringcheck, stringmatch[, start[, compare]])")
    @Description(
        "Returns the position of an occurrence of one string within another, "
        + "from the end of string.")
    public static int inStrRev(
        String stringCheck,
        String stringMatch,
        int start /* default -1 */)
    {
        return inStrRev(stringCheck, stringMatch, start, 0);
    }

    @FunctionName("InStrRev")
    @Signature("InStrRev(stringcheck, stringmatch[, start[, compare]])")
    @Description(
        "Returns the position of an occurrence of one string within another, "
        + "from the end of string.")
    public static int inStrRev(
        String stringCheck,
        String stringMatch,
        int start /* default -1 */,
        int compare /* default BinaryCompare */)
    {
        // todo: implement binary vs. text compare
        if (start == 0 || start < -1) {
            throw new InvalidArgumentException(
                    "start must be -1 or a location in the string to start");
        }
        if (start != -1) {
            return stringCheck.lastIndexOf(stringMatch, start - 1) + 1;
        } else {
            return stringCheck.lastIndexOf(stringMatch) + 1;
        }
    }

    // public String join(Object sourceArray, Object delimiter)

    @FunctionName("LCase")
    @Signature("LCase(string)")
    @Description("Returns a String that has been converted to lowercase.")
    public static String lCase(String string) {
        return string.toLowerCase();
    }

    // public Object lCase$(Object string)
    // public String lTrim$(String string)

    @FunctionName("LTrim")
    @Signature("LTrim(string)")
    @Description(
        "Returns a Variant (String) containing a copy of a specified string "
        + "without leading spaces.")
    public static String lTrim(String string) {
        int i = 0, n = string.length();
        while (i < n) {
            if (string.charAt(i) > ' ') {
                break;
            }
            i++;
        }
        return string.substring(i);
    }

    // public String left$(String string, int length)
    // public String leftB$(String string, int length)
    // public Object leftB(Object string, int length)

    @FunctionName("Left")
    @Signature("Left(string, length)")
    @Description(
        "Returns a specified number of characters from the left side of a "
        + "string.")
    public static String left(String string, int length) {
        final int stringLength = string.length();
        if (length >= stringLength) {
            return string;
        }
        return string.substring(0, length);
    }

    // public Object lenB(Object expression)

    // len is already implemented in BuiltinFunTable... defer

    // @FunctionName("Len")
    // @Signature("Len(String)")
    // @Description("Returns a Long containing the number of characters in a
    // string.")
    // public static int len(String expression) {
    // return expression.length();
    // }

    // public String mid$(String string, int start, Object length)
    // public String midB$(String string, int start, Object length)
    // public Object midB(Object string, int start, Object length)

    @FunctionName("Mid")
    @Signature("Mid(value, beginIndex[, length])")
    @Description("Returns a specified number of characters from a string.")
    public static String mid(String value, int beginIndex) {
        // If we used 'value.length() - beginIndex' as the default value for
        // length, we'd have problems if beginIndex is huge;
        // so 'value.length()' looks like an overestimate - but will always
        // return the correct result.
        final int length = value.length();
        return mid(value, beginIndex, length);
    }

    @FunctionName("Mid")
    @Signature("Mid(value, beginIndex[, length])")
    @Description("Returns a specified number of characters from a string.")
    public static String mid(String value, int beginIndex, int length) {
        // Arguments are 1-based. Spec says that the function gives an error if
        // Start <= 0 or Length < 0.
        if (beginIndex <= 0) {
            throw new InvalidArgumentException(
                "Invalid parameter. "
                + "Start parameter of Mid function must be positive");
        }
        if (length < 0) {
            throw new InvalidArgumentException(
                "Invalid parameter. "
                + "Length parameter of Mid function must be non-negative");
        }

        if (beginIndex > value.length()) {
            return "";
        }

        // Shift from 1-based to 0-based.
        --beginIndex;
        int endIndex = beginIndex + length;
        return endIndex >= value.length() ? value.substring(beginIndex) : value
                .substring(beginIndex, endIndex);
    }

    @FunctionName("MonthName")
    @Signature("MonthName(month, abbreviate)")
    @Description("Returns a string indicating the specified month.")
    public static String monthName(int month, boolean abbreviate) {
        // VB months are 1-based, Java months are 0-based
        --month;
        return (abbreviate ? getDateFormatSymbols().getShortMonths()
                : getDateFormatSymbols().getMonths())[month];
    }

    /**
     * Returns an instance of {@link DateFormatSymbols} for the current locale.
     *
     * <p>
     * Todo: inherit locale from connection.
     *
     * @return a DateFormatSymbols object
     */
    private static DateFormatSymbols getDateFormatSymbols() {
        // We would use DataFormatSymbols.getInstance(), but it is only
        // available from JDK 1.6 onwards.
        return DATE_FORMAT_SYMBOLS;
    }

    // public String rTrim$(String string)

    @FunctionName("RTrim")
    @Signature("RTrim(string)")
    @Description(
        "Returns a Variant (String) containing a copy of a specified string "
        + "without trailing spaces.")
    public static String rTrim(String string) {
        int i = string.length() - 1;
        while (i >= 0) {
            if (string.charAt(i) > ' ') {
                break;
            }
            i--;
        }
        return string.substring(0, i + 1);
    }

    @FunctionName("Replace")
    @Signature(
        "Replace(expression, find, replace[, start[, count[, compare]]])")
    @Description(
        "Returns a string in which a specified substring has been replaced "
        + "with another substring a specified number of times.")
    public static String replace(
        String expression,
        String find,
        String replace,
        int start,
        int count,
        int compare)
    {
        // compare is currently ignored
        Util.discard(compare);
        return _replace(expression, find, replace, start, count);
    }

    @FunctionName("Replace")
    @Signature(
        "Replace(expression, find, replace[, start[, count[, compare]]])")
    @Description(
        "Returns a string in which a specified substring has been replaced "
        + "with another substring a specified number of times.")
    public static String replace(
        String expression,
        String find,
        String replace,
        int start,
        int count)
    {
        return _replace(expression, find, replace, start, count);
    }

    @FunctionName("Replace")
    @Signature(
        "Replace(expression, find, replace[, start[, count[, compare]]])")
    @Description(
        "Returns a string in which a specified substring has been replaced "
        + "with another substring a specified number of times.")
    public static String replace(
        String expression,
        String find,
        String replace,
        int start)
    {
        return _replace(expression, find, replace, start, -1);
    }

    @FunctionName("Replace")
    @Signature(
        "Replace(expression, find, replace[, start[, count[, compare]]])")
    @Description("")
    public static String replace(
        String expression,
        String find,
        String replace)
    {
        // compare is currently ignored
        return _replace(expression, find, replace, 1, -1);
    }

    private static String _replace(
        String expression,
        String find,
        String replace,
        int start /* default 1 */,
        int count /* default -1 */)
    {
        final StringBuilder buf = new StringBuilder(expression);
        int i = 0;
        int pos = start - 1;
        while (true) {
            if (i++ == count) {
                break;
            }
            final int j = buf.indexOf(find, pos);
            if (j == -1) {
                break;
            }
            buf.replace(j, j + find.length(), replace);
            pos = j + replace.length();
        }
        return buf.toString();
    }

    // public String right$(String string, int length)
    // public String rightB$(String string, int length)
    // public Object rightB(Object string, int length)

    @FunctionName("Right")
    @Signature("Right(string, length)")
    @Description(
        "Returns a Variant (String) containing a specified number of "
        + "characters from the right side of a string.")
    public static String right(String string, int length) {
        final int stringLength = string.length();
        if (length >= stringLength) {
            return string;
        }
        return string.substring(stringLength - length, stringLength);
    }

    // public String space$(int number)

    @FunctionName("Space")
    @Signature("Space(number)")
    @Description(
        "Returns a Variant (String) consisting of the specified number of "
        + "spaces.")
    public static String space(int number) {
        return string(number, ' ');
    }

    // public Object split(String expression, Object delimiter, int limit /*
    // default -1 */, int compare /* default BinaryCompare */)

    @FunctionName("StrComp")
    @Signature("StrComp(string1, string2[, compare])")
    @Description(
        "Returns a Variant (Integer) indicating the result of a string "
        + "comparison.")
    public static int strComp(String string1, String string2) {
        return strComp(string1, string2, 0);
    }

    @FunctionName("StrComp")
    @Signature("StrComp(string1, string2[, compare])")
    @Description(
        "Returns a Variant (Integer) indicating the result of a string "
        + "comparison.")
    public static int strComp(
        String string1,
        String string2,
        int compare /* default BinaryCompare */)
    {
        // Note: compare is currently ignored
        // Wrapper already checked whether args are null
        assert string1 != null;
        assert string2 != null;
        return string1.compareTo(string2);
    }

    // public Object strConv(Object string, StrConv conversion, int localeID)

    @FunctionName("StrReverse")
    @Signature("StrReverse(string)")
    @Description(
        "Returns a string in which the character order of a specified string "
        + "is reversed.")
    public static String strReverse(String expression) {
        final char[] chars = expression.toCharArray();
        for (int i = 0, j = chars.length - 1; i < j; i++, j--) {
            char c = chars[i];
            chars[i] = chars[j];
            chars[j] = c;
        }
        return new String(chars);
    }

    // public String string$(int number, Object character)

    @FunctionName("String")
    @Signature("String(number, character)")
    @Description("")
    public static String string(int number, char character) {
        if (character == 0) {
            return "";
        }
        final char[] chars = new char[number];
        Arrays.fill(chars, (char) (character % 256));
        return new String(chars);
    }

    // public String trim$(String string)

    @FunctionName("Trim")
    @Signature("Trim(string)")
    @Description(
        "Returns a Variant (String) containing a copy of a specified string "
        + "without leading and trailing spaces.")
    public static String trim(String string) {
        // JDK has a method for trim, but not ltrim or rtrim
        return string.trim();
    }

    // ucase is already implemented in BuiltinFunTable... defer

    // public String uCase$(String string)

//    @FunctionName("UCase")
//    @Signature("UCase(string)")
//    @Description("Returns a String that has been converted to uppercase.")
//    public String uCase(String string) {
//        return string.toUpperCase();
//    }

    // TODO: should use connection's locale to determine first day of week,
    // not the JVM's default

    @FunctionName("WeekdayName")
    @Signature("WeekdayName(weekday, abbreviate, firstdayofweek)")
    @Description("Returns a string indicating the specified day of the week.")
    public static String weekdayName(
        int weekday,
        boolean abbreviate,
        int firstDayOfWeek)
    {
        // Java and VB agree: SUNDAY = 1, ... SATURDAY = 7
        final Calendar calendar = Calendar.getInstance();
        if (firstDayOfWeek == 0) {
            firstDayOfWeek = calendar.getFirstDayOfWeek();
        }
        // compensate for start of week
        weekday += (firstDayOfWeek - 1);
        // bring into range 1..7
        weekday = (weekday - 1) % 7 + 1;
        if (weekday <= 0) {
            // negative numbers give negative modulo
            weekday += 7;
        }
        return
            (abbreviate
             ? getDateFormatSymbols().getShortWeekdays()
             : getDateFormatSymbols().getWeekdays())
            [weekday];
    }

    // Misc

    // public Object array(Object argList)
    // public String input$(int number, int fileNumber)
    // public String inputB$(int number, int fileNumber)
    // public Object inputB(int number, int fileNumber)
    // public Object input(int number, int fileNumber)
    // public void width(int fileNumber, int width)

    // ~ Inner classes

    private enum Interval {
        yyyy("Year", Calendar.YEAR),
        q("Quarter", -1),
        m("Month", Calendar.MONTH),
        y("Day of year", Calendar.DAY_OF_YEAR),
        d("Day", Calendar.DAY_OF_MONTH),
        w("Weekday", Calendar.DAY_OF_WEEK),
        ww("Week", Calendar.WEEK_OF_YEAR),
        h("Hour", Calendar.HOUR_OF_DAY),
        n("Minute", Calendar.MINUTE),
        s("Second", Calendar.SECOND);

        private final int dateField;

        Interval(String desc, int dateField) {
            Util.discard(desc);
            this.dateField = dateField;
        }

        void add(Calendar calendar, int amount) {
            switch (this) {
            case q:
                calendar.add(Calendar.MONTH, amount * 3);
                break;
            default:
                calendar.add(dateField, amount);
                break;
            }
        }

        Calendar floor(Calendar calendar) {
            Calendar calendar2 = Calendar.getInstance();
            calendar2.setTime(calendar.getTime());
            floorInplace(calendar2);
            return calendar2;
        }

        private void floorInplace(Calendar calendar) {
            switch (this) {
            case yyyy:
                calendar.set(Calendar.DAY_OF_YEAR, 1);
                d.floorInplace(calendar);
                break;
            case q:
                int month = calendar.get(Calendar.MONTH);
                month -= month % 3;
                calendar.set(Calendar.MONTH, month);
                calendar.set(Calendar.DAY_OF_MONTH, 1);
                d.floorInplace(calendar);
                break;
            case m:
                calendar.set(Calendar.DAY_OF_MONTH, 1);
                d.floorInplace(calendar);
                break;
            case w:
                final int dow = calendar.get(Calendar.DAY_OF_WEEK);
                final int firstDayOfWeek = calendar.getFirstDayOfWeek();
                if (dow == firstDayOfWeek) {
                    // nothing to do
                } else if (dow > firstDayOfWeek) {
                    final int roll = firstDayOfWeek - dow;
                    assert roll < 0;
                    calendar.roll(Calendar.DAY_OF_WEEK, roll);
                } else {
                    final int roll = firstDayOfWeek - dow - 7;
                    assert roll < 0;
                    calendar.roll(Calendar.DAY_OF_WEEK, roll);
                }
                d.floorInplace(calendar);
                break;
            case y:
            case d:
                calendar.set(Calendar.HOUR_OF_DAY, 0);
                calendar.set(Calendar.MINUTE, 0);
                calendar.set(Calendar.SECOND, 0);
                calendar.set(Calendar.MILLISECOND, 0);
                break;
            case h:
                calendar.set(Calendar.MINUTE, 0);
                calendar.set(Calendar.SECOND, 0);
                calendar.set(Calendar.MILLISECOND, 0);
                break;
            case n:
                calendar.set(Calendar.SECOND, 0);
                calendar.set(Calendar.MILLISECOND, 0);
                break;
            case s:
                calendar.set(Calendar.MILLISECOND, 0);
                break;
            }
        }

        int diff(Calendar calendar1, Calendar calendar2, int firstDayOfWeek) {
            switch (this) {
            case q:
                return m.diff(calendar1, calendar2, firstDayOfWeek) / 3;
            default:
                return floor(calendar1).get(dateField)
                        - floor(calendar2).get(dateField);
            }
        }

        int datePart(Calendar calendar) {
            switch (this) {
            case q:
                return (m.datePart(calendar) + 2) / 3;
            case m:
                return calendar.get(dateField) + 1;
            case w:
                int dayOfWeek = calendar.get(dateField);
                dayOfWeek -= (calendar.getFirstDayOfWeek() - 1);
                dayOfWeek = dayOfWeek % 7;
                if (dayOfWeek <= 0) {
                    dayOfWeek += 7;
                }
                return dayOfWeek;
            default:
                return calendar.get(dateField);
            }
        }
    }

    private enum FirstWeekOfYear {
        vbUseSystem(
            0, "Use the NLS API setting."),

        vbFirstJan1(
            1,
            "Start with week in which January 1 occurs (default)."),

        vbFirstFourDays(
            2,
            "Start with the first week that has at least four days in the new year."),

        vbFirstFullWeek(
            3,
            "Start with first full week of the year.");

        FirstWeekOfYear(int code, String desc) {
            assert code == ordinal();
            assert desc != null;
        }

        void apply(Calendar calendar) {
            switch (this) {
            case vbUseSystem:
                break;
            case vbFirstJan1:
                calendar.setMinimalDaysInFirstWeek(1);
                break;
            case vbFirstFourDays:
                calendar.setMinimalDaysInFirstWeek(4);
                break;
            case vbFirstFullWeek:
                calendar.setMinimalDaysInFirstWeek(7);
                break;
            }
        }
    }
}

// End Vba.java
TOP

Related Classes of mondrian.olap.fun.vba.Vba

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.