Rewrite pretty much all variable and parameter handling. Makes global and local variables be handled differently. Global variables are no longer part of the variables stack, and are lazily evaluated. Fixes bugs 117658, 92929 and some unfiled bugs and fixes remaining parts of bugs 83651 and 96802. Patch by sicking, r=Pike sr=bz. Fix for bug 156464: fix rounding problems in module, allow patterns without any '0's in the integer part and fix problems with grouping for standalone. Patch by sicking, r=Pike sr=bz. Fix for bug 157340 (Probable bugs in extensions/transformiix/source/base/txMozillaString.h). Patch by peterv, r=Pike, sr=bz. Fix for bug 146967 (Clean up Transformiix strings). Patch by peterv, r=sicking, sr=jst. Fix for bug 156464 (Remove static strings from Transformiix). Patch by peterv, r=Pike, sr=jst. git-svn-id: svn://10.0.0.236/trunk@126495 18797224-902f-48f8-a5cc-f745e15eee43
517 lines
15 KiB
C++
517 lines
15 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is the TransforMiiX XSLT processor.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Jonas Sicking.
|
|
* Portions created by the Initial Developer are Copyright (C) 2001
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Jonas Sicking <sicking@bigfoot.com>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "XSLTFunctions.h"
|
|
#include "ProcessorState.h"
|
|
#include "primitives.h"
|
|
#include "txAtoms.h"
|
|
#include "txIXPathContext.h"
|
|
#include <math.h>
|
|
|
|
#ifndef TX_EXE
|
|
#include "prdtoa.h"
|
|
#else
|
|
#include <stdio.h>
|
|
#endif
|
|
|
|
const UNICODE_CHAR txFormatNumberFunctionCall::FORMAT_QUOTE = '\'';
|
|
|
|
/*
|
|
* FormatNumberFunctionCall
|
|
* A representation of the XSLT additional function: format-number()
|
|
*/
|
|
|
|
/*
|
|
* Creates a new format-number function call
|
|
*/
|
|
txFormatNumberFunctionCall::txFormatNumberFunctionCall(ProcessorState* aPs,
|
|
Node* aQNameResolveNode)
|
|
: mPs(aPs),
|
|
mQNameResolveNode(aQNameResolveNode)
|
|
{
|
|
}
|
|
|
|
/*
|
|
* Evaluates this Expr based on the given context node and processor state
|
|
* @param context the context node for evaluation of this Expr
|
|
* @param cs the ContextState containing the stack information needed
|
|
* for evaluation
|
|
* @return the result of the evaluation
|
|
*/
|
|
ExprResult* txFormatNumberFunctionCall::evaluate(txIEvalContext* aContext)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
if (!requireParams(2, 3, aContext))
|
|
return new StringResult();
|
|
|
|
// Get number and format
|
|
txListIterator iter(¶ms);
|
|
|
|
double value;
|
|
String formatStr;
|
|
txExpandedName formatName;
|
|
|
|
value = evaluateToNumber((Expr*)iter.next(), aContext);
|
|
evaluateToString((Expr*)iter.next(), aContext, formatStr);
|
|
if (iter.hasNext()) {
|
|
String formatQName;
|
|
evaluateToString((Expr*)iter.next(), aContext, formatQName);
|
|
rv = formatName.init(formatQName, mQNameResolveNode, MB_FALSE);
|
|
if (NS_FAILED(rv))
|
|
formatName.mNamespaceID = kNameSpaceID_Unknown;
|
|
}
|
|
|
|
txDecimalFormat* format = mPs->getDecimalFormat(formatName);
|
|
if (!format) {
|
|
String err("unknown decimal format for: ");
|
|
toString(err);
|
|
aContext->receiveError(err, NS_ERROR_XPATH_INVALID_ARG);
|
|
return new StringResult(err);
|
|
}
|
|
|
|
// Special cases
|
|
if (Double::isNaN(value))
|
|
return new StringResult(format->mNaN);
|
|
|
|
if (value == Double::POSITIVE_INFINITY)
|
|
return new StringResult(format->mInfinity);
|
|
|
|
if (value == Double::NEGATIVE_INFINITY) {
|
|
String res;
|
|
res.append(format->mMinusSign);
|
|
res.append(format->mInfinity);
|
|
return new StringResult(res);
|
|
}
|
|
|
|
// Value is a normal finite number
|
|
String prefix;
|
|
String suffix;
|
|
int minIntegerSize=0;
|
|
int minFractionSize=0;
|
|
int maxFractionSize=0;
|
|
int multiplier=1;
|
|
int groupSize=-1;
|
|
|
|
PRUint32 pos = 0;
|
|
PRUint32 formatLen = formatStr.length();
|
|
MBool inQuote;
|
|
|
|
// Get right subexpression
|
|
inQuote = MB_FALSE;
|
|
if (Double::isNeg(value)) {
|
|
while (pos < formatLen &&
|
|
(inQuote ||
|
|
formatStr.charAt(pos) != format->mPatternSeparator)) {
|
|
if (formatStr.charAt(pos) == FORMAT_QUOTE)
|
|
inQuote = !inQuote;
|
|
pos++;
|
|
}
|
|
|
|
if (pos == formatLen) {
|
|
pos = 0;
|
|
prefix.append(format->mMinusSign);
|
|
}
|
|
else
|
|
pos++;
|
|
}
|
|
|
|
// Parse the format string
|
|
FormatParseState pState = Prefix;
|
|
inQuote = MB_FALSE;
|
|
|
|
UNICODE_CHAR c = 0;
|
|
while (pos < formatLen && pState != Finished) {
|
|
c=formatStr.charAt(pos++);
|
|
|
|
switch (pState) {
|
|
|
|
case Prefix:
|
|
case Suffix:
|
|
if (!inQuote) {
|
|
if (c == format->mPercent) {
|
|
if (multiplier == 1)
|
|
multiplier = 100;
|
|
else {
|
|
String err(INVALID_PARAM_VALUE);
|
|
toString(err);
|
|
aContext->receiveError(err,
|
|
NS_ERROR_XPATH_EVAL_FAILED);
|
|
return new StringResult(err);
|
|
}
|
|
}
|
|
else if (c == format->mPerMille) {
|
|
if (multiplier == 1)
|
|
multiplier = 1000;
|
|
else {
|
|
String err(INVALID_PARAM_VALUE);
|
|
toString(err);
|
|
aContext->receiveError(err,
|
|
NS_ERROR_XPATH_EVAL_FAILED);
|
|
return new StringResult(err);
|
|
}
|
|
}
|
|
else if (c == format->mDecimalSeparator ||
|
|
c == format->mGroupingSeparator ||
|
|
c == format->mZeroDigit ||
|
|
c == format->mDigit ||
|
|
c == format->mPatternSeparator) {
|
|
pState = pState == Prefix ? IntDigit : Finished;
|
|
pos--;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (c == FORMAT_QUOTE)
|
|
inQuote = !inQuote;
|
|
else if (pState == Prefix)
|
|
prefix.append(c);
|
|
else
|
|
suffix.append(c);
|
|
break;
|
|
|
|
case IntDigit:
|
|
if (c == format->mGroupingSeparator)
|
|
groupSize=0;
|
|
else if (c == format->mDigit) {
|
|
if (groupSize >= 0)
|
|
groupSize++;
|
|
}
|
|
else {
|
|
pState = IntZero;
|
|
pos--;
|
|
}
|
|
break;
|
|
|
|
case IntZero:
|
|
if (c == format->mGroupingSeparator)
|
|
groupSize = 0;
|
|
else if (c == format->mZeroDigit) {
|
|
if (groupSize >= 0)
|
|
groupSize++;
|
|
minIntegerSize++;
|
|
}
|
|
else if (c == format->mDecimalSeparator) {
|
|
pState = FracZero;
|
|
}
|
|
else {
|
|
pState = Suffix;
|
|
pos--;
|
|
}
|
|
break;
|
|
|
|
case FracZero:
|
|
if (c == format->mZeroDigit) {
|
|
maxFractionSize++;
|
|
minFractionSize++;
|
|
}
|
|
else {
|
|
pState = FracDigit;
|
|
pos--;
|
|
}
|
|
break;
|
|
|
|
case FracDigit:
|
|
if (c == format->mDigit)
|
|
maxFractionSize++;
|
|
else {
|
|
pState = Suffix;
|
|
pos--;
|
|
}
|
|
break;
|
|
|
|
case Finished:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Did we manage to parse the entire formatstring and was it valid
|
|
if ((c != format->mPatternSeparator && pos < formatLen) ||
|
|
inQuote ||
|
|
groupSize == 0) {
|
|
String err(INVALID_PARAM_VALUE);
|
|
toString(err);
|
|
aContext->receiveError(err,
|
|
NS_ERROR_XPATH_EVAL_FAILED);
|
|
return new StringResult(err);
|
|
}
|
|
|
|
|
|
/*
|
|
* FINALLY we're done with the parsing
|
|
* now build the result string
|
|
*/
|
|
|
|
value = fabs(value) * multiplier;
|
|
|
|
// Prefix
|
|
String res(prefix);
|
|
|
|
#ifdef TX_EXE
|
|
|
|
int bufsize;
|
|
if (value > 1)
|
|
bufsize = (int)log10(value) + 1;
|
|
else
|
|
bufsize = 1;
|
|
|
|
if (bufsize < minIntegerSize)
|
|
bufsize = minIntegerSize;
|
|
|
|
bufsize += maxFractionSize + 3; // decimal separator + ending null +
|
|
// rounding safety
|
|
|
|
char* buf = new char[bufsize];
|
|
if (!buf) {
|
|
//XXX ErrorReport: out of memory
|
|
return new StringResult;
|
|
}
|
|
|
|
int totalSize = minIntegerSize + maxFractionSize;
|
|
if (maxFractionSize > 0)
|
|
totalSize++; //to account for decimal point
|
|
|
|
char formatBuf[30];
|
|
sprintf(formatBuf, "%%0%d.%df", totalSize, maxFractionSize);
|
|
|
|
sprintf(buf, formatBuf, value);
|
|
|
|
// Find decimalseparator for grouping
|
|
int intDigits;
|
|
for (intDigits = 0; buf[intDigits] && buf[intDigits] != '.'; intDigits++);
|
|
|
|
if (groupSize < 0)
|
|
groupSize = intDigits * 2; //to simplify grouping
|
|
|
|
// Integer digits
|
|
int i;
|
|
for (i = 0; i < intDigits; ++i) {
|
|
if ((intDigits-i)%groupSize == 0 && i != 0)
|
|
res.append(format->mGroupingSeparator);
|
|
|
|
res.append((UNICODE_CHAR)(buf[i] - '0' + format->mZeroDigit));
|
|
}
|
|
|
|
// Fractions
|
|
MBool printDeci = MB_TRUE;
|
|
int extraZeros = 0;
|
|
|
|
if (buf[i])
|
|
i++; // skip decimal separator
|
|
|
|
for (; buf[i]; i++) {
|
|
if (i-intDigits-1 >= minFractionSize && buf[i] == '0') {
|
|
extraZeros++;
|
|
}
|
|
else {
|
|
if (printDeci) {
|
|
res.append(format->mDecimalSeparator);
|
|
printDeci = MB_FALSE;
|
|
}
|
|
while (extraZeros) {
|
|
res.append(format->mZeroDigit);
|
|
extraZeros--;
|
|
}
|
|
|
|
res.append((UNICODE_CHAR)(buf[i] - '0' + format->mZeroDigit));
|
|
}
|
|
}
|
|
|
|
if (!intDigits && printDeci) {
|
|
// If we havn't added any characters we add a '0'
|
|
// This can only happen for formats like '##.##'
|
|
res.append(format->mZeroDigit);
|
|
}
|
|
|
|
delete [] buf;
|
|
|
|
|
|
#else // TX_EXE
|
|
|
|
int bufsize;
|
|
if (value > 1)
|
|
bufsize = (int)log10(value) + 30;
|
|
else
|
|
bufsize = 1 + 30;
|
|
|
|
char* buf = new char[bufsize];
|
|
if (!buf) {
|
|
//XXX ErrorReport: out of memory
|
|
return new StringResult;
|
|
}
|
|
|
|
PRIntn bufIntDigits, sign;
|
|
char* endp;
|
|
PR_dtoa(value, 0, 0, &bufIntDigits, &sign, &endp, buf, bufsize-1);
|
|
|
|
int buflen = endp - buf;
|
|
int intDigits;
|
|
intDigits = bufIntDigits > minIntegerSize ? bufIntDigits : minIntegerSize;
|
|
|
|
if (groupSize < 0)
|
|
groupSize = intDigits + 10; //to simplify grouping
|
|
|
|
// XXX We shouldn't use SetLength.
|
|
res.getNSString().SetLength(res.length() +
|
|
intDigits + // integer digits
|
|
1 + // decimal separator
|
|
maxFractionSize + // fractions
|
|
(intDigits-1)/groupSize); // group separators
|
|
|
|
PRInt32 i = bufIntDigits + maxFractionSize - 1;
|
|
MBool carry = (i+1 < buflen) && (buf[i+1] >= '5');
|
|
MBool hasFraction = MB_FALSE;
|
|
|
|
PRUint32 resPos = res.length()-1;
|
|
|
|
// Fractions
|
|
for (; i >= bufIntDigits; --i) {
|
|
int digit;
|
|
if (i >= buflen || i < 0) {
|
|
digit = 0;
|
|
}
|
|
else {
|
|
digit = buf[i] - '0';
|
|
}
|
|
|
|
if (carry) {
|
|
digit = (digit + 1) % 10;
|
|
carry = digit == 0;
|
|
}
|
|
|
|
if (hasFraction || digit != 0 || i < bufIntDigits+minFractionSize) {
|
|
hasFraction = MB_TRUE;
|
|
res.replace(resPos--,
|
|
(UNICODE_CHAR)(digit + format->mZeroDigit));
|
|
}
|
|
else {
|
|
res.truncate(resPos--);
|
|
}
|
|
}
|
|
|
|
// Decimal separator
|
|
if (hasFraction) {
|
|
res.replace(resPos--, format->mDecimalSeparator);
|
|
}
|
|
else {
|
|
res.truncate(resPos--);
|
|
}
|
|
|
|
// Integer digits
|
|
for (i = 0; i < intDigits; ++i) {
|
|
int digit;
|
|
if (bufIntDigits-i-1 >= buflen || bufIntDigits-i-1 < 0) {
|
|
digit = 0;
|
|
}
|
|
else {
|
|
digit = buf[bufIntDigits-i-1] - '0';
|
|
}
|
|
|
|
if (carry) {
|
|
digit = (digit + 1) % 10;
|
|
carry = digit == 0;
|
|
}
|
|
|
|
if (i != 0 && i%groupSize == 0) {
|
|
res.replace(resPos--, format->mGroupingSeparator);
|
|
}
|
|
|
|
res.replace(resPos--, (UNICODE_CHAR)(digit + format->mZeroDigit));
|
|
}
|
|
|
|
if (carry) {
|
|
if (i%groupSize == 0) {
|
|
res.insert(resPos + 1, format->mGroupingSeparator);
|
|
}
|
|
res.insert(resPos + 1, (UNICODE_CHAR)(1 + format->mZeroDigit));
|
|
}
|
|
|
|
if (!hasFraction && !intDigits && !carry) {
|
|
// If we havn't added any characters we add a '0'
|
|
// This can only happen for formats like '##.##'
|
|
res.append(format->mZeroDigit);
|
|
}
|
|
|
|
delete [] buf;
|
|
|
|
#endif // TX_EXE
|
|
|
|
// Build suffix
|
|
res.append(suffix);
|
|
|
|
return new StringResult(res);
|
|
} //-- evaluate
|
|
|
|
nsresult txFormatNumberFunctionCall::getNameAtom(txAtom** aAtom)
|
|
{
|
|
*aAtom = txXSLTAtoms::formatNumber;
|
|
TX_ADDREF_ATOM(*aAtom);
|
|
return NS_OK;
|
|
}
|
|
|
|
/*
|
|
* txDecimalFormat
|
|
* A representation of the XSLT element <xsl:decimal-format>
|
|
*/
|
|
|
|
txDecimalFormat::txDecimalFormat() : mInfinity("Infinity"),
|
|
mNaN("NaN")
|
|
{
|
|
mDecimalSeparator = '.';
|
|
mGroupingSeparator = ',';
|
|
mMinusSign = '-';
|
|
mPercent = '%';
|
|
mPerMille = 0x2030;
|
|
mZeroDigit = '0';
|
|
mDigit = '#';
|
|
mPatternSeparator = ';';
|
|
}
|
|
|
|
MBool txDecimalFormat::isEqual(txDecimalFormat* other)
|
|
{
|
|
return mDecimalSeparator == other->mDecimalSeparator &&
|
|
mGroupingSeparator == other->mGroupingSeparator &&
|
|
mInfinity.isEqual(other->mInfinity) &&
|
|
mMinusSign == other->mMinusSign &&
|
|
mNaN.isEqual(other->mNaN) &&
|
|
mPercent == other->mPercent &&
|
|
mPerMille == other->mPerMille &&
|
|
mZeroDigit == other->mZeroDigit &&
|
|
mDigit == other->mDigit &&
|
|
mPatternSeparator == other->mPatternSeparator;
|
|
}
|