// Copyright 2014 PDFium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
#include "PublicMethods.h"
#include <algorithm>
#include "Field.h"
#include "JS_Context.h"
#include "JS_Define.h"
#include "JS_EventHandler.h"
#include "JS_Object.h"
#include "JS_Runtime.h"
#include "JS_Value.h"
#include "color.h"
#include "core/include/fxcrt/fx_ext.h"
#include "fpdfsdk/include/fsdk_mgr.h" // For CPDFDoc_Environment.
#include "fpdfsdk/include/javascript/IJavaScript.h"
#include "resource.h"
#include "util.h"
#define DOUBLE_CORRECT 0.000000000000001
BEGIN_JS_STATIC_GLOBAL_FUN(CJS_PublicMethods)
JS_STATIC_GLOBAL_FUN_ENTRY(AFNumber_Format)
JS_STATIC_GLOBAL_FUN_ENTRY(AFNumber_Keystroke)
JS_STATIC_GLOBAL_FUN_ENTRY(AFPercent_Format)
JS_STATIC_GLOBAL_FUN_ENTRY(AFPercent_Keystroke)
JS_STATIC_GLOBAL_FUN_ENTRY(AFDate_FormatEx)
JS_STATIC_GLOBAL_FUN_ENTRY(AFDate_KeystrokeEx)
JS_STATIC_GLOBAL_FUN_ENTRY(AFDate_Format)
JS_STATIC_GLOBAL_FUN_ENTRY(AFDate_Keystroke)
JS_STATIC_GLOBAL_FUN_ENTRY(AFTime_FormatEx)
JS_STATIC_GLOBAL_FUN_ENTRY(AFTime_KeystrokeEx)
JS_STATIC_GLOBAL_FUN_ENTRY(AFTime_Format)
JS_STATIC_GLOBAL_FUN_ENTRY(AFTime_Keystroke)
JS_STATIC_GLOBAL_FUN_ENTRY(AFSpecial_Format)
JS_STATIC_GLOBAL_FUN_ENTRY(AFSpecial_Keystroke)
JS_STATIC_GLOBAL_FUN_ENTRY(AFSpecial_KeystrokeEx)
JS_STATIC_GLOBAL_FUN_ENTRY(AFSimple)
JS_STATIC_GLOBAL_FUN_ENTRY(AFMakeNumber)
JS_STATIC_GLOBAL_FUN_ENTRY(AFSimple_Calculate)
JS_STATIC_GLOBAL_FUN_ENTRY(AFRange_Validate)
JS_STATIC_GLOBAL_FUN_ENTRY(AFMergeChange)
JS_STATIC_GLOBAL_FUN_ENTRY(AFParseDateEx)
JS_STATIC_GLOBAL_FUN_ENTRY(AFExtractNums)
END_JS_STATIC_GLOBAL_FUN()
IMPLEMENT_JS_STATIC_GLOBAL_FUN(CJS_PublicMethods)
static const FX_WCHAR* const months[] = {L"Jan",
L"Feb",
L"Mar",
L"Apr",
L"May",
L"Jun",
L"Jul",
L"Aug",
L"Sep",
L"Oct",
L"Nov",
L"Dec"};
static const FX_WCHAR* const fullmonths[] = {L"January",
L"February",
L"March",
L"April",
L"May",
L"June",
L"July",
L"August",
L"September",
L"October",
L"November",
L"December"};
FX_BOOL CJS_PublicMethods::IsNumber(const FX_WCHAR* string) {
CFX_WideString sTrim = StrTrim(string);
const FX_WCHAR* pTrim = sTrim.c_str();
const FX_WCHAR* p = pTrim;
FX_BOOL bDot = FALSE;
FX_BOOL bKXJS = FALSE;
wchar_t c;
while ((c = *p)) {
if (c == '.' || c == ',') {
if (bDot)
return FALSE;
bDot = TRUE;
} else if (c == '-' || c == '+') {
if (p != pTrim)
return FALSE;
} else if (c == 'e' || c == 'E') {
if (bKXJS)
return FALSE;
p++;
c = *p;
if (c == '+' || c == '-') {
bKXJS = TRUE;
} else {
return FALSE;
}
} else if (!FXSYS_iswdigit(c)) {
return FALSE;
}
p++;
}
return TRUE;
}
FX_BOOL CJS_PublicMethods::maskSatisfied(wchar_t c_Change, wchar_t c_Mask) {
switch (c_Mask) {
case L'9':
return FXSYS_iswdigit(c_Change);
case L'A':
return FXSYS_iswalpha(c_Change);
case L'O':
return FXSYS_iswalnum(c_Change);
case L'X':
return TRUE;
default:
return (c_Change == c_Mask);
}
}
FX_BOOL CJS_PublicMethods::isReservedMaskChar(wchar_t ch) {
return ch == L'9' || ch == L'A' || ch == L'O' || ch == L'X';
}
double CJS_PublicMethods::AF_Simple(const FX_WCHAR* sFuction,
double dValue1,
double dValue2) {
if (FXSYS_wcsicmp(sFuction, L"AVG") == 0 ||
FXSYS_wcsicmp(sFuction, L"SUM") == 0) {
return dValue1 + dValue2;
}
if (FXSYS_wcsicmp(sFuction, L"PRD") == 0) {
return dValue1 * dValue2;
}
if (FXSYS_wcsicmp(sFuction, L"MIN") == 0) {
return std::min(dValue1, dValue2);
}
if (FXSYS_wcsicmp(sFuction, L"MAX") == 0) {
return std::max(dValue1, dValue2);
}
return dValue1;
}
CFX_WideString CJS_PublicMethods::StrLTrim(const FX_WCHAR* pStr) {
while (*pStr && *pStr == L' ')
pStr++;
return pStr;
}
CFX_WideString CJS_PublicMethods::StrRTrim(const FX_WCHAR* pStr) {
const FX_WCHAR* p = pStr;
while (*p)
p++;
while (p > pStr && *(p - 1) == L' ')
p--;
return CFX_WideString(pStr, p - pStr);
}
CFX_WideString CJS_PublicMethods::StrTrim(const FX_WCHAR* pStr) {
return StrRTrim(StrLTrim(pStr).c_str());
}
CFX_ByteString CJS_PublicMethods::StrLTrim(const FX_CHAR* pStr) {
while (*pStr && *pStr == ' ')
pStr++;
return pStr;
}
CFX_ByteString CJS_PublicMethods::StrRTrim(const FX_CHAR* pStr) {
const FX_CHAR* p = pStr;
while (*p)
p++;
while (p > pStr && *(p - 1) == L' ')
p--;
return CFX_ByteString(pStr, p - pStr);
}
CFX_ByteString CJS_PublicMethods::StrTrim(const FX_CHAR* pStr) {
return StrRTrim(StrLTrim(pStr));
}
double CJS_PublicMethods::ParseNumber(const FX_WCHAR* swSource,
FX_BOOL& bAllDigits,
FX_BOOL& bDot,
FX_BOOL& bSign,
FX_BOOL& bKXJS) {
bDot = FALSE;
bSign = FALSE;
bKXJS = FALSE;
FX_BOOL bDigitExist = FALSE;
const FX_WCHAR* p = swSource;
wchar_t c;
const FX_WCHAR* pStart = NULL;
const FX_WCHAR* pEnd = NULL;
while ((c = *p)) {
if (!pStart && c != L' ') {
pStart = p;
}
pEnd = p;
p++;
}
if (!pStart) {
bAllDigits = FALSE;
return 0;
}
while (pEnd != pStart) {
if (*pEnd == L' ')
pEnd--;
else
break;
}
double dRet = 0;
p = pStart;
bAllDigits = TRUE;
CFX_WideString swDigits;
while (p <= pEnd) {
c = *p;
if (FXSYS_iswdigit(c)) {
swDigits += c;
bDigitExist = TRUE;
} else {
switch (c) {
case L' ':
bAllDigits = FALSE;
break;
case L'.':
case L',':
if (!bDot) {
if (bDigitExist) {
swDigits += L'.';
} else {
swDigits += L'0';
swDigits += L'.';
bDigitExist = TRUE;
}
bDot = TRUE;
break;
}
case 'e':
case 'E':
if (!bKXJS) {
p++;
c = *p;
if (c == '+' || c == '-') {
bKXJS = TRUE;
swDigits += 'e';
swDigits += c;
}
break;
}
case L'-':
if (!bDigitExist && !bSign) {
swDigits += c;
bSign = TRUE;
break;
}
default:
bAllDigits = FALSE;
if (p != pStart && !bDot && bDigitExist) {
swDigits += L'.';
bDot = TRUE;
} else {
bDot = FALSE;
bDigitExist = FALSE;
swDigits = L"";
}
break;
}
}
p++;
}
if (swDigits.GetLength() > 0 && swDigits.GetLength() < 17) {
CFX_ByteString sDigits = swDigits.UTF8Encode();
if (bKXJS) {
dRet = atof(sDigits);
} else {
if (bDot) {
char* pStopString;
dRet = ::strtod(sDigits, &pStopString);
} else {
dRet = atol(sDigits);
}
}
}
return dRet;
}
double CJS_PublicMethods::ParseStringToNumber(const FX_WCHAR* swSource) {
FX_BOOL bAllDigits = FALSE;
FX_BOOL bDot = FALSE;
FX_BOOL bSign = FALSE;
FX_BOOL bKXJS = FALSE;
return ParseNumber(swSource, bAllDigits, bDot, bSign, bKXJS);
}
FX_BOOL CJS_PublicMethods::ConvertStringToNumber(const FX_WCHAR* swSource,
double& dRet,
FX_BOOL& bDot) {
FX_BOOL bAllDigits = FALSE;
FX_BOOL bSign = FALSE;
FX_BOOL bKXJS = FALSE;
dRet = ParseNumber(swSource, bAllDigits, bDot, bSign, bKXJS);
return bAllDigits;
}
CJS_Array CJS_PublicMethods::AF_MakeArrayFromList(CJS_Runtime* pRuntime,
CJS_Value val) {
CJS_Array StrArray(pRuntime);
if (val.IsArrayObject()) {
val.ConvertToArray(StrArray);
return StrArray;
}
CFX_WideString wsStr = val.ToCFXWideString();
CFX_ByteString t = CFX_ByteString::FromUnicode(wsStr);
const char* p = (const char*)t;
int ch = ',';
int nIndex = 0;
while (*p) {
const char* pTemp = strchr(p, ch);
if (!pTemp) {
StrArray.SetElement(nIndex, CJS_Value(pRuntime, StrTrim(p).c_str()));
break;
}
char* pSub = new char[pTemp - p + 1];
strncpy(pSub, p, pTemp - p);
*(pSub + (pTemp - p)) = '\0';
StrArray.SetElement(nIndex, CJS_Value(pRuntime, StrTrim(pSub).c_str()));
delete[] pSub;
nIndex++;
p = ++pTemp;
}
return StrArray;
}
int CJS_PublicMethods::ParseStringInteger(const CFX_WideString& string,
int nStart,
int& nSkip,
int nMaxStep) {
int nRet = 0;
nSkip = 0;
for (int i = nStart, sz = string.GetLength(); i < sz; i++) {
if (i - nStart > 10)
break;
FX_WCHAR c = string.GetAt(i);
if (!FXSYS_iswdigit(c))
break;
nRet = nRet * 10 + FXSYS_toDecimalDigitWide(c);
nSkip = i - nStart + 1;
if (nSkip >= nMaxStep)
break;
}
return nRet;
}
CFX_WideString CJS_PublicMethods::ParseStringString(
const CFX_WideString& string,
int nStart,
int& nSkip) {
CFX_WideString swRet;
nSkip = 0;
for (int i = nStart, sz = string.GetLength(); i < sz; i++) {
FX_WCHAR c = string.GetAt(i);
if (!FXSYS_iswdigit(c))
break;
swRet += c;
nSkip = i - nStart + 1;
}
return swRet;
}
double CJS_PublicMethods::ParseNormalDate(const CFX_WideString& value,
bool* bWrongFormat) {
double dt = JS_GetDateTime();
int nYear = JS_GetYearFromTime(dt);
int nMonth = JS_GetMonthFromTime(dt) + 1;
int nDay = JS_GetDayFromTime(dt);
int nHour = JS_GetHourFromTime(dt);
int nMin = JS_GetMinFromTime(dt);
int nSec = JS_GetSecFromTime(dt);
int number[3];
int nSkip = 0;
int nLen = value.GetLength();
int nIndex = 0;
int i = 0;
while (i < nLen) {
if (nIndex > 2)
break;
FX_WCHAR c = value.GetAt(i);
if (FXSYS_iswdigit(c)) {
number[nIndex++] = ParseStringInteger(value, i, nSkip, 4);
i += nSkip;
} else {
i++;
}
}
if (nIndex == 2) {
// case2: month/day
// case3: day/month
if ((number[0] >= 1 && number[0] <= 12) &&
(number[1] >= 1 && number[1] <= 31)) {
nMonth = number[0];
nDay = number[1];
} else if ((number[0] >= 1 && number[0] <= 31) &&
(number[1] >= 1 && number[1] <= 12)) {
nDay = number[0];
nMonth = number[1];
}
if (bWrongFormat)
*bWrongFormat = false;
} else if (nIndex == 3) {
// case1: year/month/day
// case2: month/day/year
// case3: day/month/year
if (number[0] > 12 && (number[1] >= 1 && number[1] <= 12) &&
(number[2] >= 1 && number[2] <= 31)) {
nYear = number[0];
nMonth = number[1];
nDay = number[2];
} else if ((number[0] >= 1 && number[0] <= 12) &&
(number[1] >= 1 && number[1] <= 31) && number[2] > 31) {
nMonth = number[0];
nDay = number[1];
nYear = number[2];
} else if ((number[0] >= 1 && number[0] <= 31) &&
(number[1] >= 1 && number[1] <= 12) && number[2] > 31) {
nDay = number[0];
nMonth = number[1];
nYear = number[2];
}
if (bWrongFormat)
*bWrongFormat = false;
} else {
if (bWrongFormat)
*bWrongFormat = true;
return dt;
}
CFX_WideString swTemp;
swTemp.Format(L"%d/%d/%d %d:%d:%d", nMonth, nDay, nYear, nHour, nMin, nSec);
return JS_DateParse(swTemp.c_str());
}
double CJS_PublicMethods::MakeRegularDate(const CFX_WideString& value,
const CFX_WideString& format,
bool* bWrongFormat) {
double dt = JS_GetDateTime();
if (format.IsEmpty() || value.IsEmpty())
return dt;
int nYear = JS_GetYearFromTime(dt);
int nMonth = JS_GetMonthFromTime(dt) + 1;
int nDay = JS_GetDayFromTime(dt);
int nHour = JS_GetHourFromTime(dt);
int nMin = JS_GetMinFromTime(dt);
int nSec = JS_GetSecFromTime(dt);
int nYearSub = 99; // nYear - 2000;
FX_BOOL bPm = FALSE;
FX_BOOL bExit = FALSE;
bool bBadFormat = false;
int i = 0;
int j = 0;
while (i < format.GetLength()) {
if (bExit)
break;
FX_WCHAR c = format.GetAt(i);
switch (c) {
case ':':
case '.':
case '-':
case '\\':
case '/':
i++;
j++;
break;
case 'y':
case 'm':
case 'd':
case 'H':
case 'h':
case 'M':
case 's':
case 't': {
int oldj = j;
int nSkip = 0;
int remaining = format.GetLength() - i - 1;
if (remaining == 0 || format.GetAt(i + 1) != c) {
switch (c) {
case 'y':
i++;
j++;
break;
case 'm':
nMonth = ParseStringInteger(value, j, nSkip, 2);
i++;
j += nSkip;
break;
case 'd':
nDay = ParseStringInteger(value, j, nSkip, 2);
i++;
j += nSkip;
break;
case 'H':
nHour = ParseStringInteger(value, j, nSkip, 2);
i++;
j += nSkip;
break;
case 'h':
nHour = ParseStringInteger(value, j, nSkip, 2);
i++;
j += nSkip;
break;
case 'M':
nMin = ParseStringInteger(value, j, nSkip, 2);
i++;
j += nSkip;
break;
case 's':
nSec = ParseStringInteger(value, j, nSkip, 2);
i++;
j += nSkip;
break;
case 't':
bPm = (j < value.GetLength() && value.GetAt(j) == 'p');
i++;
j++;
break;
}
} else if (remaining == 1 || format.GetAt(i + 2) != c) {
switch (c) {
case 'y':
nYear = ParseStringInteger(value, j, nSkip, 4);
i += 2;
j += nSkip;
break;
case 'm':
nMonth = ParseStringInteger(value, j, nSkip, 2);
i += 2;
j += nSkip;
break;
case 'd':
nDay = ParseStringInteger(value, j, nSkip, 2);
i += 2;
j += nSkip;
break;
case 'H':
nHour = ParseStringInteger(value, j, nSkip, 2);
i += 2;
j += nSkip;
break;
case 'h':
nHour = ParseStringInteger(value, j, nSkip, 2);
i += 2;
j += nSkip;
break;
case 'M':
nMin = ParseStringInteger(value, j, nSkip, 2);
i += 2;
j += nSkip;
break;
case 's':
nSec = ParseStringInteger(value, j, nSkip, 2);
i += 2;
j += nSkip;
break;
case 't':
bPm = (j + 1 < value.GetLength() && value.GetAt(j) == 'p' &&
value.GetAt(j + 1) == 'm');
i += 2;
j += 2;
break;
}
} else if (remaining == 2 || format.GetAt(i + 3) != c) {
switch (c) {
case 'm': {
CFX_WideString sMonth = ParseStringString(value, j, nSkip);
FX_BOOL bFind = FALSE;
for (int m = 0; m < 12; m++) {
if (sMonth.CompareNoCase(months[m]) == 0) {
nMonth = m + 1;
i += 3;
j += nSkip;
bFind = TRUE;
break;
}
}
if (!bFind) {
nMonth = ParseStringInteger(value, j, nSkip, 3);
i += 3;
j += nSkip;
}
} break;
case 'y':
break;
default:
i += 3;
j += 3;
break;
}
} else if (remaining == 3 || format.GetAt(i + 4) != c) {
switch (c) {
case 'y':
nYear = ParseStringInteger(value, j, nSkip, 4);
j += nSkip;
i += 4;
break;
case 'm': {
FX_BOOL bFind = FALSE;
CFX_WideString sMonth = ParseStringString(value, j, nSkip);
sMonth.MakeLower();
for (int m = 0; m < 12; m++) {
CFX_WideString sFullMonths = fullmonths[m];
sFullMonths.MakeLower();
if (sFullMonths.Find(sMonth.c_str(), 0) != -1) {
nMonth = m + 1;
i += 4;
j += nSkip;
bFind = TRUE;
break;
}
}
if (!bFind) {
nMonth = ParseStringInteger(value, j, nSkip, 4);
i += 4;
j += nSkip;
}
} break;
default:
i += 4;
j += 4;
break;
}
} else {
if (j >= value.GetLength() || format.GetAt(i) != value.GetAt(j)) {
bBadFormat = true;
bExit = TRUE;
}
i++;
j++;
}
if (oldj == j) {
bBadFormat = true;
bExit = TRUE;
}
}
break;
default:
if (value.GetLength() <= j) {
bExit = TRUE;
} else if (format.GetAt(i) != value.GetAt(j)) {
bBadFormat = true;
bExit = TRUE;
}
i++;
j++;
break;
}
}
if (bPm)
nHour += 12;
if (nYear >= 0 && nYear <= nYearSub)
nYear += 2000;
if (nMonth < 1 || nMonth > 12)
bBadFormat = true;
if (nDay < 1 || nDay > 31)
bBadFormat = true;
if (nHour < 0 || nHour > 24)
bBadFormat = true;
if (nMin < 0 || nMin > 60)
bBadFormat = true;
if (nSec < 0 || nSec > 60)
bBadFormat = true;
double dRet = 0;
if (bBadFormat) {
dRet = ParseNormalDate(value, &bBadFormat);
} else {
dRet = JS_MakeDate(JS_MakeDay(nYear, nMonth - 1, nDay),
JS_MakeTime(nHour, nMin, nSec, 0));
if (JS_PortIsNan(dRet)) {
dRet = JS_DateParse(value.c_str());
}
}
if (JS_PortIsNan(dRet)) {
dRet = ParseNormalDate(value, &bBadFormat);
}
if (bWrongFormat)
*bWrongFormat = bBadFormat;
return dRet;
}
CFX_WideString CJS_PublicMethods::MakeFormatDate(double dDate,
const CFX_WideString& format) {
CFX_WideString sRet = L"", sPart = L"";
int nYear = JS_GetYearFromTime(dDate);
int nMonth = JS_GetMonthFromTime(dDate) + 1;
int nDay = JS_GetDayFromTime(dDate);
int nHour = JS_GetHourFromTime(dDate);
int nMin = JS_GetMinFromTime(dDate);
int nSec = JS_GetSecFromTime(dDate);
int i = 0;
while (i < format.GetLength()) {
FX_WCHAR c = format.GetAt(i);
int remaining = format.GetLength() - i - 1;
sPart = L"";
switch (c) {
case 'y':
case 'm':
case 'd':
case 'H':
case 'h':
case 'M':
case 's':
case 't':
if (remaining == 0 || format.GetAt(i + 1) != c) {
switch (c) {
case 'y':
sPart += c;
break;
case 'm':
sPart.Format(L"%d", nMonth);
break;
case 'd':
sPart.Format(L"%d", nDay);
break;
case 'H':
sPart.Format(L"%d", nHour);
break;
case 'h':
sPart.Format(L"%d", nHour > 12 ? nHour - 12 : nHour);
break;
case 'M':
sPart.Format(L"%d", nMin);
break;
case 's':
sPart.Format(L"%d", nSec);
break;
case 't':
sPart += nHour > 12 ? 'p' : 'a';
break;
}
i++;
} else if (remaining == 1 || format.GetAt(i + 2) != c) {
switch (c) {
case 'y':
sPart.Format(L"%02d", nYear - (nYear / 100) * 100);
break;
case 'm':
sPart.Format(L"%02d", nMonth);
break;
case 'd':
sPart.Format(L"%02d", nDay);
break;
case 'H':
sPart.Format(L"%02d", nHour);
break;
case 'h':
sPart.Format(L"%02d", nHour > 12 ? nHour - 12 : nHour);
break;
case 'M':
sPart.Format(L"%02d", nMin);
break;
case 's':
sPart.Format(L"%02d", nSec);
break;
case 't':
sPart = nHour > 12 ? L"pm" : L"am";
break;
}
i += 2;
} else if (remaining == 2 || format.GetAt(i + 3) != c) {
switch (c) {
case 'm':
i += 3;
if (nMonth > 0 && nMonth <= 12)
sPart += months[nMonth - 1];
break;
default:
i += 3;
sPart += c;
sPart += c;
sPart += c;
break;
}
} else if (remaining == 3 || format.GetAt(i + 4) != c) {
switch (c) {
case 'y':
sPart.Format(L"%04d", nYear);
i += 4;
break;
case 'm':
i += 4;
if (nMonth > 0 && nMonth <= 12)
sPart += fullmonths[nMonth - 1];
break;
default:
i += 4;
sPart += c;
sPart += c;
sPart += c;
sPart += c;
break;
}
} else {
i++;
sPart += c;
}
break;
default:
i++;
sPart += c;
break;
}
sRet += sPart;
}
return sRet;
}
/* -------------------------------------------------------------------------- */
// function AFNumber_Format(nDec, sepStyle, negStyle, currStyle, strCurrency,
// bCurrencyPrepend)
FX_BOOL CJS_PublicMethods::AFNumber_Format(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
#if _FX_OS_ != _FX_ANDROID_
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 6) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
CJS_Runtime* pRuntime = CJS_Runtime::FromContext(cc);
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (!pEvent->m_pValue)
return FALSE;
CFX_WideString& Value = pEvent->Value();
CFX_ByteString strValue = StrTrim(CFX_ByteString::FromUnicode(Value));
if (strValue.IsEmpty())
return TRUE;
int iDec = params[0].ToInt();
int iSepStyle = params[1].ToInt();
int iNegStyle = params[2].ToInt();
// params[3] is iCurrStyle, it's not used.
std::wstring wstrCurrency(params[4].ToCFXWideString().c_str());
FX_BOOL bCurrencyPrepend = params[5].ToBool();
if (iDec < 0)
iDec = -iDec;
if (iSepStyle < 0 || iSepStyle > 3)
iSepStyle = 0;
if (iNegStyle < 0 || iNegStyle > 3)
iNegStyle = 0;
//////////////////////////////////////////////////////
// for processing decimal places
strValue.Replace(",", ".");
double dValue = atof(strValue);
if (iDec > 0)
dValue += DOUBLE_CORRECT;
int iDec2;
int iNegative = 0;
strValue = fcvt(dValue, iDec, &iDec2, &iNegative);
if (strValue.IsEmpty()) {
dValue = 0;
strValue = fcvt(dValue, iDec, &iDec2, &iNegative);
if (strValue.IsEmpty()) {
strValue = "0";
iDec2 = 1;
}
}
if (iDec2 < 0) {
for (int iNum = 0; iNum < abs(iDec2); iNum++) {
strValue = "0" + strValue;
}
iDec2 = 0;
}
int iMax = strValue.GetLength();
if (iDec2 > iMax) {
for (int iNum = 0; iNum <= iDec2 - iMax; iNum++) {
strValue += "0";
}
iMax = iDec2 + 1;
}
///////////////////////////////////////////////////////
// for processing seperator style
if (iDec2 < iMax) {
if (iSepStyle == 0 || iSepStyle == 1) {
strValue.Insert(iDec2, '.');
iMax++;
} else if (iSepStyle == 2 || iSepStyle == 3) {
strValue.Insert(iDec2, ',');
iMax++;
}
if (iDec2 == 0)
strValue.Insert(iDec2, '0');
}
if (iSepStyle == 0 || iSepStyle == 2) {
char cSeperator;
if (iSepStyle == 0)
cSeperator = ',';
else
cSeperator = '.';
for (int iDecPositive = iDec2 - 3; iDecPositive > 0; iDecPositive -= 3) {
strValue.Insert(iDecPositive, cSeperator);
iMax++;
}
}
//////////////////////////////////////////////////////////////////////
// for processing currency string
Value = CFX_WideString::FromLocal(strValue);
std::wstring strValue2 = Value.c_str();
if (bCurrencyPrepend)
strValue2 = wstrCurrency + strValue2;
else
strValue2 = strValue2 + wstrCurrency;
/////////////////////////////////////////////////////////////////////////
// for processing negative style
if (iNegative) {
if (iNegStyle == 0) {
strValue2.insert(0, L"-");
}
if (iNegStyle == 2 || iNegStyle == 3) {
strValue2.insert(0, L"(");
strValue2.insert(strValue2.length(), L")");
}
if (iNegStyle == 1 || iNegStyle == 3) {
if (Field* fTarget = pEvent->Target_Field()) {
CJS_Array arColor(pRuntime);
CJS_Value vColElm(pRuntime);
vColElm = L"RGB";
arColor.SetElement(0, vColElm);
vColElm = 1;
arColor.SetElement(1, vColElm);
vColElm = 0;
arColor.SetElement(2, vColElm);
arColor.SetElement(3, vColElm);
CJS_PropValue vProp(pRuntime);
vProp.StartGetting();
vProp << arColor;
vProp.StartSetting();
fTarget->textColor(cc, vProp, sError); // red
}
}
} else {
if (iNegStyle == 1 || iNegStyle == 3) {
if (Field* fTarget = pEvent->Target_Field()) {
CJS_Array arColor(pRuntime);
CJS_Value vColElm(pRuntime);
vColElm = L"RGB";
arColor.SetElement(0, vColElm);
vColElm = 0;
arColor.SetElement(1, vColElm);
arColor.SetElement(2, vColElm);
arColor.SetElement(3, vColElm);
CJS_PropValue vProp(pRuntime);
vProp.StartGetting();
fTarget->textColor(cc, vProp, sError);
CJS_Array aProp(pRuntime);
vProp.ConvertToArray(aProp);
CPWL_Color crProp;
CPWL_Color crColor;
color::ConvertArrayToPWLColor(aProp, crProp);
color::ConvertArrayToPWLColor(arColor, crColor);
if (crColor != crProp) {
CJS_PropValue vProp2(pRuntime);
vProp2.StartGetting();
vProp2 << arColor;
vProp2.StartSetting();
fTarget->textColor(cc, vProp2, sError);
}
}
}
}
Value = strValue2.c_str();
#endif
return TRUE;
}
// function AFNumber_Keystroke(nDec, sepStyle, negStyle, currStyle, strCurrency,
// bCurrencyPrepend)
FX_BOOL CJS_PublicMethods::AFNumber_Keystroke(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (params.size() < 2)
return FALSE;
int iSepStyle = params[1].ToInt();
if (iSepStyle < 0 || iSepStyle > 3)
iSepStyle = 0;
if (!pEvent->m_pValue)
return FALSE;
CFX_WideString& val = pEvent->Value();
CFX_WideString& w_strChange = pEvent->Change();
CFX_WideString w_strValue = val;
if (pEvent->WillCommit()) {
CFX_WideString wstrChange = w_strChange;
CFX_WideString wstrValue = StrLTrim(w_strValue.c_str());
if (wstrValue.IsEmpty())
return TRUE;
CFX_WideString swTemp = wstrValue;
swTemp.Replace(L",", L".");
if (!IsNumber(swTemp.c_str())) {
pEvent->Rc() = FALSE;
sError = JSGetStringFromID(pContext, IDS_STRING_JSAFNUMBER_KEYSTROKE);
Alert(pContext, sError.c_str());
return TRUE;
}
return TRUE; // it happens after the last keystroke and before validating,
}
std::wstring w_strValue2 = w_strValue.c_str();
std::wstring w_strChange2 = w_strChange.c_str();
std::wstring w_strSelected;
if (-1 != pEvent->SelStart())
w_strSelected = w_strValue2.substr(pEvent->SelStart(),
(pEvent->SelEnd() - pEvent->SelStart()));
bool bHasSign = (w_strValue2.find('-') != std::wstring::npos) &&
(w_strSelected.find('-') == std::wstring::npos);
if (bHasSign) {
// can't insert "change" in front to sign postion.
if (pEvent->SelStart() == 0) {
FX_BOOL& bRc = pEvent->Rc();
bRc = FALSE;
return TRUE;
}
}
char cSep = L'.';
switch (iSepStyle) {
case 0:
case 1:
cSep = L'.';
break;
case 2:
case 3:
cSep = L',';
break;
}
bool bHasSep = (w_strValue2.find(cSep) != std::wstring::npos);
for (std::wstring::iterator it = w_strChange2.begin();
it != w_strChange2.end(); it++) {
if (*it == cSep) {
if (bHasSep) {
FX_BOOL& bRc = pEvent->Rc();
bRc = FALSE;
return TRUE;
}
bHasSep = TRUE;
continue;
}
if (*it == L'-') {
if (bHasSign) {
FX_BOOL& bRc = pEvent->Rc();
bRc = FALSE;
return TRUE;
}
// sign's position is not correct
if (it != w_strChange2.begin()) {
FX_BOOL& bRc = pEvent->Rc();
bRc = FALSE;
return TRUE;
}
if (pEvent->SelStart() != 0) {
FX_BOOL& bRc = pEvent->Rc();
bRc = FALSE;
return TRUE;
}
bHasSign = TRUE;
continue;
}
if (!FXSYS_iswdigit(*it)) {
FX_BOOL& bRc = pEvent->Rc();
bRc = FALSE;
return TRUE;
}
}
std::wstring w_prefix = w_strValue2.substr(0, pEvent->SelStart());
std::wstring w_postfix;
if (pEvent->SelEnd() < (int)w_strValue2.length())
w_postfix = w_strValue2.substr(pEvent->SelEnd());
w_strValue2 = w_prefix + w_strChange2 + w_postfix;
w_strValue = w_strValue2.c_str();
val = w_strValue;
return TRUE;
}
// function AFPercent_Format(nDec, sepStyle)
FX_BOOL CJS_PublicMethods::AFPercent_Format(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
#if _FX_OS_ != _FX_ANDROID_
CJS_Context* pContext = (CJS_Context*)cc;
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (params.size() != 2) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
if (!pEvent->m_pValue)
return FALSE;
CFX_WideString& Value = pEvent->Value();
CFX_ByteString strValue = StrTrim(CFX_ByteString::FromUnicode(Value));
if (strValue.IsEmpty())
return TRUE;
int iDec = params[0].ToInt();
if (iDec < 0)
iDec = -iDec;
int iSepStyle = params[1].ToInt();
if (iSepStyle < 0 || iSepStyle > 3)
iSepStyle = 0;
//////////////////////////////////////////////////////
// for processing decimal places
double dValue = atof(strValue);
dValue *= 100;
if (iDec > 0)
dValue += DOUBLE_CORRECT;
int iDec2;
int iNegative = 0;
strValue = fcvt(dValue, iDec, &iDec2, &iNegative);
if (strValue.IsEmpty()) {
dValue = 0;
strValue = fcvt(dValue, iDec, &iDec2, &iNegative);
}
if (iDec2 < 0) {
for (int iNum = 0; iNum < abs(iDec2); iNum++) {
strValue = "0" + strValue;
}
iDec2 = 0;
}
int iMax = strValue.GetLength();
if (iDec2 > iMax) {
for (int iNum = 0; iNum <= iDec2 - iMax; iNum++) {
strValue += "0";
}
iMax = iDec2 + 1;
}
///////////////////////////////////////////////////////
// for processing seperator style
if (iDec2 < iMax) {
if (iSepStyle == 0 || iSepStyle == 1) {
strValue.Insert(iDec2, '.');
iMax++;
} else if (iSepStyle == 2 || iSepStyle == 3) {
strValue.Insert(iDec2, ',');
iMax++;
}
if (iDec2 == 0)
strValue.Insert(iDec2, '0');
}
if (iSepStyle == 0 || iSepStyle == 2) {
char cSeperator;
if (iSepStyle == 0)
cSeperator = ',';
else
cSeperator = '.';
for (int iDecPositive = iDec2 - 3; iDecPositive > 0; iDecPositive -= 3) {
strValue.Insert(iDecPositive, cSeperator);
iMax++;
}
}
////////////////////////////////////////////////////////////////////
// negative mark
if (iNegative)
strValue = "-" + strValue;
strValue += "%";
Value = CFX_WideString::FromLocal(strValue);
#endif
return TRUE;
}
// AFPercent_Keystroke(nDec, sepStyle)
FX_BOOL CJS_PublicMethods::AFPercent_Keystroke(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
return AFNumber_Keystroke(cc, params, vRet, sError);
}
// function AFDate_FormatEx(cFormat)
FX_BOOL CJS_PublicMethods::AFDate_FormatEx(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
if (!pEvent->m_pValue)
return FALSE;
CFX_WideString& val = pEvent->Value();
CFX_WideString strValue = val;
if (strValue.IsEmpty())
return TRUE;
CFX_WideString sFormat = params[0].ToCFXWideString();
double dDate = 0.0f;
if (strValue.Find(L"GMT") != -1) {
// for GMT format time
// such as "Tue Aug 11 14:24:16 GMT+08002009"
dDate = MakeInterDate(strValue);
} else {
dDate = MakeRegularDate(strValue, sFormat, nullptr);
}
if (JS_PortIsNan(dDate)) {
CFX_WideString swMsg;
swMsg.Format(JSGetStringFromID(pContext, IDS_STRING_JSPARSEDATE).c_str(),
sFormat.c_str());
Alert(pContext, swMsg.c_str());
return FALSE;
}
val = MakeFormatDate(dDate, sFormat);
return TRUE;
}
double CJS_PublicMethods::MakeInterDate(CFX_WideString strValue) {
int nHour;
int nMin;
int nSec;
int nYear;
int nMonth;
int nDay;
CFX_WideStringArray wsArray;
CFX_WideString sMonth = L"";
CFX_WideString sTemp = L"";
int nSize = strValue.GetLength();
for (int i = 0; i < nSize; i++) {
FX_WCHAR c = strValue.GetAt(i);
if (c == L' ' || c == L':') {
wsArray.Add(sTemp);
sTemp = L"";
continue;
}
sTemp += c;
}
wsArray.Add(sTemp);
if (wsArray.GetSize() != 8)
return 0;
sTemp = wsArray[1];
if (sTemp.Compare(L"Jan") == 0)
nMonth = 1;
if (sTemp.Compare(L"Feb") == 0)
nMonth = 2;
if (sTemp.Compare(L"Mar") == 0)
nMonth = 3;
if (sTemp.Compare(L"Apr") == 0)
nMonth = 4;
if (sTemp.Compare(L"May") == 0)
nMonth = 5;
if (sTemp.Compare(L"Jun") == 0)
nMonth = 6;
if (sTemp.Compare(L"Jul") == 0)
nMonth = 7;
if (sTemp.Compare(L"Aug") == 0)
nMonth = 8;
if (sTemp.Compare(L"Sep") == 0)
nMonth = 9;
if (sTemp.Compare(L"Oct") == 0)
nMonth = 10;
if (sTemp.Compare(L"Nov") == 0)
nMonth = 11;
if (sTemp.Compare(L"Dec") == 0)
nMonth = 12;
nDay = (int)ParseStringToNumber(wsArray[2].c_str());
nHour = (int)ParseStringToNumber(wsArray[3].c_str());
nMin = (int)ParseStringToNumber(wsArray[4].c_str());
nSec = (int)ParseStringToNumber(wsArray[5].c_str());
nYear = (int)ParseStringToNumber(wsArray[7].c_str());
double dRet = JS_MakeDate(JS_MakeDay(nYear, nMonth - 1, nDay),
JS_MakeTime(nHour, nMin, nSec, 0));
if (JS_PortIsNan(dRet)) {
dRet = JS_DateParse(strValue.c_str());
}
return dRet;
}
// AFDate_KeystrokeEx(cFormat)
FX_BOOL CJS_PublicMethods::AFDate_KeystrokeEx(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (params.size() != 1) {
sError = L"AFDate_KeystrokeEx's parameters' size r not correct";
return FALSE;
}
if (pEvent->WillCommit()) {
if (!pEvent->m_pValue)
return FALSE;
CFX_WideString strValue = pEvent->Value();
if (strValue.IsEmpty())
return TRUE;
CFX_WideString sFormat = params[0].ToCFXWideString();
bool bWrongFormat = FALSE;
double dRet = MakeRegularDate(strValue, sFormat, &bWrongFormat);
if (bWrongFormat || JS_PortIsNan(dRet)) {
CFX_WideString swMsg;
swMsg.Format(JSGetStringFromID(pContext, IDS_STRING_JSPARSEDATE).c_str(),
sFormat.c_str());
Alert(pContext, swMsg.c_str());
pEvent->Rc() = FALSE;
return TRUE;
}
}
return TRUE;
}
FX_BOOL CJS_PublicMethods::AFDate_Format(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
int iIndex = params[0].ToInt();
const FX_WCHAR* cFormats[] = {L"m/d",
L"m/d/yy",
L"mm/dd/yy",
L"mm/yy",
L"d-mmm",
L"d-mmm-yy",
L"dd-mmm-yy",
L"yy-mm-dd",
L"mmm-yy",
L"mmmm-yy",
L"mmm d, yyyy",
L"mmmm d, yyyy",
L"m/d/yy h:MM tt",
L"m/d/yy HH:MM"};
if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
iIndex = 0;
std::vector<CJS_Value> newParams;
newParams.push_back(
CJS_Value(CJS_Runtime::FromContext(cc), cFormats[iIndex]));
return AFDate_FormatEx(cc, newParams, vRet, sError);
}
// AFDate_KeystrokeEx(cFormat)
FX_BOOL CJS_PublicMethods::AFDate_Keystroke(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
int iIndex = params[0].ToInt();
const FX_WCHAR* cFormats[] = {L"m/d",
L"m/d/yy",
L"mm/dd/yy",
L"mm/yy",
L"d-mmm",
L"d-mmm-yy",
L"dd-mmm-yy",
L"yy-mm-dd",
L"mmm-yy",
L"mmmm-yy",
L"mmm d, yyyy",
L"mmmm d, yyyy",
L"m/d/yy h:MM tt",
L"m/d/yy HH:MM"};
if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
iIndex = 0;
std::vector<CJS_Value> newParams;
newParams.push_back(
CJS_Value(CJS_Runtime::FromContext(cc), cFormats[iIndex]));
return AFDate_KeystrokeEx(cc, newParams, vRet, sError);
}
// function AFTime_Format(ptf)
FX_BOOL CJS_PublicMethods::AFTime_Format(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
int iIndex = params[0].ToInt();
const FX_WCHAR* cFormats[] = {L"HH:MM", L"h:MM tt", L"HH:MM:ss",
L"h:MM:ss tt"};
if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
iIndex = 0;
std::vector<CJS_Value> newParams;
newParams.push_back(
CJS_Value(CJS_Runtime::FromContext(cc), cFormats[iIndex]));
return AFDate_FormatEx(cc, newParams, vRet, sError);
}
FX_BOOL CJS_PublicMethods::AFTime_Keystroke(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
int iIndex = params[0].ToInt();
const FX_WCHAR* cFormats[] = {L"HH:MM", L"h:MM tt", L"HH:MM:ss",
L"h:MM:ss tt"};
if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
iIndex = 0;
std::vector<CJS_Value> newParams;
newParams.push_back(
CJS_Value(CJS_Runtime::FromContext(cc), cFormats[iIndex]));
return AFDate_KeystrokeEx(cc, newParams, vRet, sError);
}
FX_BOOL CJS_PublicMethods::AFTime_FormatEx(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
return AFDate_FormatEx(cc, params, vRet, sError);
}
FX_BOOL CJS_PublicMethods::AFTime_KeystrokeEx(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
return AFDate_KeystrokeEx(cc, params, vRet, sError);
}
// function AFSpecial_Format(psf)
FX_BOOL CJS_PublicMethods::AFSpecial_Format(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
std::string cFormat;
int iIndex = params[0].ToInt();
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (!pEvent->m_pValue)
return FALSE;
CFX_WideString& Value = pEvent->Value();
std::string strSrc = CFX_ByteString::FromUnicode(Value).c_str();
switch (iIndex) {
case 0:
cFormat = "99999";
break;
case 1:
cFormat = "99999-9999";
break;
case 2: {
std::string NumberStr;
util::printx("9999999999", strSrc, NumberStr);
if (NumberStr.length() >= 10)
cFormat = "(999) 999-9999";
else
cFormat = "999-9999";
break;
}
case 3:
cFormat = "999-99-9999";
break;
}
std::string strDes;
util::printx(cFormat, strSrc, strDes);
Value = CFX_WideString::FromLocal(strDes.c_str());
return TRUE;
}
// function AFSpecial_KeystrokeEx(mask)
FX_BOOL CJS_PublicMethods::AFSpecial_KeystrokeEx(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (params.size() < 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
if (!pEvent->m_pValue)
return FALSE;
CFX_WideString& valEvent = pEvent->Value();
CFX_WideString wstrMask = params[0].ToCFXWideString();
if (wstrMask.IsEmpty())
return TRUE;
const size_t wstrMaskLen = wstrMask.GetLength();
const std::wstring wstrValue = valEvent.c_str();
if (pEvent->WillCommit()) {
if (wstrValue.empty())
return TRUE;
size_t iIndexMask = 0;
for (const auto& w_Value : wstrValue) {
if (!maskSatisfied(w_Value, wstrMask[iIndexMask]))
break;
iIndexMask++;
}
if (iIndexMask != wstrMaskLen ||
(iIndexMask != wstrValue.size() && wstrMaskLen != 0)) {
Alert(
pContext,
JSGetStringFromID(pContext, IDS_STRING_JSAFNUMBER_KEYSTROKE).c_str());
pEvent->Rc() = FALSE;
}
return TRUE;
}
CFX_WideString& wideChange = pEvent->Change();
std::wstring wChange = wideChange.c_str();
if (wChange.empty())
return TRUE;
int iIndexMask = pEvent->SelStart();
size_t combined_len = wstrValue.length() + wChange.length() -
(pEvent->SelEnd() - pEvent->SelStart());
if (combined_len > wstrMaskLen) {
Alert(pContext,
JSGetStringFromID(pContext, IDS_STRING_JSPARAM_TOOLONG).c_str());
pEvent->Rc() = FALSE;
return TRUE;
}
if (iIndexMask >= wstrMaskLen && (!wChange.empty())) {
Alert(pContext,
JSGetStringFromID(pContext, IDS_STRING_JSPARAM_TOOLONG).c_str());
pEvent->Rc() = FALSE;
return TRUE;
}
for (std::wstring::iterator it = wChange.begin(); it != wChange.end(); it++) {
if (iIndexMask >= wstrMaskLen) {
Alert(pContext,
JSGetStringFromID(pContext, IDS_STRING_JSPARAM_TOOLONG).c_str());
pEvent->Rc() = FALSE;
return TRUE;
}
wchar_t w_Mask = wstrMask[iIndexMask];
if (!isReservedMaskChar(w_Mask)) {
*it = w_Mask;
}
wchar_t w_Change = *it;
if (!maskSatisfied(w_Change, w_Mask)) {
pEvent->Rc() = FALSE;
return TRUE;
}
iIndexMask++;
}
wideChange = wChange.c_str();
return TRUE;
}
// function AFSpecial_Keystroke(psf)
FX_BOOL CJS_PublicMethods::AFSpecial_Keystroke(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (!pEvent->m_pValue)
return FALSE;
std::string cFormat;
int iIndex = params[0].ToInt();
CFX_WideString& val = pEvent->Value();
std::string strSrc = CFX_ByteString::FromUnicode(val).c_str();
std::wstring wstrChange = pEvent->Change().c_str();
switch (iIndex) {
case 0:
cFormat = "99999";
break;
case 1:
// cFormat = "99999-9999";
cFormat = "999999999";
break;
case 2: {
std::string NumberStr;
util::printx("9999999999", strSrc, NumberStr);
if (strSrc.length() + wstrChange.length() > 7)
// cFormat = "(999) 999-9999";
cFormat = "9999999999";
else
// cFormat = "999-9999";
cFormat = "9999999";
break;
}
case 3:
// cFormat = "999-99-9999";
cFormat = "999999999";
break;
}
std::vector<CJS_Value> params2;
params2.push_back(CJS_Value(CJS_Runtime::FromContext(cc), cFormat.c_str()));
return AFSpecial_KeystrokeEx(cc, params2, vRet, sError);
}
FX_BOOL CJS_PublicMethods::AFMergeChange(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
CJS_EventHandler* pEventHandler = pContext->GetEventHandler();
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
CFX_WideString swValue;
if (pEventHandler->m_pValue)
swValue = pEventHandler->Value();
if (pEventHandler->WillCommit()) {
vRet = swValue.c_str();
return TRUE;
}
CFX_WideString prefix, postfix;
if (pEventHandler->SelStart() >= 0)
prefix = swValue.Mid(0, pEventHandler->SelStart());
else
prefix = L"";
if (pEventHandler->SelEnd() >= 0 &&
pEventHandler->SelEnd() <= swValue.GetLength())
postfix = swValue.Mid(pEventHandler->SelEnd(),
swValue.GetLength() - pEventHandler->SelEnd());
else
postfix = L"";
vRet = (prefix + pEventHandler->Change() + postfix).c_str();
return TRUE;
}
FX_BOOL CJS_PublicMethods::AFParseDateEx(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
ASSERT(pContext);
if (params.size() != 2) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
CFX_WideString sValue = params[0].ToCFXWideString();
CFX_WideString sFormat = params[1].ToCFXWideString();
double dDate = MakeRegularDate(sValue, sFormat, nullptr);
if (JS_PortIsNan(dDate)) {
CFX_WideString swMsg;
swMsg.Format(JSGetStringFromID(pContext, IDS_STRING_JSPARSEDATE).c_str(),
sFormat.c_str());
Alert((CJS_Context*)cc, swMsg.c_str());
return FALSE;
}
vRet = dDate;
return TRUE;
}
FX_BOOL CJS_PublicMethods::AFSimple(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
if (params.size() != 3) {
CJS_Context* pContext = (CJS_Context*)cc;
ASSERT(pContext);
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
vRet = (double)AF_Simple(params[0].ToCFXWideString().c_str(),
params[1].ToDouble(), params[2].ToDouble());
return TRUE;
}
FX_BOOL CJS_PublicMethods::AFMakeNumber(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
if (params.size() != 1) {
CJS_Context* pContext = (CJS_Context*)cc;
ASSERT(pContext);
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
vRet = ParseStringToNumber(params[0].ToCFXWideString().c_str());
return TRUE;
}
FX_BOOL CJS_PublicMethods::AFSimple_Calculate(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 2) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
CJS_Value params1 = params[1];
if (!params1.IsArrayObject() && params1.GetType() != CJS_Value::VT_string) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
CPDFSDK_Document* pReaderDoc = pContext->GetReaderDocument();
CPDFSDK_InterForm* pReaderInterForm = pReaderDoc->GetInterForm();
CPDF_InterForm* pInterForm = pReaderInterForm->GetInterForm();
CFX_WideString sFunction = params[0].ToCFXWideString();
double dValue = wcscmp(sFunction.c_str(), L"PRD") == 0 ? 1.0 : 0.0;
CJS_Runtime* pRuntime = CJS_Runtime::FromContext(cc);
CJS_Array FieldNameArray = AF_MakeArrayFromList(pRuntime, params1);
int nFieldsCount = 0;
for (int i = 0, isz = FieldNameArray.GetLength(); i < isz; i++) {
CJS_Value jsValue(pRuntime);
FieldNameArray.GetElement(i, jsValue);
CFX_WideString wsFieldName = jsValue.ToCFXWideString();
for (int j = 0, jsz = pInterForm->CountFields(wsFieldName); j < jsz; j++) {
if (CPDF_FormField* pFormField = pInterForm->GetField(j, wsFieldName)) {
double dTemp = 0.0;
switch (pFormField->GetFieldType()) {
case FIELDTYPE_TEXTFIELD:
case FIELDTYPE_COMBOBOX: {
dTemp = ParseStringToNumber(pFormField->GetValue().c_str());
break;
}
case FIELDTYPE_PUSHBUTTON: {
dTemp = 0.0;
break;
}
case FIELDTYPE_CHECKBOX:
case FIELDTYPE_RADIOBUTTON: {
dTemp = 0.0;
for (int c = 0, csz = pFormField->CountControls(); c < csz; c++) {
if (CPDF_FormControl* pFormCtrl = pFormField->GetControl(c)) {
if (pFormCtrl->IsChecked()) {
dTemp +=
ParseStringToNumber(pFormCtrl->GetExportValue().c_str());
break;
}
}
}
break;
}
case FIELDTYPE_LISTBOX: {
if (pFormField->CountSelectedItems() > 1)
break;
dTemp = ParseStringToNumber(pFormField->GetValue().c_str());
break;
}
default:
break;
}
if (i == 0 && j == 0 && (wcscmp(sFunction.c_str(), L"MIN") == 0 ||
wcscmp(sFunction.c_str(), L"MAX") == 0))
dValue = dTemp;
dValue = AF_Simple(sFunction.c_str(), dValue, dTemp);
nFieldsCount++;
}
}
}
if (wcscmp(sFunction.c_str(), L"AVG") == 0 && nFieldsCount > 0)
dValue /= nFieldsCount;
dValue = (double)floor(dValue * FXSYS_pow((double)10, (double)6) + 0.49) /
FXSYS_pow((double)10, (double)6);
CJS_Value jsValue(pRuntime, dValue);
if (pContext->GetEventHandler()->m_pValue)
pContext->GetEventHandler()->Value() = jsValue.ToCFXWideString();
return TRUE;
}
/* This function validates the current event to ensure that its value is
** within the specified range. */
FX_BOOL CJS_PublicMethods::AFRange_Validate(
IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
CJS_EventHandler* pEvent = pContext->GetEventHandler();
if (params.size() != 4) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
if (!pEvent->m_pValue)
return FALSE;
if (pEvent->Value().IsEmpty())
return TRUE;
double dEentValue = atof(CFX_ByteString::FromUnicode(pEvent->Value()));
FX_BOOL bGreaterThan = params[0].ToBool();
double dGreaterThan = params[1].ToDouble();
FX_BOOL bLessThan = params[2].ToBool();
double dLessThan = params[3].ToDouble();
CFX_WideString swMsg;
if (bGreaterThan && bLessThan) {
if (dEentValue < dGreaterThan || dEentValue > dLessThan)
swMsg.Format(JSGetStringFromID(pContext, IDS_STRING_JSRANGE1).c_str(),
params[1].ToCFXWideString().c_str(),
params[3].ToCFXWideString().c_str());
} else if (bGreaterThan) {
if (dEentValue < dGreaterThan)
swMsg.Format(JSGetStringFromID(pContext, IDS_STRING_JSRANGE2).c_str(),
params[1].ToCFXWideString().c_str());
} else if (bLessThan) {
if (dEentValue > dLessThan)
swMsg.Format(JSGetStringFromID(pContext, IDS_STRING_JSRANGE3).c_str(),
params[3].ToCFXWideString().c_str());
}
if (!swMsg.IsEmpty()) {
Alert(pContext, swMsg.c_str());
pEvent->Rc() = FALSE;
}
return TRUE;
}
FX_BOOL CJS_PublicMethods::AFExtractNums(IJS_Context* cc,
const std::vector<CJS_Value>& params,
CJS_Value& vRet,
CFX_WideString& sError) {
CJS_Context* pContext = (CJS_Context*)cc;
if (params.size() != 1) {
sError = JSGetStringFromID(pContext, IDS_STRING_JSPARAMERROR);
return FALSE;
}
CJS_Runtime* pRuntime = CJS_Runtime::FromContext(cc);
CJS_Array nums(pRuntime);
CFX_WideString str = params[0].ToCFXWideString();
CFX_WideString sPart;
if (str.GetAt(0) == L'.' || str.GetAt(0) == L',')
str = L"0" + str;
int nIndex = 0;
for (int i = 0, sz = str.GetLength(); i < sz; i++) {
FX_WCHAR wc = str.GetAt(i);
if (FXSYS_iswdigit(wc)) {
sPart += wc;
} else {
if (sPart.GetLength() > 0) {
nums.SetElement(nIndex, CJS_Value(pRuntime, sPart.c_str()));
sPart = L"";
nIndex++;
}
}
}
if (sPart.GetLength() > 0) {
nums.SetElement(nIndex, CJS_Value(pRuntime, sPart.c_str()));
}
if (nums.GetLength() > 0)
vRet = nums;
else
vRet.SetNull();
return TRUE;
}