summaryrefslogtreecommitdiff
path: root/xfa/fxfa/fm2js
diff options
context:
space:
mode:
Diffstat (limited to 'xfa/fxfa/fm2js')
-rw-r--r--xfa/fxfa/fm2js/cxfa_fm2jscontext.cpp6159
-rw-r--r--xfa/fxfa/fm2js/cxfa_fm2jscontext.h445
-rw-r--r--xfa/fxfa/fm2js/cxfa_fm2jscontext_embeddertest.cpp1446
3 files changed, 0 insertions, 8050 deletions
diff --git a/xfa/fxfa/fm2js/cxfa_fm2jscontext.cpp b/xfa/fxfa/fm2js/cxfa_fm2jscontext.cpp
deleted file mode 100644
index 2f271d8c60..0000000000
--- a/xfa/fxfa/fm2js/cxfa_fm2jscontext.cpp
+++ /dev/null
@@ -1,6159 +0,0 @@
-// 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 "xfa/fxfa/fm2js/cxfa_fm2jscontext.h"
-
-#include <time.h>
-
-#include <algorithm>
-#include <string>
-
-#include "core/fxcrt/cfx_decimal.h"
-#include "core/fxcrt/cfx_widetextbuf.h"
-#include "core/fxcrt/fx_extension.h"
-#include "core/fxcrt/fx_random.h"
-#include "fxjs/cfxjse_arguments.h"
-#include "fxjs/cfxjse_class.h"
-#include "fxjs/cfxjse_value.h"
-#include "third_party/base/ptr_util.h"
-#include "third_party/base/stl_util.h"
-#include "xfa/fxfa/cxfa_ffnotify.h"
-#include "xfa/fxfa/fm2js/cxfa_fmparser.h"
-#include "xfa/fxfa/fm2js/cxfa_fmtojavascriptdepth.h"
-#include "xfa/fxfa/parser/cxfa_document.h"
-#include "xfa/fxfa/parser/cxfa_localevalue.h"
-#include "xfa/fxfa/parser/cxfa_node.h"
-#include "xfa/fxfa/parser/cxfa_scriptcontext.h"
-#include "xfa/fxfa/parser/cxfa_timezoneprovider.h"
-#include "xfa/fxfa/parser/xfa_utils.h"
-
-namespace {
-
-const double kFinancialPrecision = 0.00000001;
-
-struct XFA_FMHtmlReserveCode {
- uint32_t m_uCode;
- const wchar_t* m_htmlReserve;
-};
-
-// Sorted by m_htmlReserve
-XFA_FMHtmlReserveCode reservesForDecode[] = {
- {198, L"AElig"}, {193, L"Aacute"}, {194, L"Acirc"},
- {192, L"Agrave"}, {913, L"Alpha"}, {197, L"Aring"},
- {195, L"Atilde"}, {196, L"Auml"}, {914, L"Beta"},
- {199, L"Ccedil"}, {935, L"Chi"}, {8225, L"Dagger"},
- {916, L"Delta"}, {208, L"ETH"}, {201, L"Eacute"},
- {202, L"Ecirc"}, {200, L"Egrave"}, {917, L"Epsilon"},
- {919, L"Eta"}, {203, L"Euml"}, {915, L"Gamma"},
- {922, L"Kappa"}, {923, L"Lambda"}, {924, L"Mu"},
- {209, L"Ntilde"}, {925, L"Nu"}, {338, L"OElig"},
- {211, L"Oacute"}, {212, L"Ocirc"}, {210, L"Ograve"},
- {937, L"Omega"}, {927, L"Omicron"}, {216, L"Oslash"},
- {213, L"Otilde"}, {214, L"Ouml"}, {934, L"Phi"},
- {928, L"Pi"}, {936, L"Psi"}, {929, L"Rho"},
- {352, L"Scaron"}, {931, L"Sigma"}, {222, L"THORN"},
- {932, L"Tau"}, {920, L"Theta"}, {218, L"Uacute"},
- {219, L"Ucirc"}, {217, L"Ugrave"}, {933, L"Upsilon"},
- {220, L"Uuml"}, {926, L"Xi"}, {221, L"Yacute"},
- {376, L"Yuml"}, {918, L"Zeta"}, {225, L"aacute"},
- {226, L"acirc"}, {180, L"acute"}, {230, L"aelig"},
- {224, L"agrave"}, {8501, L"alefsym"}, {945, L"alpha"},
- {38, L"amp"}, {8743, L"and"}, {8736, L"ang"},
- {39, L"apos"}, {229, L"aring"}, {8776, L"asymp"},
- {227, L"atilde"}, {228, L"auml"}, {8222, L"bdquo"},
- {946, L"beta"}, {166, L"brvbar"}, {8226, L"bull"},
- {8745, L"cap"}, {231, L"ccedil"}, {184, L"cedil"},
- {162, L"cent"}, {967, L"chi"}, {710, L"circ"},
- {9827, L"clubs"}, {8773, L"cong"}, {169, L"copy"},
- {8629, L"crarr"}, {8746, L"cup"}, {164, L"current"},
- {8659, L"dArr"}, {8224, L"dagger"}, {8595, L"darr"},
- {176, L"deg"}, {948, L"delta"}, {9830, L"diams"},
- {247, L"divide"}, {233, L"eacute"}, {234, L"ecirc"},
- {232, L"egrave"}, {8709, L"empty"}, {8195, L"emsp"},
- {8194, L"ensp"}, {949, L"epsilon"}, {8801, L"equiv"},
- {951, L"eta"}, {240, L"eth"}, {235, L"euml"},
- {8364, L"euro"}, {8707, L"exist"}, {402, L"fnof"},
- {8704, L"forall"}, {189, L"frac12"}, {188, L"frac14"},
- {190, L"frac34"}, {8260, L"frasl"}, {947, L"gamma"},
- {8805, L"ge"}, {62, L"gt"}, {8660, L"hArr"},
- {8596, L"harr"}, {9829, L"hearts"}, {8230, L"hellip"},
- {237, L"iacute"}, {238, L"icirc"}, {161, L"iexcl"},
- {236, L"igrave"}, {8465, L"image"}, {8734, L"infin"},
- {8747, L"int"}, {953, L"iota"}, {191, L"iquest"},
- {8712, L"isin"}, {239, L"iuml"}, {954, L"kappa"},
- {8656, L"lArr"}, {205, L"lacute"}, {955, L"lambda"},
- {9001, L"lang"}, {171, L"laquo"}, {8592, L"larr"},
- {8968, L"lceil"}, {206, L"lcirc"}, {8220, L"ldquo"},
- {8804, L"le"}, {8970, L"lfloor"}, {204, L"lgrave"},
- {921, L"lota"}, {8727, L"lowast"}, {9674, L"loz"},
- {8206, L"lrm"}, {8249, L"lsaquo"}, {8216, L"lsquo"},
- {60, L"lt"}, {207, L"luml"}, {175, L"macr"},
- {8212, L"mdash"}, {181, L"micro"}, {183, L"middot"},
- {8722, L"minus"}, {956, L"mu"}, {8711, L"nabla"},
- {160, L"nbsp"}, {8211, L"ndash"}, {8800, L"ne"},
- {8715, L"ni"}, {172, L"not"}, {8713, L"notin"},
- {8836, L"nsub"}, {241, L"ntilde"}, {957, L"nu"},
- {243, L"oacute"}, {244, L"ocirc"}, {339, L"oelig"},
- {242, L"ograve"}, {8254, L"oline"}, {969, L"omega"},
- {959, L"omicron"}, {8853, L"oplus"}, {8744, L"or"},
- {170, L"ordf"}, {186, L"ordm"}, {248, L"oslash"},
- {245, L"otilde"}, {8855, L"otimes"}, {246, L"ouml"},
- {182, L"para"}, {8706, L"part"}, {8240, L"permil"},
- {8869, L"perp"}, {966, L"phi"}, {960, L"pi"},
- {982, L"piv"}, {177, L"plusmn"}, {8242, L"prime"},
- {8719, L"prod"}, {8733, L"prop"}, {968, L"psi"},
- {163, L"pund"}, {34, L"quot"}, {8658, L"rArr"},
- {8730, L"radic"}, {9002, L"rang"}, {187, L"raquo"},
- {8594, L"rarr"}, {8969, L"rceil"}, {8476, L"real"},
- {174, L"reg"}, {8971, L"rfloor"}, {961, L"rho"},
- {8207, L"rlm"}, {8250, L"rsaquo"}, {8217, L"rsquo"},
- {353, L"saron"}, {8218, L"sbquo"}, {8901, L"sdot"},
- {167, L"sect"}, {173, L"shy"}, {963, L"sigma"},
- {962, L"sigmaf"}, {8764, L"sim"}, {9824, L"spades"},
- {8834, L"sub"}, {8838, L"sube"}, {8721, L"sum"},
- {8835, L"sup"}, {185, L"sup1"}, {178, L"sup2"},
- {179, L"sup3"}, {8839, L"supe"}, {223, L"szlig"},
- {964, L"tau"}, {8221, L"tdquo"}, {8756, L"there4"},
- {952, L"theta"}, {977, L"thetasym"}, {8201, L"thinsp"},
- {254, L"thorn"}, {732, L"tilde"}, {215, L"times"},
- {8482, L"trade"}, {8657, L"uArr"}, {250, L"uacute"},
- {8593, L"uarr"}, {251, L"ucirc"}, {249, L"ugrave"},
- {168, L"uml"}, {978, L"upsih"}, {965, L"upsilon"},
- {252, L"uuml"}, {8472, L"weierp"}, {958, L"xi"},
- {253, L"yacute"}, {165, L"yen"}, {255, L"yuml"},
- {950, L"zeta"}, {8205, L"zwj"}, {8204, L"zwnj"},
-};
-
-// Sorted by m_uCode
-const XFA_FMHtmlReserveCode reservesForEncode[] = {
- {34, L"quot"}, {38, L"amp"}, {39, L"apos"},
- {60, L"lt"}, {62, L"gt"}, {160, L"nbsp"},
- {161, L"iexcl"}, {162, L"cent"}, {163, L"pund"},
- {164, L"current"}, {165, L"yen"}, {166, L"brvbar"},
- {167, L"sect"}, {168, L"uml"}, {169, L"copy"},
- {170, L"ordf"}, {171, L"laquo"}, {172, L"not"},
- {173, L"shy"}, {174, L"reg"}, {175, L"macr"},
- {176, L"deg"}, {177, L"plusmn"}, {178, L"sup2"},
- {179, L"sup3"}, {180, L"acute"}, {181, L"micro"},
- {182, L"para"}, {183, L"middot"}, {184, L"cedil"},
- {185, L"sup1"}, {186, L"ordm"}, {187, L"raquo"},
- {188, L"frac14"}, {189, L"frac12"}, {190, L"frac34"},
- {191, L"iquest"}, {192, L"Agrave"}, {193, L"Aacute"},
- {194, L"Acirc"}, {195, L"Atilde"}, {196, L"Auml"},
- {197, L"Aring"}, {198, L"AElig"}, {199, L"Ccedil"},
- {200, L"Egrave"}, {201, L"Eacute"}, {202, L"Ecirc"},
- {203, L"Euml"}, {204, L"lgrave"}, {205, L"lacute"},
- {206, L"lcirc"}, {207, L"luml"}, {208, L"ETH"},
- {209, L"Ntilde"}, {210, L"Ograve"}, {211, L"Oacute"},
- {212, L"Ocirc"}, {213, L"Otilde"}, {214, L"Ouml"},
- {215, L"times"}, {216, L"Oslash"}, {217, L"Ugrave"},
- {218, L"Uacute"}, {219, L"Ucirc"}, {220, L"Uuml"},
- {221, L"Yacute"}, {222, L"THORN"}, {223, L"szlig"},
- {224, L"agrave"}, {225, L"aacute"}, {226, L"acirc"},
- {227, L"atilde"}, {228, L"auml"}, {229, L"aring"},
- {230, L"aelig"}, {231, L"ccedil"}, {232, L"egrave"},
- {233, L"eacute"}, {234, L"ecirc"}, {235, L"euml"},
- {236, L"igrave"}, {237, L"iacute"}, {238, L"icirc"},
- {239, L"iuml"}, {240, L"eth"}, {241, L"ntilde"},
- {242, L"ograve"}, {243, L"oacute"}, {244, L"ocirc"},
- {245, L"otilde"}, {246, L"ouml"}, {247, L"divide"},
- {248, L"oslash"}, {249, L"ugrave"}, {250, L"uacute"},
- {251, L"ucirc"}, {252, L"uuml"}, {253, L"yacute"},
- {254, L"thorn"}, {255, L"yuml"}, {338, L"OElig"},
- {339, L"oelig"}, {352, L"Scaron"}, {353, L"saron"},
- {376, L"Yuml"}, {402, L"fnof"}, {710, L"circ"},
- {732, L"tilde"}, {913, L"Alpha"}, {914, L"Beta"},
- {915, L"Gamma"}, {916, L"Delta"}, {917, L"Epsilon"},
- {918, L"Zeta"}, {919, L"Eta"}, {920, L"Theta"},
- {921, L"lota"}, {922, L"Kappa"}, {923, L"Lambda"},
- {924, L"Mu"}, {925, L"Nu"}, {926, L"Xi"},
- {927, L"Omicron"}, {928, L"Pi"}, {929, L"Rho"},
- {931, L"Sigma"}, {932, L"Tau"}, {933, L"Upsilon"},
- {934, L"Phi"}, {935, L"Chi"}, {936, L"Psi"},
- {937, L"Omega"}, {945, L"alpha"}, {946, L"beta"},
- {947, L"gamma"}, {948, L"delta"}, {949, L"epsilon"},
- {950, L"zeta"}, {951, L"eta"}, {952, L"theta"},
- {953, L"iota"}, {954, L"kappa"}, {955, L"lambda"},
- {956, L"mu"}, {957, L"nu"}, {958, L"xi"},
- {959, L"omicron"}, {960, L"pi"}, {961, L"rho"},
- {962, L"sigmaf"}, {963, L"sigma"}, {964, L"tau"},
- {965, L"upsilon"}, {966, L"phi"}, {967, L"chi"},
- {968, L"psi"}, {969, L"omega"}, {977, L"thetasym"},
- {978, L"upsih"}, {982, L"piv"}, {8194, L"ensp"},
- {8195, L"emsp"}, {8201, L"thinsp"}, {8204, L"zwnj"},
- {8205, L"zwj"}, {8206, L"lrm"}, {8207, L"rlm"},
- {8211, L"ndash"}, {8212, L"mdash"}, {8216, L"lsquo"},
- {8217, L"rsquo"}, {8218, L"sbquo"}, {8220, L"ldquo"},
- {8221, L"tdquo"}, {8222, L"bdquo"}, {8224, L"dagger"},
- {8225, L"Dagger"}, {8226, L"bull"}, {8230, L"hellip"},
- {8240, L"permil"}, {8242, L"prime"}, {8249, L"lsaquo"},
- {8250, L"rsaquo"}, {8254, L"oline"}, {8260, L"frasl"},
- {8364, L"euro"}, {8465, L"image"}, {8472, L"weierp"},
- {8476, L"real"}, {8482, L"trade"}, {8501, L"alefsym"},
- {8592, L"larr"}, {8593, L"uarr"}, {8594, L"rarr"},
- {8595, L"darr"}, {8596, L"harr"}, {8629, L"crarr"},
- {8656, L"lArr"}, {8657, L"uArr"}, {8658, L"rArr"},
- {8659, L"dArr"}, {8660, L"hArr"}, {8704, L"forall"},
- {8706, L"part"}, {8707, L"exist"}, {8709, L"empty"},
- {8711, L"nabla"}, {8712, L"isin"}, {8713, L"notin"},
- {8715, L"ni"}, {8719, L"prod"}, {8721, L"sum"},
- {8722, L"minus"}, {8727, L"lowast"}, {8730, L"radic"},
- {8733, L"prop"}, {8734, L"infin"}, {8736, L"ang"},
- {8743, L"and"}, {8744, L"or"}, {8745, L"cap"},
- {8746, L"cup"}, {8747, L"int"}, {8756, L"there4"},
- {8764, L"sim"}, {8773, L"cong"}, {8776, L"asymp"},
- {8800, L"ne"}, {8801, L"equiv"}, {8804, L"le"},
- {8805, L"ge"}, {8834, L"sub"}, {8835, L"sup"},
- {8836, L"nsub"}, {8838, L"sube"}, {8839, L"supe"},
- {8853, L"oplus"}, {8855, L"otimes"}, {8869, L"perp"},
- {8901, L"sdot"}, {8968, L"lceil"}, {8969, L"rceil"},
- {8970, L"lfloor"}, {8971, L"rfloor"}, {9001, L"lang"},
- {9002, L"rang"}, {9674, L"loz"}, {9824, L"spades"},
- {9827, L"clubs"}, {9829, L"hearts"}, {9830, L"diams"},
-};
-
-const FXJSE_FUNCTION_DESCRIPTOR formcalc_fm2js_functions[] = {
- {"Abs", CXFA_FM2JSContext::Abs},
- {"Avg", CXFA_FM2JSContext::Avg},
- {"Ceil", CXFA_FM2JSContext::Ceil},
- {"Count", CXFA_FM2JSContext::Count},
- {"Floor", CXFA_FM2JSContext::Floor},
- {"Max", CXFA_FM2JSContext::Max},
- {"Min", CXFA_FM2JSContext::Min},
- {"Mod", CXFA_FM2JSContext::Mod},
- {"Round", CXFA_FM2JSContext::Round},
- {"Sum", CXFA_FM2JSContext::Sum},
- {"Date", CXFA_FM2JSContext::Date},
- {"Date2Num", CXFA_FM2JSContext::Date2Num},
- {"DateFmt", CXFA_FM2JSContext::DateFmt},
- {"IsoDate2Num", CXFA_FM2JSContext::IsoDate2Num},
- {"IsoTime2Num", CXFA_FM2JSContext::IsoTime2Num},
- {"LocalDateFmt", CXFA_FM2JSContext::LocalDateFmt},
- {"LocalTimeFmt", CXFA_FM2JSContext::LocalTimeFmt},
- {"Num2Date", CXFA_FM2JSContext::Num2Date},
- {"Num2GMTime", CXFA_FM2JSContext::Num2GMTime},
- {"Num2Time", CXFA_FM2JSContext::Num2Time},
- {"Time", CXFA_FM2JSContext::Time},
- {"Time2Num", CXFA_FM2JSContext::Time2Num},
- {"TimeFmt", CXFA_FM2JSContext::TimeFmt},
- {"Apr", CXFA_FM2JSContext::Apr},
- {"Cterm", CXFA_FM2JSContext::CTerm},
- {"FV", CXFA_FM2JSContext::FV},
- {"Ipmt", CXFA_FM2JSContext::IPmt},
- {"NPV", CXFA_FM2JSContext::NPV},
- {"Pmt", CXFA_FM2JSContext::Pmt},
- {"PPmt", CXFA_FM2JSContext::PPmt},
- {"PV", CXFA_FM2JSContext::PV},
- {"Rate", CXFA_FM2JSContext::Rate},
- {"Term", CXFA_FM2JSContext::Term},
- {"Choose", CXFA_FM2JSContext::Choose},
- {"Exists", CXFA_FM2JSContext::Exists},
- {"HasValue", CXFA_FM2JSContext::HasValue},
- {"Oneof", CXFA_FM2JSContext::Oneof},
- {"Within", CXFA_FM2JSContext::Within},
- {"If", CXFA_FM2JSContext::If},
- {"Eval", CXFA_FM2JSContext::Eval},
- {"Translate", CXFA_FM2JSContext::eval_translation},
- {"Ref", CXFA_FM2JSContext::Ref},
- {"UnitType", CXFA_FM2JSContext::UnitType},
- {"UnitValue", CXFA_FM2JSContext::UnitValue},
- {"At", CXFA_FM2JSContext::At},
- {"Concat", CXFA_FM2JSContext::Concat},
- {"Decode", CXFA_FM2JSContext::Decode},
- {"Encode", CXFA_FM2JSContext::Encode},
- {"Format", CXFA_FM2JSContext::Format},
- {"Left", CXFA_FM2JSContext::Left},
- {"Len", CXFA_FM2JSContext::Len},
- {"Lower", CXFA_FM2JSContext::Lower},
- {"Ltrim", CXFA_FM2JSContext::Ltrim},
- {"Parse", CXFA_FM2JSContext::Parse},
- {"Replace", CXFA_FM2JSContext::Replace},
- {"Right", CXFA_FM2JSContext::Right},
- {"Rtrim", CXFA_FM2JSContext::Rtrim},
- {"Space", CXFA_FM2JSContext::Space},
- {"Str", CXFA_FM2JSContext::Str},
- {"Stuff", CXFA_FM2JSContext::Stuff},
- {"Substr", CXFA_FM2JSContext::Substr},
- {"Uuid", CXFA_FM2JSContext::Uuid},
- {"Upper", CXFA_FM2JSContext::Upper},
- {"WordNum", CXFA_FM2JSContext::WordNum},
- {"Get", CXFA_FM2JSContext::Get},
- {"Post", CXFA_FM2JSContext::Post},
- {"Put", CXFA_FM2JSContext::Put},
- {"pos_op", CXFA_FM2JSContext::positive_operator},
- {"neg_op", CXFA_FM2JSContext::negative_operator},
- {"log_or_op", CXFA_FM2JSContext::logical_or_operator},
- {"log_and_op", CXFA_FM2JSContext::logical_and_operator},
- {"log_not_op", CXFA_FM2JSContext::logical_not_operator},
- {"eq_op", CXFA_FM2JSContext::equality_operator},
- {"neq_op", CXFA_FM2JSContext::notequality_operator},
- {"lt_op", CXFA_FM2JSContext::less_operator},
- {"le_op", CXFA_FM2JSContext::lessequal_operator},
- {"gt_op", CXFA_FM2JSContext::greater_operator},
- {"ge_op", CXFA_FM2JSContext::greaterequal_operator},
- {"plus_op", CXFA_FM2JSContext::plus_operator},
- {"minus_op", CXFA_FM2JSContext::minus_operator},
- {"mul_op", CXFA_FM2JSContext::multiple_operator},
- {"div_op", CXFA_FM2JSContext::divide_operator},
- {"asgn_val_op", CXFA_FM2JSContext::assign_value_operator},
- {"dot_acc", CXFA_FM2JSContext::dot_accessor},
- {"dotdot_acc", CXFA_FM2JSContext::dotdot_accessor},
- {"concat_obj", CXFA_FM2JSContext::concat_fm_object},
- {"is_obj", CXFA_FM2JSContext::is_fm_object},
- {"is_ary", CXFA_FM2JSContext::is_fm_array},
- {"get_val", CXFA_FM2JSContext::get_fm_value},
- {"get_jsobj", CXFA_FM2JSContext::get_fm_jsobj},
- {"var_filter", CXFA_FM2JSContext::fm_var_filter},
-};
-
-const FXJSE_CLASS_DESCRIPTOR formcalc_fm2js_descriptor = {
- "XFA_FM2JS_FormCalcClass", // name
- nullptr, // constructor
- nullptr, // properties
- formcalc_fm2js_functions, // methods
- 0, // number of properties
- FX_ArraySize(formcalc_fm2js_functions), // number of methods
- nullptr, // dynamic prop type
- nullptr, // dynamic prop getter
- nullptr, // dynamic prop setter
- nullptr, // dynamic prop deleter
- nullptr, // dynamic prop method call
-};
-
-const uint8_t g_sAltTable_Date[] = {
- 255, 255, 255, 3, 9, 255, 255, 255, 255, 255, 255,
- 255, 2, 255, 255, 255, 255, 255, 255, 255, 255, 255,
- 255, 255, 1, 255, 255, 255, 255, 255, 255, 255, 255,
-};
-static_assert(FX_ArraySize(g_sAltTable_Date) == L'a' - L'A' + 1,
- "Invalid g_sAltTable_Date size.");
-
-const uint8_t g_sAltTable_Time[] = {
- 14, 255, 255, 3, 9, 255, 255, 15, 255, 255, 255,
- 255, 6, 255, 255, 255, 255, 255, 7, 255, 255, 255,
- 255, 255, 1, 17, 255, 255, 255, 255, 255, 255, 255,
-};
-static_assert(FX_ArraySize(g_sAltTable_Time) == L'a' - L'A' + 1,
- "Invalid g_sAltTable_Time size.");
-
-void AlternateDateTimeSymbols(WideString& wsPattern,
- const WideString& wsAltSymbols,
- const uint8_t* pAltTable) {
- int32_t nLength = wsPattern.GetLength();
- bool bInConstRange = false;
- bool bEscape = false;
- int32_t i = 0;
- while (i < nLength) {
- wchar_t wc = wsPattern[i];
- if (wc == L'\'') {
- bInConstRange = !bInConstRange;
- if (bEscape) {
- i++;
- } else {
- wsPattern.Delete(i);
- nLength--;
- }
- bEscape = !bEscape;
- continue;
- }
- if (!bInConstRange && wc >= L'A' && wc <= L'a') {
- uint8_t nAlt = pAltTable[wc - L'A'];
- if (nAlt != 255)
- wsPattern.SetAt(i, wsAltSymbols[nAlt]);
- }
- i++;
- bEscape = false;
- }
-}
-
-bool PatternStringType(const ByteStringView& szPattern, uint32_t& patternType) {
- WideString wsPattern = WideString::FromUTF8(szPattern);
- if (L"datetime" == wsPattern.Left(8)) {
- patternType = XFA_VT_DATETIME;
- return true;
- }
- if (L"date" == wsPattern.Left(4)) {
- auto pos = wsPattern.Find(L"time");
- patternType =
- pos.has_value() && pos.value() != 0 ? XFA_VT_DATETIME : XFA_VT_DATE;
- return true;
- }
- if (L"time" == wsPattern.Left(4)) {
- patternType = XFA_VT_TIME;
- return true;
- }
- if (L"text" == wsPattern.Left(4)) {
- patternType = XFA_VT_TEXT;
- return true;
- }
- if (L"num" == wsPattern.Left(3)) {
- if (L"integer" == wsPattern.Mid(4, 7)) {
- patternType = XFA_VT_INTEGER;
- } else if (L"decimal" == wsPattern.Mid(4, 7)) {
- patternType = XFA_VT_DECIMAL;
- } else if (L"currency" == wsPattern.Mid(4, 8)) {
- patternType = XFA_VT_FLOAT;
- } else if (L"percent" == wsPattern.Mid(4, 7)) {
- patternType = XFA_VT_FLOAT;
- } else {
- patternType = XFA_VT_FLOAT;
- }
- return true;
- }
-
- patternType = XFA_VT_NULL;
- wsPattern.MakeLower();
- const wchar_t* pData = wsPattern.c_str();
- int32_t iLength = wsPattern.GetLength();
- int32_t iIndex = 0;
- bool bSingleQuotation = false;
- wchar_t patternChar;
- while (iIndex < iLength) {
- patternChar = pData[iIndex];
- if (patternChar == 0x27) {
- bSingleQuotation = !bSingleQuotation;
- } else if (!bSingleQuotation &&
- (patternChar == 'y' || patternChar == 'j')) {
- patternType = XFA_VT_DATE;
- iIndex++;
- wchar_t timePatternChar;
- while (iIndex < iLength) {
- timePatternChar = pData[iIndex];
- if (timePatternChar == 0x27) {
- bSingleQuotation = !bSingleQuotation;
- } else if (!bSingleQuotation && timePatternChar == 't') {
- patternType = XFA_VT_DATETIME;
- break;
- }
- iIndex++;
- }
- break;
- } else if (!bSingleQuotation &&
- (patternChar == 'h' || patternChar == 'k')) {
- patternType = XFA_VT_TIME;
- break;
- } else if (!bSingleQuotation &&
- (patternChar == 'a' || patternChar == 'x' ||
- patternChar == 'o' || patternChar == '0')) {
- patternType = XFA_VT_TEXT;
- if (patternChar == 'x' || patternChar == 'o' || patternChar == '0') {
- break;
- }
- } else if (!bSingleQuotation &&
- (patternChar == 'z' || patternChar == 's' ||
- patternChar == 'e' || patternChar == 'v' ||
- patternChar == '8' || patternChar == ',' ||
- patternChar == '.' || patternChar == '$')) {
- patternType = XFA_VT_FLOAT;
- if (patternChar == 'v' || patternChar == '8' || patternChar == '$') {
- break;
- }
- }
- iIndex++;
- }
- if (patternType == XFA_VT_NULL) {
- patternType = XFA_VT_TEXT | XFA_VT_FLOAT;
- }
- return false;
-}
-
-CXFA_FM2JSContext* ToJSContext(CFXJSE_Value* pValue, CFXJSE_Class* pClass) {
- CFXJSE_HostObject* pHostObj = pValue->ToHostObject(pClass);
- if (!pHostObj || pHostObj->type() != CFXJSE_HostObject::kFM2JS)
- return nullptr;
- return static_cast<CXFA_FM2JSContext*>(pHostObj);
-}
-
-bool IsWhitespace(char c) {
- return c == 0x20 || c == 0x09 || c == 0x0B || c == 0x0C || c == 0x0A ||
- c == 0x0D;
-}
-
-IFX_Locale* LocaleFromString(CXFA_Document* pDoc,
- CXFA_LocaleMgr* pMgr,
- const ByteStringView& szLocale) {
- if (!szLocale.IsEmpty())
- return pMgr->GetLocaleByName(WideString::FromUTF8(szLocale));
-
- CXFA_Node* pThisNode = ToNode(pDoc->GetScriptContext()->GetThisObject());
- ASSERT(pThisNode);
- return CXFA_WidgetData(pThisNode).GetLocal();
-}
-
-WideString FormatFromString(IFX_Locale* pLocale,
- const ByteStringView& szFormat) {
- if (!szFormat.IsEmpty())
- return WideString::FromUTF8(szFormat);
-
- return pLocale->GetDatePattern(FX_LOCALEDATETIMESUBCATEGORY_Default);
-}
-
-FX_LOCALEDATETIMESUBCATEGORY SubCategoryFromInt(int32_t iStyle) {
- switch (iStyle) {
- case 1:
- return FX_LOCALEDATETIMESUBCATEGORY_Short;
- case 3:
- return FX_LOCALEDATETIMESUBCATEGORY_Long;
- case 4:
- return FX_LOCALEDATETIMESUBCATEGORY_Full;
- case 0:
- case 2:
- default:
- return FX_LOCALEDATETIMESUBCATEGORY_Medium;
- }
-}
-
-bool IsPartOfNumber(char ch) {
- return std::isdigit(ch) || ch == '-' || ch == '.';
-}
-
-bool IsPartOfNumberW(wchar_t ch) {
- return std::iswdigit(ch) || ch == L'-' || ch == L'.';
-}
-
-ByteString GUIDString(bool bSeparator) {
- uint8_t data[16];
- FX_Random_GenerateMT(reinterpret_cast<uint32_t*>(data), 4);
- data[6] = (data[6] & 0x0F) | 0x40;
-
- ByteString bsStr;
- char* pBuf = bsStr.GetBuffer(40);
- for (int32_t i = 0; i < 16; ++i, pBuf += 2) {
- if (bSeparator && (i == 4 || i == 6 || i == 8 || i == 10))
- *pBuf++ = L'-';
-
- FXSYS_IntToTwoHexChars(data[i], pBuf);
- }
- bsStr.ReleaseBuffer(bSeparator ? 36 : 32);
- return bsStr;
-}
-
-} // namespace
-
-// static
-void CXFA_FM2JSContext::Abs(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Abs");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double dValue = ValueToDouble(pThis, argOne.get());
- if (dValue < 0)
- dValue = -dValue;
-
- args.GetReturnValue()->SetDouble(dValue);
-}
-
-// static
-void CXFA_FM2JSContext::Avg(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- uint32_t uCount = 0;
- double dSum = 0.0;
- for (int32_t i = 0; i < argc; i++) {
- std::unique_ptr<CFXJSE_Value> argValue = args.GetValue(i);
- if (argValue->IsNull())
- continue;
-
- if (!argValue->IsArray()) {
- dSum += ValueToDouble(pThis, argValue.get());
- uCount++;
- continue;
- }
-
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
-
- if (iLength > 2) {
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectPropertyByIdx(1, propertyValue.get());
-
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- if (propertyValue->IsNull()) {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- auto defaultPropValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(jsObjectValue.get(), defaultPropValue.get());
- if (defaultPropValue->IsNull())
- continue;
-
- dSum += ValueToDouble(pThis, defaultPropValue.get());
- uCount++;
- }
- } else {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- jsObjectValue->GetObjectProperty(
- propertyValue->ToString().AsStringView(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- dSum += ValueToDouble(pThis, newPropertyValue.get());
- uCount++;
- }
- }
- }
- }
- if (uCount == 0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- args.GetReturnValue()->SetDouble(dSum / uCount);
-}
-
-// static
-void CXFA_FM2JSContext::Ceil(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Ceil");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argValue = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- args.GetReturnValue()->SetFloat(ceil(ValueToFloat(pThis, argValue.get())));
-}
-
-// static
-void CXFA_FM2JSContext::Count(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- int32_t iCount = 0;
- for (int32_t i = 0; i < args.GetLength(); i++) {
- std::unique_ptr<CFXJSE_Value> argValue = args.GetValue(i);
- if (argValue->IsNull())
- continue;
-
- if (argValue->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectProperty("length", lengthValue.get());
-
- int32_t iLength = lengthValue->ToInteger();
- if (iLength <= 2) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectPropertyByIdx(1, propertyValue.get());
- argValue->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- if (!newPropertyValue->IsNull())
- iCount++;
- }
- } else {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- jsObjectValue->GetObjectProperty(
- propertyValue->ToString().AsStringView(), newPropertyValue.get());
- iCount += newPropertyValue->IsNull() ? 0 : 1;
- }
- }
- } else if (argValue->IsObject()) {
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(argValue.get(), newPropertyValue.get());
- if (!newPropertyValue->IsNull())
- iCount++;
- } else {
- iCount++;
- }
- }
- args.GetReturnValue()->SetInteger(iCount);
-}
-
-// static
-void CXFA_FM2JSContext::Floor(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Floor");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argValue = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- args.GetReturnValue()->SetFloat(floor(ValueToFloat(pThis, argValue.get())));
-}
-
-// static
-void CXFA_FM2JSContext::Max(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- uint32_t uCount = 0;
- double dMaxValue = 0.0;
- for (int32_t i = 0; i < args.GetLength(); i++) {
- std::unique_ptr<CFXJSE_Value> argValue = args.GetValue(i);
- if (argValue->IsNull())
- continue;
-
- if (argValue->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- if (iLength <= 2) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectPropertyByIdx(1, propertyValue.get());
- argValue->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- uCount++;
- double dValue = ValueToDouble(pThis, newPropertyValue.get());
- dMaxValue = (uCount == 1) ? dValue : std::max(dMaxValue, dValue);
- }
- } else {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- jsObjectValue->GetObjectProperty(
- propertyValue->ToString().AsStringView(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- uCount++;
- double dValue = ValueToDouble(pThis, newPropertyValue.get());
- dMaxValue = (uCount == 1) ? dValue : std::max(dMaxValue, dValue);
- }
- }
- } else if (argValue->IsObject()) {
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(argValue.get(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- uCount++;
- double dValue = ValueToDouble(pThis, newPropertyValue.get());
- dMaxValue = (uCount == 1) ? dValue : std::max(dMaxValue, dValue);
- } else {
- uCount++;
- double dValue = ValueToDouble(pThis, argValue.get());
- dMaxValue = (uCount == 1) ? dValue : std::max(dMaxValue, dValue);
- }
- }
- if (uCount == 0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- args.GetReturnValue()->SetDouble(dMaxValue);
-}
-
-// static
-void CXFA_FM2JSContext::Min(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- uint32_t uCount = 0;
- double dMinValue = 0.0;
- for (int32_t i = 0; i < args.GetLength(); i++) {
- std::unique_ptr<CFXJSE_Value> argValue = args.GetValue(i);
- if (argValue->IsNull())
- continue;
-
- if (argValue->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- if (iLength <= 2) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectPropertyByIdx(1, propertyValue.get());
- argValue->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- uCount++;
- double dValue = ValueToDouble(pThis, newPropertyValue.get());
- dMinValue = uCount == 1 ? dValue : std::min(dMinValue, dValue);
- }
- } else {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- jsObjectValue->GetObjectProperty(
- propertyValue->ToString().AsStringView(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- uCount++;
- double dValue = ValueToDouble(pThis, newPropertyValue.get());
- dMinValue = uCount == 1 ? dValue : std::min(dMinValue, dValue);
- }
- }
- } else if (argValue->IsObject()) {
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(argValue.get(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- uCount++;
- double dValue = ValueToDouble(pThis, newPropertyValue.get());
- dMinValue = uCount == 1 ? dValue : std::min(dMinValue, dValue);
- } else {
- uCount++;
- double dValue = ValueToDouble(pThis, argValue.get());
- dMinValue = uCount == 1 ? dValue : std::min(dMinValue, dValue);
- }
- }
- if (uCount == 0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- args.GetReturnValue()->SetDouble(dMinValue);
-}
-
-// static
-void CXFA_FM2JSContext::Mod(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 2) {
- pContext->ThrowParamCountMismatchException(L"Mod");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- std::unique_ptr<CFXJSE_Value> argTwo = args.GetValue(1);
- if (argOne->IsNull() || argTwo->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- bool argOneResult;
- double dDividend = ExtractDouble(pThis, argOne.get(), &argOneResult);
- bool argTwoResult;
- double dDivisor = ExtractDouble(pThis, argTwo.get(), &argTwoResult);
- if (!argOneResult || !argTwoResult) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- if (dDivisor == 0.0) {
- pContext->ThrowDivideByZeroException();
- return;
- }
-
- args.GetReturnValue()->SetDouble(dDividend -
- dDivisor * (int32_t)(dDividend / dDivisor));
-}
-
-// static
-void CXFA_FM2JSContext::Round(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 2) {
- pContext->ThrowParamCountMismatchException(L"Round");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- if (argOne->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- bool dValueRet;
- double dValue = ExtractDouble(pThis, argOne.get(), &dValueRet);
- if (!dValueRet) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- uint8_t uPrecision = 0;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> argTwo = args.GetValue(1);
- if (argTwo->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- bool dPrecisionRet;
- double dPrecision = ExtractDouble(pThis, argTwo.get(), &dPrecisionRet);
- if (!dPrecisionRet) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- uPrecision = static_cast<uint8_t>(pdfium::clamp(dPrecision, 0.0, 12.0));
- }
-
- CFX_Decimal decimalValue(static_cast<float>(dValue), uPrecision);
- args.GetReturnValue()->SetDouble(decimalValue);
-}
-
-// static
-void CXFA_FM2JSContext::Sum(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc == 0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- uint32_t uCount = 0;
- double dSum = 0.0;
- for (int32_t i = 0; i < argc; i++) {
- std::unique_ptr<CFXJSE_Value> argValue = args.GetValue(i);
- if (argValue->IsNull())
- continue;
-
- if (argValue->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- if (iLength <= 2) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValue->GetObjectPropertyByIdx(1, propertyValue.get());
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- if (propertyValue->IsNull()) {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- dSum += ValueToDouble(pThis, jsObjectValue.get());
- uCount++;
- }
- } else {
- for (int32_t j = 2; j < iLength; j++) {
- argValue->GetObjectPropertyByIdx(j, jsObjectValue.get());
- jsObjectValue->GetObjectProperty(
- propertyValue->ToString().AsStringView(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- dSum += ValueToDouble(pThis, newPropertyValue.get());
- uCount++;
- }
- }
- } else if (argValue->IsObject()) {
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(argValue.get(), newPropertyValue.get());
- if (newPropertyValue->IsNull())
- continue;
-
- dSum += ValueToDouble(pThis, argValue.get());
- uCount++;
- } else {
- dSum += ValueToDouble(pThis, argValue.get());
- uCount++;
- }
- }
- if (uCount == 0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- args.GetReturnValue()->SetDouble(dSum);
-}
-
-// static
-void CXFA_FM2JSContext::Date(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 0) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Date");
- return;
- }
-
- time_t currentTime;
- time(&currentTime);
- struct tm* pTmStruct = gmtime(&currentTime);
-
- ByteString bufferYear;
- ByteString bufferMon;
- ByteString bufferDay;
- bufferYear.Format("%d", pTmStruct->tm_year + 1900);
- bufferMon.Format("%02d", pTmStruct->tm_mon + 1);
- bufferDay.Format("%02d", pTmStruct->tm_mday);
-
- ByteString bufferCurrent = bufferYear + bufferMon + bufferDay;
- args.GetReturnValue()->SetInteger(
- DateString2Num(bufferCurrent.AsStringView()));
-}
-
-// static
-void CXFA_FM2JSContext::Date2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Date2Num");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> dateValue = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, dateValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString dateString = ValueToUTF8String(dateValue.get());
- ByteString formatString;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> formatValue = GetSimpleValue(pThis, args, 1);
- if (ValueIsNull(pThis, formatValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- formatString = ValueToUTF8String(formatValue.get());
- }
-
- ByteString localString;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> localValue = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, localValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- localString = ValueToUTF8String(localValue.get());
- }
-
- ByteString szIsoDateString =
- Local2IsoDate(pThis, dateString.AsStringView(),
- formatString.AsStringView(), localString.AsStringView());
- args.GetReturnValue()->SetInteger(
- DateString2Num(szIsoDateString.AsStringView()));
-}
-
-// static
-void CXFA_FM2JSContext::DateFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc > 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Date2Num");
- return;
- }
-
- int32_t iStyle = 0;
- if (argc > 0) {
- std::unique_ptr<CFXJSE_Value> argStyle = GetSimpleValue(pThis, args, 0);
- if (argStyle->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- iStyle = (int32_t)ValueToFloat(pThis, argStyle.get());
- if (iStyle < 0 || iStyle > 4)
- iStyle = 0;
- }
-
- ByteString szLocal;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> argLocal = GetSimpleValue(pThis, args, 1);
- if (argLocal->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- szLocal = ValueToUTF8String(argLocal.get());
- }
-
- ByteString formatStr =
- GetStandardDateFormat(pThis, iStyle, szLocal.AsStringView());
- args.GetReturnValue()->SetString(formatStr.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::IsoDate2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)
- ->ThrowParamCountMismatchException(L"IsoDate2Num");
- return;
- }
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (argOne->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- ByteString szArgString = ValueToUTF8String(argOne.get());
- args.GetReturnValue()->SetInteger(DateString2Num(szArgString.AsStringView()));
-}
-
-// static
-void CXFA_FM2JSContext::IsoTime2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 1) {
- pContext->ThrowParamCountMismatchException(L"IsoTime2Num");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- CXFA_Document* pDoc = pContext->GetDocument();
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- ByteString szArgString = ValueToUTF8String(argOne.get());
- auto pos = szArgString.Find('T', 0);
- if (!pos.has_value() || pos.value() == szArgString.GetLength() - 1) {
- args.GetReturnValue()->SetInteger(0);
- return;
- }
- szArgString = szArgString.Right(szArgString.GetLength() - (pos.value() + 1));
-
- CXFA_LocaleValue timeValue(
- XFA_VT_TIME, WideString::FromUTF8(szArgString.AsStringView()), pMgr);
- if (!timeValue.IsValid()) {
- args.GetReturnValue()->SetInteger(0);
- return;
- }
-
- CFX_DateTime uniTime = timeValue.GetTime();
- int32_t hour = uniTime.GetHour();
- int32_t min = uniTime.GetMinute();
- int32_t second = uniTime.GetSecond();
- int32_t milSecond = uniTime.GetMillisecond();
-
- // TODO(dsinclair): See if there is other time conversion code in pdfium and
- // consolidate.
- int32_t mins = hour * 60 + min;
- mins -= (pMgr->GetDefLocale()->GetTimeZone().tzHour * 60);
- while (mins > 1440)
- mins -= 1440;
- while (mins < 0)
- mins += 1440;
- hour = mins / 60;
- min = mins % 60;
-
- args.GetReturnValue()->SetInteger(hour * 3600000 + min * 60000 +
- second * 1000 + milSecond + 1);
-}
-
-// static
-void CXFA_FM2JSContext::LocalDateFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc > 2) {
- ToJSContext(pThis, nullptr)
- ->ThrowParamCountMismatchException(L"LocalDateFmt");
- return;
- }
-
- int32_t iStyle = 0;
- if (argc > 0) {
- std::unique_ptr<CFXJSE_Value> argStyle = GetSimpleValue(pThis, args, 0);
- if (argStyle->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- iStyle = (int32_t)ValueToFloat(pThis, argStyle.get());
- if (iStyle > 4 || iStyle < 0)
- iStyle = 0;
- }
-
- ByteString szLocal;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> argLocal = GetSimpleValue(pThis, args, 1);
- if (argLocal->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- szLocal = ValueToUTF8String(argLocal.get());
- }
-
- ByteString formatStr =
- GetLocalDateFormat(pThis, iStyle, szLocal.AsStringView(), false);
- args.GetReturnValue()->SetString(formatStr.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::LocalTimeFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc > 2) {
- ToJSContext(pThis, nullptr)
- ->ThrowParamCountMismatchException(L"LocalTimeFmt");
- return;
- }
-
- int32_t iStyle = 0;
- if (argc > 0) {
- std::unique_ptr<CFXJSE_Value> argStyle = GetSimpleValue(pThis, args, 0);
- if (argStyle->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- iStyle = (int32_t)ValueToFloat(pThis, argStyle.get());
- if (iStyle > 4 || iStyle < 0)
- iStyle = 0;
- }
-
- ByteString szLocal;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> argLocal = GetSimpleValue(pThis, args, 1);
- if (argLocal->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- szLocal = ValueToUTF8String(argLocal.get());
- }
-
- ByteString formatStr =
- GetLocalTimeFormat(pThis, iStyle, szLocal.AsStringView(), false);
- args.GetReturnValue()->SetString(formatStr.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Num2Date(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Num2Date");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> dateValue = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, dateValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- int32_t dDate = (int32_t)ValueToFloat(pThis, dateValue.get());
- if (dDate < 1) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString formatString;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> formatValue = GetSimpleValue(pThis, args, 1);
- if (ValueIsNull(pThis, formatValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- formatString = ValueToUTF8String(formatValue.get());
- }
-
- ByteString localString;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> localValue = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, localValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- localString = ValueToUTF8String(localValue.get());
- }
-
- int32_t iYear = 1900;
- int32_t iMonth = 1;
- int32_t iDay = 1;
- int32_t i = 0;
- while (dDate > 0) {
- if (iMonth == 2) {
- if ((!((iYear + i) % 4) && ((iYear + i) % 100)) || !((iYear + i) % 400)) {
- if (dDate > 29) {
- ++iMonth;
- if (iMonth > 12) {
- iMonth = 1;
- ++i;
- }
- iDay = 1;
- dDate -= 29;
- } else {
- iDay += static_cast<int32_t>(dDate) - 1;
- dDate = 0;
- }
- } else {
- if (dDate > 28) {
- ++iMonth;
- if (iMonth > 12) {
- iMonth = 1;
- ++i;
- }
- iDay = 1;
- dDate -= 28;
- } else {
- iDay += static_cast<int32_t>(dDate) - 1;
- dDate = 0;
- }
- }
- } else if (iMonth < 8) {
- if ((iMonth % 2 == 0)) {
- if (dDate > 30) {
- ++iMonth;
- if (iMonth > 12) {
- iMonth = 1;
- ++i;
- }
- iDay = 1;
- dDate -= 30;
- } else {
- iDay += static_cast<int32_t>(dDate) - 1;
- dDate = 0;
- }
- } else {
- if (dDate > 31) {
- ++iMonth;
- if (iMonth > 12) {
- iMonth = 1;
- ++i;
- }
- iDay = 1;
- dDate -= 31;
- } else {
- iDay += static_cast<int32_t>(dDate) - 1;
- dDate = 0;
- }
- }
- } else {
- if (iMonth % 2 != 0) {
- if (dDate > 30) {
- ++iMonth;
- if (iMonth > 12) {
- iMonth = 1;
- ++i;
- }
- iDay = 1;
- dDate -= 30;
- } else {
- iDay += static_cast<int32_t>(dDate) - 1;
- dDate = 0;
- }
- } else {
- if (dDate > 31) {
- ++iMonth;
- if (iMonth > 12) {
- iMonth = 1;
- ++i;
- }
- iDay = 1;
- dDate -= 31;
- } else {
- iDay += static_cast<int32_t>(dDate) - 1;
- dDate = 0;
- }
- }
- }
- }
-
- ByteString szIsoDateString;
- szIsoDateString.Format("%d%02d%02d", iYear + i, iMonth, iDay);
- ByteString szLocalDateString =
- IsoDate2Local(pThis, szIsoDateString.AsStringView(),
- formatString.AsStringView(), localString.AsStringView());
- args.GetReturnValue()->SetString(szLocalDateString.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Num2GMTime(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 3) {
- ToJSContext(pThis, nullptr)
- ->ThrowParamCountMismatchException(L"Num2GMTime");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> timeValue = GetSimpleValue(pThis, args, 0);
- if (timeValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- int32_t iTime = (int32_t)ValueToFloat(pThis, timeValue.get());
- if (abs(iTime) < 1.0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString formatString;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> formatValue = GetSimpleValue(pThis, args, 1);
- if (formatValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- formatString = ValueToUTF8String(formatValue.get());
- }
-
- ByteString localString;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> localValue = GetSimpleValue(pThis, args, 2);
- if (localValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- localString = ValueToUTF8String(localValue.get());
- }
-
- ByteString szGMTTimeString =
- Num2AllTime(pThis, iTime, formatString.AsStringView(),
- localString.AsStringView(), true);
- args.GetReturnValue()->SetString(szGMTTimeString.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Num2Time(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Num2Time");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> timeValue = GetSimpleValue(pThis, args, 0);
- if (timeValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- float fTime = ValueToFloat(pThis, timeValue.get());
- if (fabs(fTime) < 1.0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString formatString;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> formatValue = GetSimpleValue(pThis, args, 1);
- if (formatValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- formatString = ValueToUTF8String(formatValue.get());
- }
-
- ByteString localString;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> localValue = GetSimpleValue(pThis, args, 2);
- if (localValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- localString = ValueToUTF8String(localValue.get());
- }
-
- ByteString szLocalTimeString = Num2AllTime(pThis, static_cast<int32_t>(fTime),
- formatString.AsStringView(),
- localString.AsStringView(), false);
- args.GetReturnValue()->SetString(szLocalTimeString.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Time(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 0) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Time");
- return;
- }
-
- time_t now;
- time(&now);
-
- struct tm* pGmt = gmtime(&now);
- args.GetReturnValue()->SetInteger(
- (pGmt->tm_hour * 3600 + pGmt->tm_min * 60 + pGmt->tm_sec) * 1000);
-}
-
-// static
-void CXFA_FM2JSContext::Time2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Time2Num");
- return;
- }
-
- ByteString timeString;
- std::unique_ptr<CFXJSE_Value> timeValue = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, timeValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- timeString = ValueToUTF8String(timeValue.get());
-
- ByteString formatString;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> formatValue = GetSimpleValue(pThis, args, 1);
- if (ValueIsNull(pThis, formatValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- formatString = ValueToUTF8String(formatValue.get());
- }
-
- ByteString localString;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> localValue = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, localValue.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- localString = ValueToUTF8String(localValue.get());
- }
-
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- IFX_Locale* pLocale = nullptr;
- if (localString.IsEmpty()) {
- CXFA_Node* pThisNode = ToNode(pDoc->GetScriptContext()->GetThisObject());
- ASSERT(pThisNode);
- CXFA_WidgetData widgetData(pThisNode);
- pLocale = widgetData.GetLocal();
- } else {
- pLocale =
- pMgr->GetLocaleByName(WideString::FromUTF8(localString.AsStringView()));
- }
-
- WideString wsFormat;
- if (formatString.IsEmpty())
- wsFormat = pLocale->GetTimePattern(FX_LOCALEDATETIMESUBCATEGORY_Default);
- else
- wsFormat = WideString::FromUTF8(formatString.AsStringView());
-
- wsFormat = L"time{" + wsFormat + L"}";
- CXFA_LocaleValue localeValue(XFA_VT_TIME,
- WideString::FromUTF8(timeString.AsStringView()),
- wsFormat, pLocale, pMgr);
- if (!localeValue.IsValid()) {
- args.GetReturnValue()->SetInteger(0);
- return;
- }
-
- CFX_DateTime uniTime = localeValue.GetTime();
- int32_t hour = uniTime.GetHour();
- int32_t min = uniTime.GetMinute();
- int32_t second = uniTime.GetSecond();
- int32_t milSecond = uniTime.GetMillisecond();
- int32_t mins = hour * 60 + min;
-
- mins -= (CXFA_TimeZoneProvider().GetTimeZone().tzHour * 60);
- while (mins > 1440)
- mins -= 1440;
-
- while (mins < 0)
- mins += 1440;
-
- hour = mins / 60;
- min = mins % 60;
- args.GetReturnValue()->SetInteger(hour * 3600000 + min * 60000 +
- second * 1000 + milSecond + 1);
-}
-
-// static
-void CXFA_FM2JSContext::TimeFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc > 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"TimeFmt");
- return;
- }
-
- int32_t iStyle = 0;
- if (argc > 0) {
- std::unique_ptr<CFXJSE_Value> argStyle = GetSimpleValue(pThis, args, 0);
- if (argStyle->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- iStyle = (int32_t)ValueToFloat(pThis, argStyle.get());
- if (iStyle > 4 || iStyle < 0)
- iStyle = 0;
- }
-
- ByteString szLocal;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> argLocal = GetSimpleValue(pThis, args, 1);
- if (argLocal->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- szLocal = ValueToUTF8String(argLocal.get());
- }
-
- ByteString formatStr =
- GetStandardTimeFormat(pThis, iStyle, szLocal.AsStringView());
- args.GetReturnValue()->SetString(formatStr.AsStringView());
-}
-
-// static
-bool CXFA_FM2JSContext::IsIsoDateFormat(const char* pData,
- int32_t iLength,
- int32_t& iStyle,
- int32_t& iYear,
- int32_t& iMonth,
- int32_t& iDay) {
- iYear = 0;
- iMonth = 1;
- iDay = 1;
-
- if (iLength < 4)
- return false;
-
- char strYear[5];
- strYear[4] = '\0';
- for (int32_t i = 0; i < 4; ++i) {
- if (!std::isdigit(pData[i]))
- return false;
-
- strYear[i] = pData[i];
- }
- iYear = FXSYS_atoi(strYear);
- iStyle = 0;
- if (iLength == 4)
- return true;
-
- iStyle = pData[4] == '-' ? 1 : 0;
-
- char strTemp[3];
- strTemp[2] = '\0';
- int32_t iPosOff = iStyle == 0 ? 4 : 5;
- if (!std::isdigit(pData[iPosOff]) || !std::isdigit(pData[iPosOff + 1]))
- return false;
-
- strTemp[0] = pData[iPosOff];
- strTemp[1] = pData[iPosOff + 1];
- iMonth = FXSYS_atoi(strTemp);
- if (iMonth > 12 || iMonth < 1)
- return false;
-
- if (iStyle == 0) {
- iPosOff += 2;
- if (iLength == 6)
- return true;
- } else {
- iPosOff += 3;
- if (iLength == 7)
- return true;
- }
- if (!std::isdigit(pData[iPosOff]) || !std::isdigit(pData[iPosOff + 1]))
- return false;
-
- strTemp[0] = pData[iPosOff];
- strTemp[1] = pData[iPosOff + 1];
- iDay = FXSYS_atoi(strTemp);
- if (iPosOff + 2 < iLength)
- return false;
-
- if ((!(iYear % 4) && (iYear % 100)) || !(iYear % 400)) {
- if (iMonth == 2 && iDay > 29)
- return false;
- } else {
- if (iMonth == 2 && iDay > 28)
- return false;
- }
- if (iMonth != 2) {
- if (iMonth < 8) {
- if (iDay > (iMonth % 2 == 0 ? 30 : 31))
- return false;
- } else if (iDay > (iMonth % 2 == 0 ? 31 : 30)) {
- return false;
- }
- }
- return true;
-}
-
-// static
-bool CXFA_FM2JSContext::IsIsoTimeFormat(const char* pData,
- int32_t iLength,
- int32_t& iHour,
- int32_t& iMinute,
- int32_t& iSecond,
- int32_t& iMilliSecond,
- int32_t& iZoneHour,
- int32_t& iZoneMinute) {
- iHour = 0;
- iMinute = 0;
- iSecond = 0;
- iMilliSecond = 0;
- iZoneHour = 0;
- iZoneMinute = 0;
- if (!pData)
- return false;
-
- char strTemp[3];
- strTemp[2] = '\0';
- int32_t iZone = 0;
- int32_t i = 0;
- while (i < iLength) {
- if (!std::isdigit(pData[i]) && pData[i] != ':') {
- iZone = i;
- break;
- }
- ++i;
- }
- if (i == iLength)
- iZone = iLength;
-
- int32_t iPos = 0;
- int32_t iIndex = 0;
- while (iIndex < iZone) {
- if (!std::isdigit(pData[iIndex]))
- return false;
-
- strTemp[0] = pData[iIndex];
- if (!std::isdigit(pData[iIndex + 1]))
- return false;
-
- strTemp[1] = pData[iIndex + 1];
- if (FXSYS_atoi(strTemp) > 60)
- return false;
-
- if (pData[2] == ':') {
- if (iPos == 0) {
- iHour = FXSYS_atoi(strTemp);
- ++iPos;
- } else if (iPos == 1) {
- iMinute = FXSYS_atoi(strTemp);
- ++iPos;
- } else {
- iSecond = FXSYS_atoi(strTemp);
- }
- iIndex += 3;
- } else {
- if (iPos == 0) {
- iHour = FXSYS_atoi(strTemp);
- ++iPos;
- } else if (iPos == 1) {
- iMinute = FXSYS_atoi(strTemp);
- ++iPos;
- } else if (iPos == 2) {
- iSecond = FXSYS_atoi(strTemp);
- ++iPos;
- }
- iIndex += 2;
- }
- }
-
- if (iIndex < iLength && pData[iIndex] == '.') {
- constexpr int kSubSecondLength = 3;
- if (iIndex + kSubSecondLength >= iLength)
- return false;
-
- ++iIndex;
- char strSec[kSubSecondLength + 1];
- for (int i = 0; i < kSubSecondLength; ++i) {
- char c = pData[iIndex + i];
- if (!std::isdigit(c))
- return false;
- strSec[i] = c;
- }
- strSec[kSubSecondLength] = '\0';
-
- iMilliSecond = FXSYS_atoi(strSec);
- if (iMilliSecond > 100) {
- iMilliSecond = 0;
- return false;
- }
- iIndex += kSubSecondLength;
- }
-
- if (iIndex < iLength && FXSYS_tolower(pData[iIndex]) == 'z')
- return true;
-
- int32_t iSign = 1;
- if (iIndex < iLength) {
- if (pData[iIndex] == '+') {
- ++iIndex;
- } else if (pData[iIndex] == '-') {
- iSign = -1;
- ++iIndex;
- }
- }
- iPos = 0;
- while (iIndex < iLength) {
- if (!std::isdigit(pData[iIndex]))
- return false;
-
- strTemp[0] = pData[iIndex];
- if (!std::isdigit(pData[iIndex + 1]))
- return false;
-
- strTemp[1] = pData[iIndex + 1];
- if (FXSYS_atoi(strTemp) > 60)
- return false;
-
- if (pData[2] == ':') {
- if (iPos == 0) {
- iZoneHour = FXSYS_atoi(strTemp);
- } else if (iPos == 1) {
- iZoneMinute = FXSYS_atoi(strTemp);
- }
- iIndex += 3;
- } else {
- if (!iPos) {
- iZoneHour = FXSYS_atoi(strTemp);
- ++iPos;
- } else if (iPos == 1) {
- iZoneMinute = FXSYS_atoi(strTemp);
- ++iPos;
- }
- iIndex += 2;
- }
- }
- if (iIndex < iLength)
- return false;
-
- iZoneHour *= iSign;
- return true;
-}
-
-// static
-bool CXFA_FM2JSContext::IsIsoDateTimeFormat(const char* pData,
- int32_t iLength,
- int32_t& iYear,
- int32_t& iMonth,
- int32_t& iDay,
- int32_t& iHour,
- int32_t& iMinute,
- int32_t& iSecond,
- int32_t& iMillionSecond,
- int32_t& iZoneHour,
- int32_t& iZoneMinute) {
- iYear = 0;
- iMonth = 0;
- iDay = 0;
- iHour = 0;
- iMinute = 0;
- iSecond = 0;
- if (!pData)
- return false;
-
- int32_t iIndex = 0;
- while (pData[iIndex] != 'T' && pData[iIndex] != 't') {
- if (iIndex >= iLength)
- return false;
- ++iIndex;
- }
- if (iIndex != 8 && iIndex != 10)
- return false;
-
- int32_t iStyle = -1;
- if (!IsIsoDateFormat(pData, iIndex, iStyle, iYear, iMonth, iDay))
- return false;
- if (pData[iIndex] != 'T' && pData[iIndex] != 't')
- return true;
-
- ++iIndex;
- if (((iLength - iIndex > 13) && (iLength - iIndex < 6)) &&
- (iLength - iIndex != 15)) {
- return true;
- }
- return IsIsoTimeFormat(pData + iIndex, iLength - iIndex, iHour, iMinute,
- iSecond, iMillionSecond, iZoneHour, iZoneMinute);
-}
-
-// static
-ByteString CXFA_FM2JSContext::Local2IsoDate(CFXJSE_Value* pThis,
- const ByteStringView& szDate,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale) {
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- if (!pDoc)
- return ByteString();
-
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- IFX_Locale* pLocale = LocaleFromString(pDoc, pMgr, szLocale);
- if (!pLocale)
- return ByteString();
-
- WideString wsFormat = FormatFromString(pLocale, szFormat);
- CFX_DateTime dt = CXFA_LocaleValue(XFA_VT_DATE, WideString::FromUTF8(szDate),
- wsFormat, pLocale, pMgr)
- .GetDate();
-
- ByteString strIsoDate;
- strIsoDate.Format("%4d-%02d-%02d", dt.GetYear(), dt.GetMonth(), dt.GetDay());
- return strIsoDate;
-}
-
-// static
-ByteString CXFA_FM2JSContext::IsoDate2Local(CFXJSE_Value* pThis,
- const ByteStringView& szDate,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale) {
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- if (!pDoc)
- return ByteString();
-
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- IFX_Locale* pLocale = LocaleFromString(pDoc, pMgr, szLocale);
- if (!pLocale)
- return ByteString();
-
- WideString wsFormat = FormatFromString(pLocale, szFormat);
- WideString wsRet;
- CXFA_LocaleValue(XFA_VT_DATE, WideString::FromUTF8(szDate), pMgr)
- .FormatPatterns(wsRet, wsFormat, pLocale, XFA_VALUEPICTURE_Display);
- return wsRet.UTF8Encode();
-}
-
-// static
-ByteString CXFA_FM2JSContext::IsoTime2Local(CFXJSE_Value* pThis,
- const ByteStringView& szTime,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale) {
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- if (!pDoc)
- return ByteString();
-
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- IFX_Locale* pLocale = LocaleFromString(pDoc, pMgr, szLocale);
- if (!pLocale)
- return ByteString();
-
- WideString wsFormat = {
- L"time{", FormatFromString(pLocale, szFormat).AsStringView(), L"}"};
- CXFA_LocaleValue widgetValue(XFA_VT_TIME, WideString::FromUTF8(szTime), pMgr);
- WideString wsRet;
- widgetValue.FormatPatterns(wsRet, wsFormat, pLocale,
- XFA_VALUEPICTURE_Display);
- return wsRet.UTF8Encode();
-}
-
-// static
-int32_t CXFA_FM2JSContext::DateString2Num(const ByteStringView& szDateString) {
- int32_t iLength = szDateString.GetLength();
- int32_t iYear = 0;
- int32_t iMonth = 0;
- int32_t iDay = 0;
- if (iLength <= 10) {
- int32_t iStyle = -1;
- if (!IsIsoDateFormat(szDateString.unterminated_c_str(), iLength, iStyle,
- iYear, iMonth, iDay)) {
- return 0;
- }
- } else {
- int32_t iHour = 0;
- int32_t iMinute = 0;
- int32_t iSecond = 0;
- int32_t iMilliSecond = 0;
- int32_t iZoneHour = 0;
- int32_t iZoneMinute = 0;
- if (!IsIsoDateTimeFormat(szDateString.unterminated_c_str(), iLength, iYear,
- iMonth, iDay, iHour, iMinute, iSecond,
- iMilliSecond, iZoneHour, iZoneMinute)) {
- return 0;
- }
- }
-
- float dDays = 0;
- int32_t i = 1;
- if (iYear < 1900)
- return 0;
-
- while (iYear - i >= 1900) {
- dDays +=
- ((!((iYear - i) % 4) && ((iYear - i) % 100)) || !((iYear - i) % 400))
- ? 366
- : 365;
- ++i;
- }
- i = 1;
- while (i < iMonth) {
- if (i == 2)
- dDays += ((!(iYear % 4) && (iYear % 100)) || !(iYear % 400)) ? 29 : 28;
- else if (i <= 7)
- dDays += (i % 2 == 0) ? 30 : 31;
- else
- dDays += (i % 2 == 0) ? 31 : 30;
-
- ++i;
- }
- i = 0;
- while (iDay - i > 0) {
- dDays += 1;
- ++i;
- }
- return (int32_t)dDays;
-}
-
-// static
-ByteString CXFA_FM2JSContext::GetLocalDateFormat(CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocale,
- bool bStandard) {
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- if (!pDoc)
- return ByteString();
-
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- IFX_Locale* pLocale = LocaleFromString(pDoc, pMgr, szLocale);
- if (!pLocale)
- return ByteString();
-
- WideString strRet = pLocale->GetDatePattern(SubCategoryFromInt(iStyle));
- if (!bStandard) {
- AlternateDateTimeSymbols(strRet, pLocale->GetDateTimeSymbols(),
- g_sAltTable_Date);
- }
- return strRet.UTF8Encode();
-}
-
-// static
-ByteString CXFA_FM2JSContext::GetLocalTimeFormat(CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocale,
- bool bStandard) {
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- if (!pDoc)
- return ByteString();
-
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- IFX_Locale* pLocale = LocaleFromString(pDoc, pMgr, szLocale);
- if (!pLocale)
- return ByteString();
-
- WideString strRet = pLocale->GetTimePattern(SubCategoryFromInt(iStyle));
- if (!bStandard) {
- AlternateDateTimeSymbols(strRet, pLocale->GetDateTimeSymbols(),
- g_sAltTable_Time);
- }
- return strRet.UTF8Encode();
-}
-
-// static
-ByteString CXFA_FM2JSContext::GetStandardDateFormat(
- CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocalStr) {
- return GetLocalDateFormat(pThis, iStyle, szLocalStr, true);
-}
-
-// static
-ByteString CXFA_FM2JSContext::GetStandardTimeFormat(
- CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocalStr) {
- return GetLocalTimeFormat(pThis, iStyle, szLocalStr, true);
-}
-
-// static
-ByteString CXFA_FM2JSContext::Num2AllTime(CFXJSE_Value* pThis,
- int32_t iTime,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale,
- bool bGM) {
- int32_t iHour = 0;
- int32_t iMin = 0;
- int32_t iSec = 0;
- iHour = static_cast<int>(iTime) / 3600000;
- iMin = (static_cast<int>(iTime) - iHour * 3600000) / 60000;
- iSec = (static_cast<int>(iTime) - iHour * 3600000 - iMin * 60000) / 1000;
-
- if (!bGM) {
- int32_t iZoneHour = 0;
- int32_t iZoneMin = 0;
- int32_t iZoneSec = 0;
- GetLocalTimeZone(iZoneHour, iZoneMin, iZoneSec);
- iHour += iZoneHour;
- iMin += iZoneMin;
- iSec += iZoneSec;
- }
-
- ByteString strIsoTime;
- strIsoTime.Format("%02d:%02d:%02d", iHour, iMin, iSec);
- return IsoTime2Local(pThis, strIsoTime.AsStringView(), szFormat, szLocale);
-}
-
-// static
-void CXFA_FM2JSContext::GetLocalTimeZone(int32_t& iHour,
- int32_t& iMin,
- int32_t& iSec) {
- time_t now;
- time(&now);
-
- struct tm* pGmt = gmtime(&now);
- struct tm* pLocal = localtime(&now);
- iHour = pLocal->tm_hour - pGmt->tm_hour;
- iMin = pLocal->tm_min - pGmt->tm_min;
- iSec = pLocal->tm_sec - pGmt->tm_sec;
-}
-
-// static
-void CXFA_FM2JSContext::Apr(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 3) {
- pContext->ThrowParamCountMismatchException(L"Apr");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double nPrincipal = ValueToDouble(pThis, argOne.get());
- double nPayment = ValueToDouble(pThis, argTwo.get());
- double nPeriods = ValueToDouble(pThis, argThree.get());
- if (nPrincipal <= 0 || nPayment <= 0 || nPeriods <= 0) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- double r = 2 * (nPeriods * nPayment - nPrincipal) / (nPeriods * nPrincipal);
- double nTemp = 1;
- for (int32_t i = 0; i < nPeriods; ++i)
- nTemp *= (1 + r);
-
- double nRet = r * nTemp / (nTemp - 1) - nPayment / nPrincipal;
- while (fabs(nRet) > kFinancialPrecision) {
- double nDerivative =
- ((nTemp + r * nPeriods * (nTemp / (1 + r))) * (nTemp - 1) -
- (r * nTemp * nPeriods * (nTemp / (1 + r)))) /
- ((nTemp - 1) * (nTemp - 1));
- if (nDerivative == 0) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- r = r - nRet / nDerivative;
- nTemp = 1;
- for (int32_t i = 0; i < nPeriods; ++i) {
- nTemp *= (1 + r);
- }
- nRet = r * nTemp / (nTemp - 1) - nPayment / nPrincipal;
- }
- args.GetReturnValue()->SetDouble(r * 12);
-}
-
-// static
-void CXFA_FM2JSContext::CTerm(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 3) {
- pContext->ThrowParamCountMismatchException(L"CTerm");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float nRate = ValueToFloat(pThis, argOne.get());
- float nFutureValue = ValueToFloat(pThis, argTwo.get());
- float nInitAmount = ValueToFloat(pThis, argThree.get());
- if ((nRate <= 0) || (nFutureValue <= 0) || (nInitAmount <= 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- args.GetReturnValue()->SetFloat(log((float)(nFutureValue / nInitAmount)) /
- log((float)(1 + nRate)));
-}
-
-// static
-void CXFA_FM2JSContext::FV(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 3) {
- pContext->ThrowParamCountMismatchException(L"FV");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double nAmount = ValueToDouble(pThis, argOne.get());
- double nRate = ValueToDouble(pThis, argTwo.get());
- double nPeriod = ValueToDouble(pThis, argThree.get());
- if ((nRate < 0) || (nPeriod <= 0) || (nAmount <= 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- double dResult = 0;
- if (nRate) {
- double nTemp = 1;
- for (int i = 0; i < nPeriod; ++i) {
- nTemp *= 1 + nRate;
- }
- dResult = nAmount * (nTemp - 1) / nRate;
- } else {
- dResult = nAmount * nPeriod;
- }
-
- args.GetReturnValue()->SetDouble(dResult);
-}
-
-// static
-void CXFA_FM2JSContext::IPmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 5) {
- pContext->ThrowParamCountMismatchException(L"IPmt");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- std::unique_ptr<CFXJSE_Value> argFour = GetSimpleValue(pThis, args, 3);
- std::unique_ptr<CFXJSE_Value> argFive = GetSimpleValue(pThis, args, 4);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get()) || ValueIsNull(pThis, argFour.get()) ||
- ValueIsNull(pThis, argFive.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float nPrincipalAmount = ValueToFloat(pThis, argOne.get());
- float nRate = ValueToFloat(pThis, argTwo.get());
- float nPayment = ValueToFloat(pThis, argThree.get());
- float nFirstMonth = ValueToFloat(pThis, argFour.get());
- float nNumberOfMonths = ValueToFloat(pThis, argFive.get());
- if ((nPrincipalAmount <= 0) || (nRate <= 0) || (nPayment <= 0) ||
- (nFirstMonth < 0) || (nNumberOfMonths < 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- float nRateOfMonth = nRate / 12;
- int32_t iNums =
- (int32_t)((log10((float)(nPayment / nPrincipalAmount)) -
- log10((float)(nPayment / nPrincipalAmount - nRateOfMonth))) /
- log10((float)(1 + nRateOfMonth)));
- int32_t iEnd = std::min((int32_t)(nFirstMonth + nNumberOfMonths - 1), iNums);
-
- if (nPayment < nPrincipalAmount * nRateOfMonth) {
- args.GetReturnValue()->SetFloat(0);
- return;
- }
-
- int32_t i = 0;
- for (i = 0; i < nFirstMonth - 1; ++i)
- nPrincipalAmount -= nPayment - nPrincipalAmount * nRateOfMonth;
-
- float nSum = 0;
- for (; i < iEnd; ++i) {
- nSum += nPrincipalAmount * nRateOfMonth;
- nPrincipalAmount -= nPayment - nPrincipalAmount * nRateOfMonth;
- }
- args.GetReturnValue()->SetFloat(nSum);
-}
-
-// static
-void CXFA_FM2JSContext::NPV(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- int32_t argc = args.GetLength();
- if (argc < 3) {
- pContext->ThrowParamCountMismatchException(L"NPV");
- return;
- }
-
- std::vector<std::unique_ptr<CFXJSE_Value>> argValues;
- for (int32_t i = 0; i < argc; i++) {
- argValues.push_back(GetSimpleValue(pThis, args, i));
- if (ValueIsNull(pThis, argValues[i].get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
- }
-
- double nRate = ValueToDouble(pThis, argValues[0].get());
- if (nRate <= 0) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- std::vector<double> data(argc - 1);
- for (int32_t i = 1; i < argc; i++)
- data.push_back(ValueToDouble(pThis, argValues[i].get()));
-
- double nSum = 0;
- int32_t iIndex = 0;
- for (int32_t i = 0; i < argc - 1; i++) {
- double nTemp = 1;
- for (int32_t j = 0; j <= i; j++)
- nTemp *= 1 + nRate;
-
- double nNum = data[iIndex++];
- nSum += nNum / nTemp;
- }
- args.GetReturnValue()->SetDouble(nSum);
-}
-
-// static
-void CXFA_FM2JSContext::Pmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 3) {
- pContext->ThrowParamCountMismatchException(L"Pmt");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float nPrincipal = ValueToFloat(pThis, argOne.get());
- float nRate = ValueToFloat(pThis, argTwo.get());
- float nPeriods = ValueToFloat(pThis, argThree.get());
- if ((nPrincipal <= 0) || (nRate <= 0) || (nPeriods <= 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- float nTmp = 1 + nRate;
- float nSum = nTmp;
- for (int32_t i = 0; i < nPeriods - 1; ++i)
- nSum *= nTmp;
-
- args.GetReturnValue()->SetFloat((nPrincipal * nRate * nSum) / (nSum - 1));
-}
-
-// static
-void CXFA_FM2JSContext::PPmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 5) {
- pContext->ThrowParamCountMismatchException(L"PPmt");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- std::unique_ptr<CFXJSE_Value> argFour = GetSimpleValue(pThis, args, 3);
- std::unique_ptr<CFXJSE_Value> argFive = GetSimpleValue(pThis, args, 4);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get()) || ValueIsNull(pThis, argFour.get()) ||
- ValueIsNull(pThis, argFive.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float nPrincipalAmount = ValueToFloat(pThis, argOne.get());
- float nRate = ValueToFloat(pThis, argTwo.get());
- float nPayment = ValueToFloat(pThis, argThree.get());
- float nFirstMonth = ValueToFloat(pThis, argFour.get());
- float nNumberOfMonths = ValueToFloat(pThis, argFive.get());
- if ((nPrincipalAmount <= 0) || (nRate <= 0) || (nPayment <= 0) ||
- (nFirstMonth < 0) || (nNumberOfMonths < 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- float nRateOfMonth = nRate / 12;
- int32_t iNums =
- (int32_t)((log10((float)(nPayment / nPrincipalAmount)) -
- log10((float)(nPayment / nPrincipalAmount - nRateOfMonth))) /
- log10((float)(1 + nRateOfMonth)));
- int32_t iEnd = std::min((int32_t)(nFirstMonth + nNumberOfMonths - 1), iNums);
- if (nPayment < nPrincipalAmount * nRateOfMonth) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- int32_t i = 0;
- for (i = 0; i < nFirstMonth - 1; ++i)
- nPrincipalAmount -= nPayment - nPrincipalAmount * nRateOfMonth;
-
- float nTemp = 0;
- float nSum = 0;
- for (; i < iEnd; ++i) {
- nTemp = nPayment - nPrincipalAmount * nRateOfMonth;
- nSum += nTemp;
- nPrincipalAmount -= nTemp;
- }
- args.GetReturnValue()->SetFloat(nSum);
-}
-
-// static
-void CXFA_FM2JSContext::PV(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 3) {
- pContext->ThrowParamCountMismatchException(L"PV");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double nAmount = ValueToDouble(pThis, argOne.get());
- double nRate = ValueToDouble(pThis, argTwo.get());
- double nPeriod = ValueToDouble(pThis, argThree.get());
- if ((nAmount <= 0) || (nRate < 0) || (nPeriod <= 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- double nTemp = 1;
- for (int32_t i = 0; i < nPeriod; ++i)
- nTemp *= 1 + nRate;
-
- nTemp = 1 / nTemp;
- args.GetReturnValue()->SetDouble(nAmount * ((1 - nTemp) / nRate));
-}
-
-// static
-void CXFA_FM2JSContext::Rate(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 3) {
- pContext->ThrowParamCountMismatchException(L"Rate");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float nFuture = ValueToFloat(pThis, argOne.get());
- float nPresent = ValueToFloat(pThis, argTwo.get());
- float nTotalNumber = ValueToFloat(pThis, argThree.get());
- if ((nFuture <= 0) || (nPresent < 0) || (nTotalNumber <= 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- args.GetReturnValue()->SetFloat(
- FXSYS_pow((float)(nFuture / nPresent), (float)(1 / nTotalNumber)) - 1);
-}
-
-// static
-void CXFA_FM2JSContext::Term(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 3) {
- pContext->ThrowParamCountMismatchException(L"Term");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get()) ||
- ValueIsNull(pThis, argThree.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float nMount = ValueToFloat(pThis, argOne.get());
- float nRate = ValueToFloat(pThis, argTwo.get());
- float nFuture = ValueToFloat(pThis, argThree.get());
- if ((nMount <= 0) || (nRate <= 0) || (nFuture <= 0)) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- args.GetReturnValue()->SetFloat(log((float)(nFuture / nMount * nRate) + 1) /
- log((float)(1 + nRate)));
-}
-
-// static
-void CXFA_FM2JSContext::Choose(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- int32_t argc = args.GetLength();
- if (argc < 2) {
- pContext->ThrowParamCountMismatchException(L"Choose");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- int32_t iIndex = (int32_t)ValueToFloat(pThis, argOne.get());
- if (iIndex < 1) {
- args.GetReturnValue()->SetString("");
- return;
- }
-
- bool bFound = false;
- bool bStopCounterFlags = false;
- int32_t iArgIndex = 1;
- int32_t iValueIndex = 0;
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- while (!bFound && !bStopCounterFlags && (iArgIndex < argc)) {
- std::unique_ptr<CFXJSE_Value> argIndexValue = args.GetValue(iArgIndex);
- if (argIndexValue->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argIndexValue->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- if (iLength > 3)
- bStopCounterFlags = true;
-
- iValueIndex += (iLength - 2);
- if (iValueIndex >= iIndex) {
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argIndexValue->GetObjectPropertyByIdx(1, propertyValue.get());
- argIndexValue->GetObjectPropertyByIdx(
- (iLength - 1) - (iValueIndex - iIndex), jsObjectValue.get());
- if (propertyValue->IsNull()) {
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- } else {
- jsObjectValue->GetObjectProperty(
- propertyValue->ToString().AsStringView(), newPropertyValue.get());
- }
- ByteString bsChosen = ValueToUTF8String(newPropertyValue.get());
- args.GetReturnValue()->SetString(bsChosen.AsStringView());
- bFound = true;
- }
- } else {
- iValueIndex++;
- if (iValueIndex == iIndex) {
- ByteString bsChosen = ValueToUTF8String(argIndexValue.get());
- args.GetReturnValue()->SetString(bsChosen.AsStringView());
- bFound = true;
- }
- }
- iArgIndex++;
- }
- if (!bFound)
- args.GetReturnValue()->SetString("");
-}
-
-// static
-void CXFA_FM2JSContext::Exists(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Exists");
- return;
- }
- args.GetReturnValue()->SetInteger(args.GetValue(0)->IsObject());
-}
-
-// static
-void CXFA_FM2JSContext::HasValue(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"HasValue");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (!argOne->IsString()) {
- args.GetReturnValue()->SetInteger(argOne->IsNumber() ||
- argOne->IsBoolean());
- return;
- }
-
- ByteString valueStr = argOne->ToString();
- valueStr.TrimLeft();
- args.GetReturnValue()->SetInteger(!valueStr.IsEmpty());
-}
-
-// static
-void CXFA_FM2JSContext::Oneof(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() < 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Oneof");
- return;
- }
-
- bool bFlags = false;
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::vector<std::unique_ptr<CFXJSE_Value>> parameterValues;
- unfoldArgs(pThis, args, &parameterValues, 1);
- for (const auto& value : parameterValues) {
- if (simpleValueCompare(pThis, argOne.get(), value.get())) {
- bFlags = true;
- break;
- }
- }
-
- args.GetReturnValue()->SetInteger(bFlags);
-}
-
-// static
-void CXFA_FM2JSContext::Within(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Within");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (argOne->IsNull()) {
- args.GetReturnValue()->SetUndefined();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argLow = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> argHigh = GetSimpleValue(pThis, args, 2);
- if (argOne->IsNumber()) {
- float oneNumber = ValueToFloat(pThis, argOne.get());
- float lowNumber = ValueToFloat(pThis, argLow.get());
- float heightNumber = ValueToFloat(pThis, argHigh.get());
- args.GetReturnValue()->SetInteger((oneNumber >= lowNumber) &&
- (oneNumber <= heightNumber));
- return;
- }
-
- ByteString oneString = ValueToUTF8String(argOne.get());
- ByteString lowString = ValueToUTF8String(argLow.get());
- ByteString heightString = ValueToUTF8String(argHigh.get());
- args.GetReturnValue()->SetInteger(
- (oneString.Compare(lowString.AsStringView()) >= 0) &&
- (oneString.Compare(heightString.AsStringView()) <= 0));
-}
-
-// static
-void CXFA_FM2JSContext::If(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"If");
- return;
- }
-
- args.GetReturnValue()->Assign(GetSimpleValue(pThis, args, 0)->ToBoolean()
- ? GetSimpleValue(pThis, args, 1).get()
- : GetSimpleValue(pThis, args, 2).get());
-}
-
-// static
-void CXFA_FM2JSContext::Eval(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 1) {
- pContext->ThrowParamCountMismatchException(L"Eval");
- return;
- }
-
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- std::unique_ptr<CFXJSE_Value> scriptValue = GetSimpleValue(pThis, args, 0);
- ByteString utf8ScriptString = ValueToUTF8String(scriptValue.get());
- if (utf8ScriptString.IsEmpty()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- CFX_WideTextBuf wsJavaScriptBuf;
- if (!CXFA_FM2JSContext::Translate(
- WideString::FromUTF8(utf8ScriptString.AsStringView()).AsStringView(),
- &wsJavaScriptBuf)) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Context> pNewContext(
- CFXJSE_Context::Create(pIsolate, nullptr, nullptr));
-
- auto returnValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- pNewContext->ExecuteScript(
- FX_UTF8Encode(wsJavaScriptBuf.AsStringView()).c_str(), returnValue.get());
-
- args.GetReturnValue()->Assign(returnValue.get());
-}
-
-// static
-void CXFA_FM2JSContext::Ref(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- if (args.GetLength() != 1) {
- pContext->ThrowParamCountMismatchException(L"Ref");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- if (!argOne->IsArray() && !argOne->IsObject() && !argOne->IsBoolean() &&
- !argOne->IsString() && !argOne->IsNull() && !argOne->IsNumber()) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- if (argOne->IsBoolean() || argOne->IsString() || argOne->IsNumber()) {
- args.GetReturnValue()->Assign(argOne.get());
- return;
- }
-
- std::vector<std::unique_ptr<CFXJSE_Value>> values;
- for (int32_t i = 0; i < 3; i++)
- values.push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- int intVal = 3;
- if (argOne->IsNull()) {
- // TODO(dsinclair): Why is this 4 when the others are all 3?
- intVal = 4;
- values[2]->SetNull();
- } else if (argOne->IsArray()) {
-#ifndef NDEBUG
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argOne->GetObjectProperty("length", lengthValue.get());
- ASSERT(lengthValue->ToInteger() >= 3);
-#endif
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argOne->GetObjectPropertyByIdx(1, propertyValue.get());
- argOne->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (!propertyValue->IsNull() || jsObjectValue->IsNull()) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- values[2]->Assign(jsObjectValue.get());
- } else if (argOne->IsObject()) {
- values[2]->Assign(argOne.get());
- }
-
- values[0]->SetInteger(intVal);
- values[1]->SetNull();
- args.GetReturnValue()->SetArray(values);
-}
-
-// static
-void CXFA_FM2JSContext::UnitType(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"UnitType");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> unitspanValue = GetSimpleValue(pThis, args, 0);
- if (unitspanValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString unitspanString = ValueToUTF8String(unitspanValue.get());
- if (unitspanString.IsEmpty()) {
- args.GetReturnValue()->SetString("in");
- return;
- }
-
- enum XFA_FM2JS_VALUETYPE_ParserStatus {
- VALUETYPE_START,
- VALUETYPE_HAVEINVALIDCHAR,
- VALUETYPE_HAVEDIGIT,
- VALUETYPE_HAVEDIGITWHITE,
- VALUETYPE_ISCM,
- VALUETYPE_ISMM,
- VALUETYPE_ISPT,
- VALUETYPE_ISMP,
- VALUETYPE_ISIN,
- };
- unitspanString.MakeLower();
- WideString wsTypeString = WideString::FromUTF8(unitspanString.AsStringView());
- const wchar_t* pData = wsTypeString.c_str();
- int32_t u = 0;
- int32_t uLen = wsTypeString.GetLength();
- while (IsWhitespace(pData[u]))
- u++;
-
- XFA_FM2JS_VALUETYPE_ParserStatus eParserStatus = VALUETYPE_START;
- wchar_t typeChar;
- // TODO(dsinclair): Cleanup this parser, figure out what the various checks
- // are for.
- while (u < uLen) {
- typeChar = pData[u];
- if (IsWhitespace(typeChar)) {
- if (eParserStatus != VALUETYPE_HAVEDIGIT &&
- eParserStatus != VALUETYPE_HAVEDIGITWHITE) {
- eParserStatus = VALUETYPE_ISIN;
- break;
- }
- eParserStatus = VALUETYPE_HAVEDIGITWHITE;
- } else if (IsPartOfNumberW(typeChar)) {
- if (eParserStatus == VALUETYPE_HAVEDIGITWHITE) {
- eParserStatus = VALUETYPE_ISIN;
- break;
- }
- eParserStatus = VALUETYPE_HAVEDIGIT;
- } else if ((typeChar == 'c' || typeChar == 'p') && (u + 1 < uLen)) {
- wchar_t nextChar = pData[u + 1];
- if ((eParserStatus == VALUETYPE_START ||
- eParserStatus == VALUETYPE_HAVEDIGIT ||
- eParserStatus == VALUETYPE_HAVEDIGITWHITE) &&
- !IsPartOfNumberW(nextChar)) {
- eParserStatus = (typeChar == 'c') ? VALUETYPE_ISCM : VALUETYPE_ISPT;
- break;
- }
- eParserStatus = VALUETYPE_HAVEINVALIDCHAR;
- } else if (typeChar == 'm' && (u + 1 < uLen)) {
- wchar_t nextChar = pData[u + 1];
- if ((eParserStatus == VALUETYPE_START ||
- eParserStatus == VALUETYPE_HAVEDIGIT ||
- eParserStatus == VALUETYPE_HAVEDIGITWHITE) &&
- !IsPartOfNumberW(nextChar)) {
- eParserStatus = VALUETYPE_ISMM;
- if (nextChar == 'p' || ((u + 5 < uLen) && pData[u + 1] == 'i' &&
- pData[u + 2] == 'l' && pData[u + 3] == 'l' &&
- pData[u + 4] == 'i' && pData[u + 5] == 'p')) {
- eParserStatus = VALUETYPE_ISMP;
- }
- break;
- }
- } else {
- eParserStatus = VALUETYPE_HAVEINVALIDCHAR;
- }
- u++;
- }
- switch (eParserStatus) {
- case VALUETYPE_ISCM:
- args.GetReturnValue()->SetString("cm");
- break;
- case VALUETYPE_ISMM:
- args.GetReturnValue()->SetString("mm");
- break;
- case VALUETYPE_ISPT:
- args.GetReturnValue()->SetString("pt");
- break;
- case VALUETYPE_ISMP:
- args.GetReturnValue()->SetString("mp");
- break;
- default:
- args.GetReturnValue()->SetString("in");
- break;
- }
-}
-
-// static
-void CXFA_FM2JSContext::UnitValue(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"UnitValue");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> unitspanValue = GetSimpleValue(pThis, args, 0);
- if (unitspanValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString unitspanString = ValueToUTF8String(unitspanValue.get());
- const char* pData = unitspanString.c_str();
- if (!pData) {
- args.GetReturnValue()->SetInteger(0);
- return;
- }
-
- size_t u = 0;
- while (IsWhitespace(pData[u]))
- ++u;
-
- while (u < unitspanString.GetLength()) {
- if (!IsPartOfNumber(pData[u]))
- break;
- ++u;
- }
-
- char* pTemp = nullptr;
- double dFirstNumber = strtod(pData, &pTemp);
- while (IsWhitespace(pData[u]))
- ++u;
-
- size_t uLen = unitspanString.GetLength();
- ByteString strFirstUnit;
- while (u < uLen) {
- if (pData[u] == ' ')
- break;
-
- strFirstUnit += pData[u];
- ++u;
- }
- strFirstUnit.MakeLower();
-
- ByteString strUnit;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> unitValue = GetSimpleValue(pThis, args, 1);
- ByteString unitTempString = ValueToUTF8String(unitValue.get());
- const char* pChar = unitTempString.c_str();
- size_t uVal = 0;
- while (IsWhitespace(pChar[uVal]))
- ++uVal;
-
- while (uVal < unitTempString.GetLength()) {
- if (!std::isdigit(pChar[uVal]) && pChar[uVal] != '.')
- break;
- ++uVal;
- }
- while (IsWhitespace(pChar[uVal]))
- ++uVal;
-
- size_t uValLen = unitTempString.GetLength();
- while (uVal < uValLen) {
- if (pChar[uVal] == ' ')
- break;
-
- strUnit += pChar[uVal];
- ++uVal;
- }
- strUnit.MakeLower();
- } else {
- strUnit = strFirstUnit;
- }
-
- double dResult = 0;
- if (strFirstUnit == "in" || strFirstUnit == "inches") {
- if (strUnit == "mm" || strUnit == "millimeters")
- dResult = dFirstNumber * 25.4;
- else if (strUnit == "cm" || strUnit == "centimeters")
- dResult = dFirstNumber * 2.54;
- else if (strUnit == "pt" || strUnit == "points")
- dResult = dFirstNumber / 72;
- else if (strUnit == "mp" || strUnit == "millipoints")
- dResult = dFirstNumber / 72000;
- else
- dResult = dFirstNumber;
- } else if (strFirstUnit == "mm" || strFirstUnit == "millimeters") {
- if (strUnit == "mm" || strUnit == "millimeters")
- dResult = dFirstNumber;
- else if (strUnit == "cm" || strUnit == "centimeters")
- dResult = dFirstNumber / 10;
- else if (strUnit == "pt" || strUnit == "points")
- dResult = dFirstNumber / 25.4 / 72;
- else if (strUnit == "mp" || strUnit == "millipoints")
- dResult = dFirstNumber / 25.4 / 72000;
- else
- dResult = dFirstNumber / 25.4;
- } else if (strFirstUnit == "cm" || strFirstUnit == "centimeters") {
- if (strUnit == "mm" || strUnit == "millimeters")
- dResult = dFirstNumber * 10;
- else if (strUnit == "cm" || strUnit == "centimeters")
- dResult = dFirstNumber;
- else if (strUnit == "pt" || strUnit == "points")
- dResult = dFirstNumber / 2.54 / 72;
- else if (strUnit == "mp" || strUnit == "millipoints")
- dResult = dFirstNumber / 2.54 / 72000;
- else
- dResult = dFirstNumber / 2.54;
- } else if (strFirstUnit == "pt" || strFirstUnit == "points") {
- if (strUnit == "mm" || strUnit == "millimeters")
- dResult = dFirstNumber / 72 * 25.4;
- else if (strUnit == "cm" || strUnit == "centimeters")
- dResult = dFirstNumber / 72 * 2.54;
- else if (strUnit == "pt" || strUnit == "points")
- dResult = dFirstNumber;
- else if (strUnit == "mp" || strUnit == "millipoints")
- dResult = dFirstNumber * 1000;
- else
- dResult = dFirstNumber / 72;
- } else if (strFirstUnit == "mp" || strFirstUnit == "millipoints") {
- if (strUnit == "mm" || strUnit == "millimeters")
- dResult = dFirstNumber / 72000 * 25.4;
- else if (strUnit == "cm" || strUnit == "centimeters")
- dResult = dFirstNumber / 72000 * 2.54;
- else if (strUnit == "pt" || strUnit == "points")
- dResult = dFirstNumber / 1000;
- else if (strUnit == "mp" || strUnit == "millipoints")
- dResult = dFirstNumber;
- else
- dResult = dFirstNumber / 72000;
- }
- args.GetReturnValue()->SetDouble(dResult);
-}
-
-// static
-void CXFA_FM2JSContext::At(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"At");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString stringTwo = ValueToUTF8String(argTwo.get());
- if (stringTwo.IsEmpty()) {
- args.GetReturnValue()->SetInteger(1);
- return;
- }
-
- ByteString stringOne = ValueToUTF8String(argOne.get());
- auto pos = stringOne.Find(stringTwo.AsStringView());
- args.GetReturnValue()->SetInteger(pos.has_value() ? pos.value() + 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::Concat(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Concat");
- return;
- }
-
- ByteString resultString;
- bool bAllNull = true;
- for (int32_t i = 0; i < argc; i++) {
- std::unique_ptr<CFXJSE_Value> value = GetSimpleValue(pThis, args, i);
- if (ValueIsNull(pThis, value.get()))
- continue;
-
- bAllNull = false;
- resultString += ValueToUTF8String(value.get());
- }
-
- if (bAllNull) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- args.GetReturnValue()->SetString(resultString.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Decode(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Decode");
- return;
- }
-
- if (argc == 1) {
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- WideString decoded = DecodeURL(
- WideString::FromUTF8(ValueToUTF8String(argOne.get()).AsStringView()));
-
- args.GetReturnValue()->SetString(
- FX_UTF8Encode(decoded.AsStringView()).AsStringView());
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString toDecodeString = ValueToUTF8String(argOne.get());
- ByteString identifyString = ValueToUTF8String(argTwo.get());
- WideString decoded;
-
- WideString toDecodeWideString =
- WideString::FromUTF8(toDecodeString.AsStringView());
-
- if (identifyString.EqualNoCase("html"))
- decoded = DecodeHTML(toDecodeWideString);
- else if (identifyString.EqualNoCase("xml"))
- decoded = DecodeXML(toDecodeWideString);
- else
- decoded = DecodeURL(toDecodeWideString);
-
- args.GetReturnValue()->SetString(
- FX_UTF8Encode(decoded.AsStringView()).AsStringView());
-}
-
-// static
-WideString CXFA_FM2JSContext::DecodeURL(const WideString& wsURLString) {
- const wchar_t* pData = wsURLString.c_str();
- size_t i = 0;
- CFX_WideTextBuf wsResultBuf;
- while (i < wsURLString.GetLength()) {
- wchar_t ch = pData[i];
- if ('%' != ch) {
- wsResultBuf.AppendChar(ch);
- ++i;
- continue;
- }
-
- wchar_t chTemp = 0;
- int32_t iCount = 0;
- while (iCount < 2) {
- ++i;
- ch = pData[i];
- if (ch <= '9' && ch >= '0') {
- // TODO(dsinclair): Premultiply and add rather then scale.
- chTemp += (ch - '0') * (!iCount ? 16 : 1);
- } else if (ch <= 'F' && ch >= 'A') {
- chTemp += (ch - 'A' + 10) * (!iCount ? 16 : 1);
- } else if (ch <= 'f' && ch >= 'a') {
- chTemp += (ch - 'a' + 10) * (!iCount ? 16 : 1);
- } else {
- return WideString();
- }
- ++iCount;
- }
- wsResultBuf.AppendChar(chTemp);
- ++i;
- }
- wsResultBuf.AppendChar(0);
- return wsResultBuf.MakeString();
-}
-
-// static
-WideString CXFA_FM2JSContext::DecodeHTML(const WideString& wsHTMLString) {
- wchar_t strString[9];
- size_t iStrIndex = 0;
- size_t iLen = wsHTMLString.GetLength();
- size_t i = 0;
- int32_t iCode = 0;
- const wchar_t* pData = wsHTMLString.c_str();
- CFX_WideTextBuf wsResultBuf;
- while (i < iLen) {
- wchar_t ch = pData[i];
- if (ch != '&') {
- wsResultBuf.AppendChar(ch);
- ++i;
- continue;
- }
-
- ++i;
- ch = pData[i];
- if (ch == '#') {
- ++i;
- ch = pData[i];
- if (ch != 'x' && ch != 'X') {
- return WideString();
- }
-
- ++i;
- ch = pData[i];
- if ((ch >= '0' && ch <= '9') || (ch <= 'f' && ch >= 'a') ||
- (ch <= 'F' && ch >= 'A')) {
- while (ch != ';' && i < iLen) {
- if (ch >= '0' && ch <= '9') {
- iCode += ch - '0';
- } else if (ch <= 'f' && ch >= 'a') {
- iCode += ch - 'a' + 10;
- } else if (ch <= 'F' && ch >= 'A') {
- iCode += ch - 'A' + 10;
- } else {
- return WideString();
- }
- ++i;
- // TODO(dsinclair): Postmultiply seems wrong, start at zero
- // and pre-multiply then can remove the post divide.
- iCode *= 16;
- ch = pData[i];
- }
- iCode /= 16;
- }
- } else {
- while (ch != ';' && i < iLen) {
- strString[iStrIndex++] = ch;
- ++i;
- ch = pData[i];
- }
- strString[iStrIndex] = 0;
- }
- uint32_t iData = 0;
- if (HTMLSTR2Code(strString, &iData)) {
- wsResultBuf.AppendChar((wchar_t)iData);
- } else {
- wsResultBuf.AppendChar(iCode);
- }
- iStrIndex = 0;
- strString[iStrIndex] = 0;
- ++i;
- }
- wsResultBuf.AppendChar(0);
-
- return wsResultBuf.MakeString();
-}
-
-// static
-WideString CXFA_FM2JSContext::DecodeXML(const WideString& wsXMLString) {
- wchar_t strString[9];
- int32_t iStrIndex = 0;
- int32_t iLen = wsXMLString.GetLength();
- int32_t i = 0;
- int32_t iCode = 0;
- wchar_t ch = 0;
- const wchar_t* pData = wsXMLString.c_str();
- CFX_WideTextBuf wsResultBuf;
- while (i < iLen) {
- ch = pData[i];
- if (ch != '&') {
- wsResultBuf.AppendChar(ch);
- ++i;
- continue;
- }
-
- // TODO(dsinclair): This is very similar to DecodeHTML, can they be
- // combined?
- ++i;
- ch = pData[i];
- if (ch == '#') {
- ++i;
- ch = pData[i];
- if (ch != 'x' && ch != 'X') {
- return WideString();
- }
-
- ++i;
- ch = pData[i];
- if ((ch >= '0' && ch <= '9') || (ch <= 'f' && ch >= 'a') ||
- (ch <= 'F' && ch >= 'A')) {
- while (ch != ';') {
- if (ch >= '0' && ch <= '9') {
- iCode += ch - '0';
- } else if (ch <= 'f' && ch >= 'a') {
- iCode += ch - 'a' + 10;
- } else if (ch <= 'F' && ch >= 'A') {
- iCode += ch - 'A' + 10;
- } else {
- return WideString();
- }
- ++i;
- iCode *= 16;
- ch = pData[i];
- }
- iCode /= 16;
- }
- } else {
- while (ch != ';' && i < iLen) {
- strString[iStrIndex++] = ch;
- ++i;
- ch = pData[i];
- }
- strString[iStrIndex] = 0;
- }
-
- const wchar_t* const strName[] = {L"quot", L"amp", L"apos", L"lt", L"gt"};
- int32_t iIndex = 0;
- while (iIndex < 5) {
- if (memcmp(strString, strName[iIndex], wcslen(strName[iIndex])) == 0) {
- break;
- }
- ++iIndex;
- }
- switch (iIndex) {
- case 0:
- wsResultBuf.AppendChar('"');
- break;
- case 1:
- wsResultBuf.AppendChar('&');
- break;
- case 2:
- wsResultBuf.AppendChar('\'');
- break;
- case 3:
- wsResultBuf.AppendChar('<');
- break;
- case 4:
- wsResultBuf.AppendChar('>');
- break;
- default:
- wsResultBuf.AppendChar(iCode);
- break;
- }
- iStrIndex = 0;
- strString[iStrIndex] = 0;
- ++i;
- iCode = 0;
- }
- wsResultBuf.AppendChar(0);
- return wsResultBuf.MakeString();
-}
-
-// static
-void CXFA_FM2JSContext::Encode(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Encode");
- return;
- }
-
- if (argc == 1) {
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- WideString encoded = EncodeURL(ValueToUTF8String(argOne.get()));
- args.GetReturnValue()->SetString(
- FX_UTF8Encode(encoded.AsStringView()).AsStringView());
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- if (ValueIsNull(pThis, argOne.get()) || ValueIsNull(pThis, argTwo.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString toEncodeString = ValueToUTF8String(argOne.get());
- ByteString identifyString = ValueToUTF8String(argTwo.get());
- WideString encoded;
- if (identifyString.EqualNoCase("html"))
- encoded = EncodeHTML(toEncodeString);
- else if (identifyString.EqualNoCase("xml"))
- encoded = EncodeXML(toEncodeString);
- else
- encoded = EncodeURL(toEncodeString);
-
- args.GetReturnValue()->SetString(
- FX_UTF8Encode(encoded.AsStringView()).AsStringView());
-}
-
-// static
-WideString CXFA_FM2JSContext::EncodeURL(const ByteString& szURLString) {
- WideString wsURLString = WideString::FromUTF8(szURLString.AsStringView());
- CFX_WideTextBuf wsResultBuf;
- wchar_t strEncode[4];
- strEncode[0] = '%';
- strEncode[3] = 0;
- wchar_t strUnsafe[] = {' ', '<', '>', '"', '#', '%', '{', '}',
- '|', '\\', '^', '~', '[', ']', '`'};
- wchar_t strReserved[] = {';', '/', '?', ':', '@', '=', '&'};
- wchar_t strSpecial[] = {'$', '-', '+', '!', '*', '\'', '(', ')', ','};
- const wchar_t* strCode = L"0123456789abcdef";
- for (auto ch : wsURLString) {
- int32_t i = 0;
- int32_t iCount = FX_ArraySize(strUnsafe);
- while (i < iCount) {
- if (ch == strUnsafe[i]) {
- int32_t iIndex = ch / 16;
- strEncode[1] = strCode[iIndex];
- strEncode[2] = strCode[ch - iIndex * 16];
- wsResultBuf << strEncode;
- break;
- }
- ++i;
- }
- if (i < iCount)
- continue;
-
- i = 0;
- iCount = FX_ArraySize(strReserved);
- while (i < iCount) {
- if (ch == strReserved[i]) {
- int32_t iIndex = ch / 16;
- strEncode[1] = strCode[iIndex];
- strEncode[2] = strCode[ch - iIndex * 16];
- wsResultBuf << strEncode;
- break;
- }
- ++i;
- }
- if (i < iCount)
- continue;
-
- i = 0;
- iCount = FX_ArraySize(strSpecial);
- while (i < iCount) {
- if (ch == strSpecial[i]) {
- wsResultBuf.AppendChar(ch);
- break;
- }
- ++i;
- }
- if (i < iCount)
- continue;
-
- if ((ch >= 0x80 && ch <= 0xff) || ch <= 0x1f || ch == 0x7f) {
- int32_t iIndex = ch / 16;
- strEncode[1] = strCode[iIndex];
- strEncode[2] = strCode[ch - iIndex * 16];
- wsResultBuf << strEncode;
- } else if (ch >= 0x20 && ch <= 0x7e) {
- wsResultBuf.AppendChar(ch);
- } else {
- const wchar_t iRadix = 16;
- WideString strTmp;
- while (ch >= iRadix) {
- wchar_t tmp = strCode[ch % iRadix];
- ch /= iRadix;
- strTmp += tmp;
- }
- strTmp += strCode[ch];
- int32_t iLen = strTmp.GetLength();
- if (iLen < 2)
- break;
-
- int32_t iIndex = 0;
- if (iLen % 2 != 0) {
- strEncode[1] = '0';
- strEncode[2] = strTmp[iLen - 1];
- iIndex = iLen - 2;
- } else {
- strEncode[1] = strTmp[iLen - 1];
- strEncode[2] = strTmp[iLen - 2];
- iIndex = iLen - 3;
- }
- wsResultBuf << strEncode;
- while (iIndex > 0) {
- strEncode[1] = strTmp[iIndex];
- strEncode[2] = strTmp[iIndex - 1];
- iIndex -= 2;
- wsResultBuf << strEncode;
- }
- }
- }
- wsResultBuf.AppendChar(0);
- return wsResultBuf.MakeString();
-}
-
-// static
-WideString CXFA_FM2JSContext::EncodeHTML(const ByteString& szHTMLString) {
- WideString wsHTMLString = WideString::FromUTF8(szHTMLString.AsStringView());
- const wchar_t* strCode = L"0123456789abcdef";
- wchar_t strEncode[9];
- strEncode[0] = '&';
- strEncode[1] = '#';
- strEncode[2] = 'x';
- strEncode[5] = ';';
- strEncode[6] = 0;
- strEncode[7] = ';';
- strEncode[8] = 0;
- CFX_WideTextBuf wsResultBuf;
- int32_t iLen = wsHTMLString.GetLength();
- int32_t i = 0;
- const wchar_t* pData = wsHTMLString.c_str();
- while (i < iLen) {
- uint32_t ch = pData[i];
- WideString htmlReserve;
- if (HTMLCode2STR(ch, &htmlReserve)) {
- wsResultBuf.AppendChar(L'&');
- wsResultBuf << htmlReserve;
- wsResultBuf.AppendChar(L';');
- } else if (ch >= 32 && ch <= 126) {
- wsResultBuf.AppendChar((wchar_t)ch);
- } else if (ch < 256) {
- int32_t iIndex = ch / 16;
- strEncode[3] = strCode[iIndex];
- strEncode[4] = strCode[ch - iIndex * 16];
- strEncode[5] = ';';
- strEncode[6] = 0;
- wsResultBuf << strEncode;
- } else {
- int32_t iBigByte = ch / 256;
- int32_t iLittleByte = ch % 256;
- strEncode[3] = strCode[iBigByte / 16];
- strEncode[4] = strCode[iBigByte % 16];
- strEncode[5] = strCode[iLittleByte / 16];
- strEncode[6] = strCode[iLittleByte % 16];
- wsResultBuf << strEncode;
- }
- ++i;
- }
- wsResultBuf.AppendChar(0);
- return wsResultBuf.MakeString();
-}
-
-// static
-WideString CXFA_FM2JSContext::EncodeXML(const ByteString& szXMLString) {
- WideString wsXMLString = WideString::FromUTF8(szXMLString.AsStringView());
- CFX_WideTextBuf wsResultBuf;
- wchar_t strEncode[9];
- strEncode[0] = '&';
- strEncode[1] = '#';
- strEncode[2] = 'x';
- strEncode[5] = ';';
- strEncode[6] = 0;
- strEncode[7] = ';';
- strEncode[8] = 0;
- const wchar_t* strCode = L"0123456789abcdef";
- for (const auto& ch : wsXMLString) {
- switch (ch) {
- case '"':
- wsResultBuf.AppendChar('&');
- wsResultBuf << WideStringView(L"quot");
- wsResultBuf.AppendChar(';');
- break;
- case '&':
- wsResultBuf.AppendChar('&');
- wsResultBuf << WideStringView(L"amp");
- wsResultBuf.AppendChar(';');
- break;
- case '\'':
- wsResultBuf.AppendChar('&');
- wsResultBuf << WideStringView(L"apos");
- wsResultBuf.AppendChar(';');
- break;
- case '<':
- wsResultBuf.AppendChar('&');
- wsResultBuf << WideStringView(L"lt");
- wsResultBuf.AppendChar(';');
- break;
- case '>':
- wsResultBuf.AppendChar('&');
- wsResultBuf << WideStringView(L"gt");
- wsResultBuf.AppendChar(';');
- break;
- default: {
- if (ch >= 32 && ch <= 126) {
- wsResultBuf.AppendChar(ch);
- } else if (ch < 256) {
- int32_t iIndex = ch / 16;
- strEncode[3] = strCode[iIndex];
- strEncode[4] = strCode[ch - iIndex * 16];
- strEncode[5] = ';';
- strEncode[6] = 0;
- wsResultBuf << strEncode;
- } else {
- int32_t iBigByte = ch / 256;
- int32_t iLittleByte = ch % 256;
- strEncode[3] = strCode[iBigByte / 16];
- strEncode[4] = strCode[iBigByte % 16];
- strEncode[5] = strCode[iLittleByte / 16];
- strEncode[6] = strCode[iLittleByte % 16];
- wsResultBuf << strEncode;
- }
- break;
- }
- }
- }
- wsResultBuf.AppendChar(0);
- return wsResultBuf.MakeString();
-}
-
-// static
-bool CXFA_FM2JSContext::HTMLSTR2Code(const WideStringView& pData,
- uint32_t* iCode) {
- auto cmpFunc = [](const XFA_FMHtmlReserveCode& iter,
- const WideStringView& val) {
- // TODO(tsepez): check usage of c_str() below.
- return wcscmp(val.unterminated_c_str(), iter.m_htmlReserve) > 0;
- };
- const XFA_FMHtmlReserveCode* result =
- std::lower_bound(std::begin(reservesForDecode),
- std::end(reservesForDecode), pData, cmpFunc);
- if (result != std::end(reservesForEncode) &&
- !wcscmp(pData.unterminated_c_str(), result->m_htmlReserve)) {
- *iCode = result->m_uCode;
- return true;
- }
- return false;
-}
-
-// static
-bool CXFA_FM2JSContext::HTMLCode2STR(uint32_t iCode,
- WideString* wsHTMLReserve) {
- auto cmpFunc = [](const XFA_FMHtmlReserveCode iter, const uint32_t& val) {
- return iter.m_uCode < val;
- };
- const XFA_FMHtmlReserveCode* result =
- std::lower_bound(std::begin(reservesForEncode),
- std::end(reservesForEncode), iCode, cmpFunc);
- if (result != std::end(reservesForEncode) && result->m_uCode == iCode) {
- *wsHTMLReserve = result->m_htmlReserve;
- return true;
- }
- return false;
-}
-
-// static
-void CXFA_FM2JSContext::Format(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() < 2) {
- pContext->ThrowParamCountMismatchException(L"Format");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- ByteString szPattern = ValueToUTF8String(argOne.get());
-
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- ByteString szValue = ValueToUTF8String(argTwo.get());
-
- CXFA_Document* pDoc = pContext->GetDocument();
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- CXFA_Node* pThisNode = ToNode(pDoc->GetScriptContext()->GetThisObject());
- ASSERT(pThisNode);
-
- CXFA_WidgetData widgetData(pThisNode);
- IFX_Locale* pLocale = widgetData.GetLocal();
- uint32_t patternType;
- WideString wsPattern = WideString::FromUTF8(szPattern.AsStringView());
- WideString wsValue = WideString::FromUTF8(szValue.AsStringView());
- if (!PatternStringType(szPattern.AsStringView(), patternType)) {
- switch (patternType) {
- case XFA_VT_DATETIME: {
- auto iTChar = wsPattern.Find(L'T');
- if (!iTChar.has_value()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- WideString wsDatePattern(L"date{");
- wsDatePattern += wsPattern.Left(iTChar.value()) + L"} ";
-
- WideString wsTimePattern(L"time{");
- wsTimePattern +=
- wsPattern.Right(wsPattern.GetLength() - (iTChar.value() + 1)) +
- L"}";
- wsPattern = wsDatePattern + wsTimePattern;
- } break;
- case XFA_VT_DATE: {
- wsPattern = L"date{" + wsPattern + L"}";
- } break;
- case XFA_VT_TIME: {
- wsPattern = L"time{" + wsPattern + L"}";
- } break;
- case XFA_VT_TEXT: {
- wsPattern = L"text{" + wsPattern + L"}";
- } break;
- case XFA_VT_FLOAT: {
- wsPattern = L"num{" + wsPattern + L"}";
- } break;
- default: {
- WideString wsTestPattern;
- wsTestPattern = L"num{" + wsPattern + L"}";
- CXFA_LocaleValue tempLocaleValue(XFA_VT_FLOAT, wsValue, wsTestPattern,
- pLocale, pMgr);
- if (tempLocaleValue.IsValid()) {
- wsPattern = wsTestPattern;
- patternType = XFA_VT_FLOAT;
- } else {
- wsTestPattern = L"text{" + wsPattern + L"}";
- wsPattern = wsTestPattern;
- patternType = XFA_VT_TEXT;
- }
- } break;
- }
- }
- CXFA_LocaleValue localeValue(patternType, wsValue, wsPattern, pLocale, pMgr);
- WideString wsRet;
- if (!localeValue.FormatPatterns(wsRet, wsPattern, pLocale,
- XFA_VALUEPICTURE_Display)) {
- args.GetReturnValue()->SetString("");
- return;
- }
-
- args.GetReturnValue()->SetString(wsRet.UTF8Encode().AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Left(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Left");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- if ((ValueIsNull(pThis, argOne.get())) ||
- (ValueIsNull(pThis, argTwo.get()))) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString sourceString = ValueToUTF8String(argOne.get());
- int32_t count = std::max(0, ValueToInteger(pThis, argTwo.get()));
- args.GetReturnValue()->SetString(sourceString.Left(count).AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Len(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Len");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString sourceString = ValueToUTF8String(argOne.get());
- args.GetReturnValue()->SetInteger(sourceString.GetLength());
-}
-
-// static
-void CXFA_FM2JSContext::Lower(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Lower");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- CFX_WideTextBuf lowStringBuf;
- ByteString argString = ValueToUTF8String(argOne.get());
- WideString wsArgString = WideString::FromUTF8(argString.AsStringView());
- const wchar_t* pData = wsArgString.c_str();
- size_t i = 0;
- while (i < argString.GetLength()) {
- int32_t ch = pData[i];
- if ((ch >= 0x41 && ch <= 0x5A) || (ch >= 0xC0 && ch <= 0xDE))
- ch += 32;
- else if (ch == 0x100 || ch == 0x102 || ch == 0x104)
- ch += 1;
-
- lowStringBuf.AppendChar(ch);
- ++i;
- }
- lowStringBuf.AppendChar(0);
-
- args.GetReturnValue()->SetString(
- FX_UTF8Encode(lowStringBuf.AsStringView()).AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Ltrim(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Ltrim");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString sourceString = ValueToUTF8String(argOne.get());
- sourceString.TrimLeft();
- args.GetReturnValue()->SetString(sourceString.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Parse(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 2) {
- pContext->ThrowParamCountMismatchException(L"Parse");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- if (ValueIsNull(pThis, argTwo.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString szPattern = ValueToUTF8String(argOne.get());
- ByteString szValue = ValueToUTF8String(argTwo.get());
- CXFA_Document* pDoc = pContext->GetDocument();
- CXFA_LocaleMgr* pMgr = pDoc->GetLocalMgr();
- CXFA_Node* pThisNode = ToNode(pDoc->GetScriptContext()->GetThisObject());
- ASSERT(pThisNode);
-
- CXFA_WidgetData widgetData(pThisNode);
- IFX_Locale* pLocale = widgetData.GetLocal();
- WideString wsPattern = WideString::FromUTF8(szPattern.AsStringView());
- WideString wsValue = WideString::FromUTF8(szValue.AsStringView());
- uint32_t patternType;
- if (PatternStringType(szPattern.AsStringView(), patternType)) {
- CXFA_LocaleValue localeValue(patternType, wsValue, wsPattern, pLocale,
- pMgr);
- if (!localeValue.IsValid()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- args.GetReturnValue()->SetString(
- localeValue.GetValue().UTF8Encode().AsStringView());
- return;
- }
-
- switch (patternType) {
- case XFA_VT_DATETIME: {
- auto iTChar = wsPattern.Find(L'T');
- if (!iTChar.has_value()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- WideString wsDatePattern(L"date{" + wsPattern.Left(iTChar.value()) +
- L"} ");
- WideString wsTimePattern(
- L"time{" +
- wsPattern.Right(wsPattern.GetLength() - (iTChar.value() + 1)) + L"}");
- wsPattern = wsDatePattern + wsTimePattern;
- CXFA_LocaleValue localeValue(patternType, wsValue, wsPattern, pLocale,
- pMgr);
- if (!localeValue.IsValid()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- args.GetReturnValue()->SetString(
- localeValue.GetValue().UTF8Encode().AsStringView());
- return;
- }
- case XFA_VT_DATE: {
- wsPattern = L"date{" + wsPattern + L"}";
- CXFA_LocaleValue localeValue(patternType, wsValue, wsPattern, pLocale,
- pMgr);
- if (!localeValue.IsValid()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- args.GetReturnValue()->SetString(
- localeValue.GetValue().UTF8Encode().AsStringView());
- return;
- }
- case XFA_VT_TIME: {
- wsPattern = L"time{" + wsPattern + L"}";
- CXFA_LocaleValue localeValue(patternType, wsValue, wsPattern, pLocale,
- pMgr);
- if (!localeValue.IsValid()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- args.GetReturnValue()->SetString(
- localeValue.GetValue().UTF8Encode().AsStringView());
- return;
- }
- case XFA_VT_TEXT: {
- wsPattern = L"text{" + wsPattern + L"}";
- CXFA_LocaleValue localeValue(XFA_VT_TEXT, wsValue, wsPattern, pLocale,
- pMgr);
- if (!localeValue.IsValid()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- args.GetReturnValue()->SetString(
- localeValue.GetValue().UTF8Encode().AsStringView());
- return;
- }
- case XFA_VT_FLOAT: {
- wsPattern = L"num{" + wsPattern + L"}";
- CXFA_LocaleValue localeValue(XFA_VT_FLOAT, wsValue, wsPattern, pLocale,
- pMgr);
- if (!localeValue.IsValid()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- args.GetReturnValue()->SetDouble(localeValue.GetDoubleNum());
- return;
- }
- default: {
- WideString wsTestPattern;
- wsTestPattern = L"num{" + wsPattern + L"}";
- CXFA_LocaleValue localeValue(XFA_VT_FLOAT, wsValue, wsTestPattern,
- pLocale, pMgr);
- if (localeValue.IsValid()) {
- args.GetReturnValue()->SetDouble(localeValue.GetDoubleNum());
- return;
- }
-
- wsTestPattern = L"text{" + wsPattern + L"}";
- CXFA_LocaleValue localeValue2(XFA_VT_TEXT, wsValue, wsTestPattern,
- pLocale, pMgr);
- if (!localeValue2.IsValid()) {
- args.GetReturnValue()->SetString("");
- return;
- }
- args.GetReturnValue()->SetString(
- localeValue2.GetValue().UTF8Encode().AsStringView());
- return;
- }
- }
-}
-
-// static
-void CXFA_FM2JSContext::Replace(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 2 || argc > 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Replace");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- ByteString oneString;
- ByteString twoString;
- if (!ValueIsNull(pThis, argOne.get()) && !ValueIsNull(pThis, argTwo.get())) {
- oneString = ValueToUTF8String(argOne.get());
- twoString = ValueToUTF8String(argTwo.get());
- }
-
- ByteString threeString;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- threeString = ValueToUTF8String(argThree.get());
- }
-
- size_t iFindLen = twoString.GetLength();
- std::ostringstream resultString;
- size_t iFindIndex = 0;
- for (size_t u = 0; u < oneString.GetLength(); ++u) {
- char ch = static_cast<char>(oneString[u]);
- if (ch != static_cast<char>(twoString[iFindIndex])) {
- resultString << ch;
- continue;
- }
-
- size_t iTemp = u + 1;
- ++iFindIndex;
- while (iFindIndex < iFindLen) {
- uint8_t chTemp = oneString[iTemp];
- if (chTemp != twoString[iFindIndex]) {
- iFindIndex = 0;
- break;
- }
-
- ++iTemp;
- ++iFindIndex;
- }
- if (iFindIndex == iFindLen) {
- resultString << threeString;
- u += iFindLen - 1;
- iFindIndex = 0;
- } else {
- resultString << ch;
- }
- }
- resultString << '\0';
- args.GetReturnValue()->SetString(ByteStringView(resultString.str().c_str()));
-}
-
-// static
-void CXFA_FM2JSContext::Right(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Right");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- if ((ValueIsNull(pThis, argOne.get())) ||
- (ValueIsNull(pThis, argTwo.get()))) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString sourceString = ValueToUTF8String(argOne.get());
- int32_t count = std::max(0, ValueToInteger(pThis, argTwo.get()));
- args.GetReturnValue()->SetString(sourceString.Right(count).AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Rtrim(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Rtrim");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- ByteString sourceString = ValueToUTF8String(argOne.get());
- sourceString.TrimRight();
- args.GetReturnValue()->SetString(sourceString.AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Space(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Space");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (argOne->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- int32_t count = std::max(0, ValueToInteger(pThis, argOne.get()));
- std::ostringstream spaceString;
- int32_t index = 0;
- while (index < count) {
- spaceString << ' ';
- index++;
- }
- spaceString << '\0';
- args.GetReturnValue()->SetString(ByteStringView(spaceString.str().c_str()));
-}
-
-// static
-void CXFA_FM2JSContext::Str(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Str");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> numberValue = GetSimpleValue(pThis, args, 0);
- if (numberValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- float fNumber = ValueToFloat(pThis, numberValue.get());
-
- int32_t iWidth = 10;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> widthValue = GetSimpleValue(pThis, args, 1);
- iWidth = static_cast<int32_t>(ValueToFloat(pThis, widthValue.get()));
- }
-
- int32_t iPrecision = 0;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> precisionValue =
- GetSimpleValue(pThis, args, 2);
- iPrecision = std::max(
- 0, static_cast<int32_t>(ValueToFloat(pThis, precisionValue.get())));
- }
-
- ByteString numberString;
- ByteString formatStr = "%";
- if (iPrecision) {
- formatStr += ".";
- formatStr += ByteString::FormatInteger(iPrecision);
- }
- formatStr += "f";
- numberString.Format(formatStr.c_str(), fNumber);
-
- const char* pData = numberString.c_str();
- int32_t iLength = numberString.GetLength();
- int32_t u = 0;
- while (u < iLength) {
- if (pData[u] == '.')
- break;
-
- ++u;
- }
-
- std::ostringstream resultBuf;
- if (u > iWidth || (iPrecision + u) >= iWidth) {
- int32_t i = 0;
- while (i < iWidth) {
- resultBuf << '*';
- ++i;
- }
- resultBuf << '\0';
- args.GetReturnValue()->SetString(ByteStringView(resultBuf.str().c_str()));
- return;
- }
-
- if (u == iLength) {
- if (iLength > iWidth) {
- int32_t i = 0;
- while (i < iWidth) {
- resultBuf << '*';
- ++i;
- }
- } else {
- int32_t i = 0;
- while (i < iWidth - iLength) {
- resultBuf << ' ';
- ++i;
- }
- resultBuf << pData;
- }
- args.GetReturnValue()->SetString(ByteStringView(resultBuf.str().c_str()));
- return;
- }
-
- int32_t iLeavingSpace = iWidth - u - iPrecision;
- if (iPrecision != 0)
- iLeavingSpace--;
-
- int32_t i = 0;
- while (i < iLeavingSpace) {
- resultBuf << ' ';
- ++i;
- }
- i = 0;
- while (i < u) {
- resultBuf << pData[i];
- ++i;
- }
- if (iPrecision != 0)
- resultBuf << '.';
-
- u++;
- i = 0;
- while (u < iLength) {
- if (i >= iPrecision)
- break;
-
- resultBuf << pData[u];
- ++i;
- ++u;
- }
- while (i < iPrecision) {
- resultBuf << '0';
- ++i;
- }
- resultBuf << '\0';
- args.GetReturnValue()->SetString(ByteStringView(resultBuf.str().c_str()));
-}
-
-// static
-void CXFA_FM2JSContext::Stuff(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 3 || argc > 4) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Stuff");
- return;
- }
-
- ByteString sourceString;
- ByteString insertString;
- int32_t iLength = 0;
- int32_t iStart = 0;
- int32_t iDelete = 0;
- std::unique_ptr<CFXJSE_Value> sourceValue = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> startValue = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> deleteValue = GetSimpleValue(pThis, args, 2);
- if (!sourceValue->IsNull() && !startValue->IsNull() &&
- !deleteValue->IsNull()) {
- sourceString = ValueToUTF8String(sourceValue.get());
- iLength = sourceString.GetLength();
- iStart = pdfium::clamp(
- static_cast<int32_t>(ValueToFloat(pThis, startValue.get())), 1,
- iLength);
- iDelete = std::max(
- 0, static_cast<int32_t>(ValueToFloat(pThis, deleteValue.get())));
- }
-
- if (argc > 3) {
- std::unique_ptr<CFXJSE_Value> insertValue = GetSimpleValue(pThis, args, 3);
- insertString = ValueToUTF8String(insertValue.get());
- }
-
- iStart -= 1;
- std::ostringstream resultString;
- int32_t i = 0;
- while (i < iStart) {
- resultString << static_cast<char>(sourceString[i]);
- ++i;
- }
- resultString << insertString.AsStringView();
- i = iStart + iDelete;
- while (i < iLength) {
- resultString << static_cast<char>(sourceString[i]);
- ++i;
- }
- resultString << '\0';
- args.GetReturnValue()->SetString(ByteStringView(resultString.str().c_str()));
-}
-
-// static
-void CXFA_FM2JSContext::Substr(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Substr");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> stringValue = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> startValue = GetSimpleValue(pThis, args, 1);
- std::unique_ptr<CFXJSE_Value> endValue = GetSimpleValue(pThis, args, 2);
- if (ValueIsNull(pThis, stringValue.get()) ||
- (ValueIsNull(pThis, startValue.get())) ||
- (ValueIsNull(pThis, endValue.get()))) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- int32_t iStart = 0;
- int32_t iCount = 0;
- ByteString szSourceStr = ValueToUTF8String(stringValue.get());
- int32_t iLength = szSourceStr.GetLength();
- if (iLength == 0) {
- args.GetReturnValue()->SetString("");
- return;
- }
-
- iStart = pdfium::clamp(
- iLength, 1, static_cast<int32_t>(ValueToFloat(pThis, startValue.get())));
- iCount =
- std::max(0, static_cast<int32_t>(ValueToFloat(pThis, endValue.get())));
-
- iStart -= 1;
- args.GetReturnValue()->SetString(
- szSourceStr.Mid(iStart, iCount).AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Uuid(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 0 || argc > 1) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Uuid");
- return;
- }
-
- int32_t iNum = 0;
- if (argc > 0) {
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- iNum = static_cast<int32_t>(ValueToFloat(pThis, argOne.get()));
- }
- args.GetReturnValue()->SetString(GUIDString(!!iNum).AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Upper(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 2) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"Upper");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (ValueIsNull(pThis, argOne.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- CFX_WideTextBuf upperStringBuf;
- ByteString argString = ValueToUTF8String(argOne.get());
- WideString wsArgString = WideString::FromUTF8(argString.AsStringView());
- const wchar_t* pData = wsArgString.c_str();
- size_t i = 0;
- while (i < wsArgString.GetLength()) {
- int32_t ch = pData[i];
- if ((ch >= 0x61 && ch <= 0x7A) || (ch >= 0xE0 && ch <= 0xFE))
- ch -= 32;
- else if (ch == 0x101 || ch == 0x103 || ch == 0x105)
- ch -= 1;
-
- upperStringBuf.AppendChar(ch);
- ++i;
- }
- upperStringBuf.AppendChar(0);
-
- args.GetReturnValue()->SetString(
- FX_UTF8Encode(upperStringBuf.AsStringView()).AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::WordNum(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- int32_t argc = args.GetLength();
- if (argc < 1 || argc > 3) {
- ToJSContext(pThis, nullptr)->ThrowParamCountMismatchException(L"WordNum");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> numberValue = GetSimpleValue(pThis, args, 0);
- if (numberValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- float fNumber = ValueToFloat(pThis, numberValue.get());
-
- int32_t iIdentifier = 0;
- if (argc > 1) {
- std::unique_ptr<CFXJSE_Value> identifierValue =
- GetSimpleValue(pThis, args, 1);
- if (identifierValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- iIdentifier =
- static_cast<int32_t>(ValueToFloat(pThis, identifierValue.get()));
- }
-
- ByteString localeString;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> localeValue = GetSimpleValue(pThis, args, 2);
- if (localeValue->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- localeString = ValueToUTF8String(localeValue.get());
- }
-
- if (fNumber < 0.0f || fNumber > 922337203685477550.0f) {
- args.GetReturnValue()->SetString("*");
- return;
- }
-
- ByteString numberString;
- numberString.Format("%.2f", fNumber);
-
- args.GetReturnValue()->SetString(
- WordUS(numberString, iIdentifier).AsStringView());
-}
-
-// static
-ByteString CXFA_FM2JSContext::TrillionUS(const ByteStringView& szData) {
- std::ostringstream strBuf;
- ByteStringView pUnits[] = {"zero", "one", "two", "three", "four",
- "five", "six", "seven", "eight", "nine"};
- ByteStringView pCapUnits[] = {"Zero", "One", "Two", "Three", "Four",
- "Five", "Six", "Seven", "Eight", "Nine"};
- ByteStringView pTens[] = {"Ten", "Eleven", "Twelve", "Thirteen",
- "Fourteen", "Fifteen", "Sixteen", "Seventeen",
- "Eighteen", "Nineteen"};
- ByteStringView pLastTens[] = {"Twenty", "Thirty", "Forty", "Fifty",
- "Sixty", "Seventy", "Eighty", "Ninety"};
- ByteStringView pComm[] = {" Hundred ", " Thousand ", " Million ", " Billion ",
- "Trillion"};
- const char* pData = szData.unterminated_c_str();
- int32_t iLength = szData.GetLength();
- int32_t iComm = 0;
- if (iLength > 12)
- iComm = 4;
- else if (iLength > 9)
- iComm = 3;
- else if (iLength > 6)
- iComm = 2;
- else if (iLength > 3)
- iComm = 1;
-
- int32_t iFirstCount = iLength % 3;
- if (iFirstCount == 0)
- iFirstCount = 3;
-
- int32_t iIndex = 0;
- if (iFirstCount == 3) {
- if (pData[iIndex] != '0') {
- strBuf << pCapUnits[pData[iIndex] - '0'];
- strBuf << pComm[0];
- }
- if (pData[iIndex + 1] == '0') {
- strBuf << pCapUnits[pData[iIndex + 2] - '0'];
- } else {
- if (pData[iIndex + 1] > '1') {
- strBuf << pLastTens[pData[iIndex + 1] - '2'];
- strBuf << "-";
- strBuf << pUnits[pData[iIndex + 2] - '0'];
- } else if (pData[iIndex + 1] == '1') {
- strBuf << pTens[pData[iIndex + 2] - '0'];
- } else if (pData[iIndex + 1] == '0') {
- strBuf << pCapUnits[pData[iIndex + 2] - '0'];
- }
- }
- iIndex += 3;
- } else if (iFirstCount == 2) {
- if (pData[iIndex] == '0') {
- strBuf << pCapUnits[pData[iIndex + 1] - '0'];
- } else {
- if (pData[iIndex] > '1') {
- strBuf << pLastTens[pData[iIndex] - '2'];
- strBuf << "-";
- strBuf << pUnits[pData[iIndex + 1] - '0'];
- } else if (pData[iIndex] == '1') {
- strBuf << pTens[pData[iIndex + 1] - '0'];
- } else if (pData[iIndex] == '0') {
- strBuf << pCapUnits[pData[iIndex + 1] - '0'];
- }
- }
- iIndex += 2;
- } else if (iFirstCount == 1) {
- strBuf << pCapUnits[pData[iIndex] - '0'];
- iIndex += 1;
- }
- if (iLength > 3 && iFirstCount > 0) {
- strBuf << pComm[iComm];
- --iComm;
- }
- while (iIndex < iLength) {
- if (pData[iIndex] != '0') {
- strBuf << pCapUnits[pData[iIndex] - '0'];
- strBuf << pComm[0];
- }
- if (pData[iIndex + 1] == '0') {
- strBuf << pCapUnits[pData[iIndex + 2] - '0'];
- } else {
- if (pData[iIndex + 1] > '1') {
- strBuf << pLastTens[pData[iIndex + 1] - '2'];
- strBuf << "-";
- strBuf << pUnits[pData[iIndex + 2] - '0'];
- } else if (pData[iIndex + 1] == '1') {
- strBuf << pTens[pData[iIndex + 2] - '0'];
- } else if (pData[iIndex + 1] == '0') {
- strBuf << pCapUnits[pData[iIndex + 2] - '0'];
- }
- }
- if (iIndex < iLength - 3) {
- strBuf << pComm[iComm];
- --iComm;
- }
- iIndex += 3;
- }
- return ByteString(strBuf);
-}
-
-// static
-ByteString CXFA_FM2JSContext::WordUS(const ByteString& szData, int32_t iStyle) {
- const char* pData = szData.c_str();
- int32_t iLength = szData.GetLength();
- if (iStyle < 0 || iStyle > 2) {
- return ByteString();
- }
-
- std::ostringstream strBuf;
-
- int32_t iIndex = 0;
- while (iIndex < iLength) {
- if (pData[iIndex] == '.')
- break;
- ++iIndex;
- }
- int32_t iInteger = iIndex;
- iIndex = 0;
- while (iIndex < iInteger) {
- int32_t iCount = (iInteger - iIndex) % 12;
- if (!iCount && iInteger - iIndex > 0)
- iCount = 12;
-
- strBuf << TrillionUS(ByteStringView(pData + iIndex, iCount));
- iIndex += iCount;
- if (iIndex < iInteger)
- strBuf << " Trillion ";
- }
-
- if (iStyle > 0)
- strBuf << " Dollars";
-
- if (iStyle > 1 && iInteger < iLength) {
- strBuf << " And ";
- iIndex = iInteger + 1;
- while (iIndex < iLength) {
- int32_t iCount = (iLength - iIndex) % 12;
- if (!iCount && iLength - iIndex > 0)
- iCount = 12;
-
- strBuf << TrillionUS(ByteStringView(pData + iIndex, iCount));
- iIndex += iCount;
- if (iIndex < iLength)
- strBuf << " Trillion ";
- }
- strBuf << " Cents";
- }
- return ByteString(strBuf);
-}
-
-// static
-void CXFA_FM2JSContext::Get(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 1) {
- pContext->ThrowParamCountMismatchException(L"Get");
- return;
- }
-
- CXFA_Document* pDoc = pContext->GetDocument();
- if (!pDoc)
- return;
-
- IXFA_AppProvider* pAppProvider = pDoc->GetNotify()->GetAppProvider();
- if (!pAppProvider)
- return;
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- ByteString urlString = ValueToUTF8String(argOne.get());
- RetainPtr<IFX_SeekableReadStream> pFile =
- pAppProvider->DownloadURL(WideString::FromUTF8(urlString.AsStringView()));
- if (!pFile)
- return;
-
- int32_t size = pFile->GetSize();
- std::vector<uint8_t> dataBuf(size);
- pFile->ReadBlock(dataBuf.data(), size);
- args.GetReturnValue()->SetString(ByteStringView(dataBuf));
-}
-
-// static
-void CXFA_FM2JSContext::Post(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- int32_t argc = args.GetLength();
- if (argc < 2 || argc > 5) {
- pContext->ThrowParamCountMismatchException(L"Post");
- return;
- }
-
- CXFA_Document* pDoc = pContext->GetDocument();
- if (!pDoc)
- return;
-
- IXFA_AppProvider* pAppProvider = pDoc->GetNotify()->GetAppProvider();
- if (!pAppProvider)
- return;
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- ByteString bsURL = ValueToUTF8String(argOne.get());
-
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- ByteString bsData = ValueToUTF8String(argTwo.get());
-
- ByteString bsContentType;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- bsContentType = ValueToUTF8String(argThree.get());
- }
-
- ByteString bsEncode;
- if (argc > 3) {
- std::unique_ptr<CFXJSE_Value> argFour = GetSimpleValue(pThis, args, 3);
- bsEncode = ValueToUTF8String(argFour.get());
- }
-
- ByteString bsHeader;
- if (argc > 4) {
- std::unique_ptr<CFXJSE_Value> argFive = GetSimpleValue(pThis, args, 4);
- bsHeader = ValueToUTF8String(argFive.get());
- }
-
- WideString decodedResponse;
- if (!pAppProvider->PostRequestURL(
- WideString::FromUTF8(bsURL.AsStringView()),
- WideString::FromUTF8(bsData.AsStringView()),
- WideString::FromUTF8(bsContentType.AsStringView()),
- WideString::FromUTF8(bsEncode.AsStringView()),
- WideString::FromUTF8(bsHeader.AsStringView()), decodedResponse)) {
- pContext->ThrowServerDeniedException();
- return;
- }
- args.GetReturnValue()->SetString(decodedResponse.UTF8Encode().AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::Put(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- int32_t argc = args.GetLength();
- if (argc < 2 || argc > 3) {
- pContext->ThrowParamCountMismatchException(L"Put");
- return;
- }
-
- CXFA_Document* pDoc = pContext->GetDocument();
- if (!pDoc)
- return;
-
- IXFA_AppProvider* pAppProvider = pDoc->GetNotify()->GetAppProvider();
- if (!pAppProvider)
- return;
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- ByteString bsURL = ValueToUTF8String(argOne.get());
-
- std::unique_ptr<CFXJSE_Value> argTwo = GetSimpleValue(pThis, args, 1);
- ByteString bsData = ValueToUTF8String(argTwo.get());
-
- ByteString bsEncode;
- if (argc > 2) {
- std::unique_ptr<CFXJSE_Value> argThree = GetSimpleValue(pThis, args, 2);
- bsEncode = ValueToUTF8String(argThree.get());
- }
-
- if (!pAppProvider->PutRequestURL(
- WideString::FromUTF8(bsURL.AsStringView()),
- WideString::FromUTF8(bsData.AsStringView()),
- WideString::FromUTF8(bsEncode.AsStringView()))) {
- pContext->ThrowServerDeniedException();
- return;
- }
-
- args.GetReturnValue()->SetString("");
-}
-
-// static
-void CXFA_FM2JSContext::assign_value_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 2) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> lValue = args.GetValue(0);
- std::unique_ptr<CFXJSE_Value> rValue = GetSimpleValue(pThis, args, 1);
- if (lValue->IsArray()) {
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- auto leftLengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- lValue->GetObjectProperty("length", leftLengthValue.get());
- int32_t iLeftLength = leftLengthValue->ToInteger();
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- lValue->GetObjectPropertyByIdx(1, propertyValue.get());
- if (propertyValue->IsNull()) {
- for (int32_t i = 2; i < iLeftLength; i++) {
- lValue->GetObjectPropertyByIdx(i, jsObjectValue.get());
- if (!SetObjectDefaultValue(jsObjectValue.get(), rValue.get())) {
- pContext->ThrowNoDefaultPropertyException(szFuncName);
- return;
- }
- }
- } else {
- for (int32_t i = 2; i < iLeftLength; i++) {
- lValue->GetObjectPropertyByIdx(i, jsObjectValue.get());
- jsObjectValue->SetObjectProperty(
- propertyValue->ToString().AsStringView(), rValue.get());
- }
- }
- } else if (lValue->IsObject()) {
- if (!SetObjectDefaultValue(lValue.get(), rValue.get())) {
- pContext->ThrowNoDefaultPropertyException(szFuncName);
- return;
- }
- }
- args.GetReturnValue()->Assign(rValue.get());
-}
-
-// static
-void CXFA_FM2JSContext::logical_or_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() && argSecond->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float first = ValueToFloat(pThis, argFirst.get());
- float second = ValueToFloat(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger((first || second) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::logical_and_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() && argSecond->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- float first = ValueToFloat(pThis, argFirst.get());
- float second = ValueToFloat(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger((first && second) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::equality_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- if (fm_ref_equal(pThis, args)) {
- args.GetReturnValue()->SetInteger(1);
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() || argSecond->IsNull()) {
- args.GetReturnValue()->SetInteger(
- (argFirst->IsNull() && argSecond->IsNull()) ? 1 : 0);
- return;
- }
-
- if (argFirst->IsString() && argSecond->IsString()) {
- args.GetReturnValue()->SetInteger(argFirst->ToString() ==
- argSecond->ToString());
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger((first == second) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::notequality_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- if (fm_ref_equal(pThis, args)) {
- args.GetReturnValue()->SetInteger(0);
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() || argSecond->IsNull()) {
- args.GetReturnValue()->SetInteger(
- (argFirst->IsNull() && argSecond->IsNull()) ? 0 : 1);
- return;
- }
-
- if (argFirst->IsString() && argSecond->IsString()) {
- args.GetReturnValue()->SetInteger(argFirst->ToString() !=
- argSecond->ToString());
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger(first != second);
-}
-
-// static
-bool CXFA_FM2JSContext::fm_ref_equal(CFXJSE_Value* pThis,
- CFXJSE_Arguments& args) {
- std::unique_ptr<CFXJSE_Value> argFirst = args.GetValue(0);
- std::unique_ptr<CFXJSE_Value> argSecond = args.GetValue(1);
- if (!argFirst->IsArray() || !argSecond->IsArray())
- return false;
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- auto firstFlagValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto secondFlagValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argFirst->GetObjectPropertyByIdx(0, firstFlagValue.get());
- argSecond->GetObjectPropertyByIdx(0, secondFlagValue.get());
- if (firstFlagValue->ToInteger() != 3 || secondFlagValue->ToInteger() != 3)
- return false;
-
- auto firstJSObject = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto secondJSObject = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argFirst->GetObjectPropertyByIdx(2, firstJSObject.get());
- argSecond->GetObjectPropertyByIdx(2, secondJSObject.get());
- if (firstJSObject->IsNull() || secondJSObject->IsNull())
- return false;
-
- return (firstJSObject->ToHostObject(nullptr) ==
- secondJSObject->ToHostObject(nullptr));
-}
-
-// static
-void CXFA_FM2JSContext::less_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() || argSecond->IsNull()) {
- args.GetReturnValue()->SetInteger(0);
- return;
- }
-
- if (argFirst->IsString() && argSecond->IsString()) {
- args.GetReturnValue()->SetInteger(
- argFirst->ToString().Compare(argSecond->ToString().AsStringView()) ==
- -1);
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger((first < second) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::lessequal_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() || argSecond->IsNull()) {
- args.GetReturnValue()->SetInteger(
- (argFirst->IsNull() && argSecond->IsNull()) ? 1 : 0);
- return;
- }
-
- if (argFirst->IsString() && argSecond->IsString()) {
- args.GetReturnValue()->SetInteger(
- argFirst->ToString().Compare(argSecond->ToString().AsStringView()) !=
- 1);
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger((first <= second) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::greater_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() || argSecond->IsNull()) {
- args.GetReturnValue()->SetInteger(0);
- return;
- }
-
- if (argFirst->IsString() && argSecond->IsString()) {
- args.GetReturnValue()->SetInteger(
- argFirst->ToString().Compare(argSecond->ToString().AsStringView()) ==
- 1);
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger((first > second) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::greaterequal_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() || argSecond->IsNull()) {
- args.GetReturnValue()->SetInteger(
- (argFirst->IsNull() && argSecond->IsNull()) ? 1 : 0);
- return;
- }
-
- if (argFirst->IsString() && argSecond->IsString()) {
- args.GetReturnValue()->SetInteger(
- argFirst->ToString().Compare(argSecond->ToString().AsStringView()) !=
- -1);
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetInteger((first >= second) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::plus_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = args.GetValue(0);
- std::unique_ptr<CFXJSE_Value> argSecond = args.GetValue(1);
- if (ValueIsNull(pThis, argFirst.get()) &&
- ValueIsNull(pThis, argSecond.get())) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetDouble(first + second);
-}
-
-// static
-void CXFA_FM2JSContext::minus_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() && argSecond->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetDouble(first - second);
-}
-
-// static
-void CXFA_FM2JSContext::multiple_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 2) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() && argSecond->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- double second = ValueToDouble(pThis, argSecond.get());
- args.GetReturnValue()->SetDouble(first * second);
-}
-
-// static
-void CXFA_FM2JSContext::divide_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 2) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argFirst = GetSimpleValue(pThis, args, 0);
- std::unique_ptr<CFXJSE_Value> argSecond = GetSimpleValue(pThis, args, 1);
- if (argFirst->IsNull() && argSecond->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double second = ValueToDouble(pThis, argSecond.get());
- if (second == 0.0) {
- pContext->ThrowDivideByZeroException();
- return;
- }
-
- double first = ValueToDouble(pThis, argFirst.get());
- args.GetReturnValue()->SetDouble(first / second);
-}
-
-// static
-void CXFA_FM2JSContext::positive_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (argOne->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- args.GetReturnValue()->SetDouble(0.0 + ValueToDouble(pThis, argOne.get()));
-}
-
-// static
-void CXFA_FM2JSContext::negative_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (argOne->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
- args.GetReturnValue()->SetDouble(0.0 - ValueToDouble(pThis, argOne.get()));
-}
-
-// static
-void CXFA_FM2JSContext::logical_not_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- if (argOne->IsNull()) {
- args.GetReturnValue()->SetNull();
- return;
- }
-
- double first = ValueToDouble(pThis, argOne.get());
- args.GetReturnValue()->SetInteger((first == 0.0) ? 1 : 0);
-}
-
-// static
-void CXFA_FM2JSContext::dot_accessor(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- int32_t argc = args.GetLength();
- if (argc < 4 || argc > 5) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- bool bIsStar = true;
- int32_t iIndexValue = 0;
- if (argc > 4) {
- bIsStar = false;
- iIndexValue = ValueToInteger(pThis, args.GetValue(4).get());
- }
-
- ByteString szName = args.GetUTF8String(2);
- ByteString szSomExp = GenerateSomExpression(
- szName.AsStringView(), args.GetInt32(3), iIndexValue, bIsStar);
-
- std::unique_ptr<CFXJSE_Value> argAccessor = args.GetValue(0);
- if (argAccessor->IsArray()) {
- auto pLengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argAccessor->GetObjectProperty("length", pLengthValue.get());
- int32_t iLength = pLengthValue->ToInteger();
- if (iLength < 3) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- auto hJSObjValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- std::vector<std::vector<std::unique_ptr<CFXJSE_Value>>> resolveValues(
- iLength - 2);
- bool bAttribute = false;
- int32_t iCounter = 0;
- for (int32_t i = 2; i < iLength; i++) {
- argAccessor->GetObjectPropertyByIdx(i, hJSObjValue.get());
-
- XFA_RESOLVENODE_RS resoveNodeRS;
- if (ResolveObjects(pThis, hJSObjValue.get(), szSomExp.AsStringView(),
- resoveNodeRS, true, szName.IsEmpty()) > 0) {
- ParseResolveResult(pThis, resoveNodeRS, hJSObjValue.get(),
- &resolveValues[i - 2], &bAttribute);
- iCounter += resolveValues[i - 2].size();
- }
- }
- if (iCounter < 1) {
- pContext->ThrowPropertyNotInObjectException(
- WideString::FromUTF8(szName.AsStringView()),
- WideString::FromUTF8(szSomExp.AsStringView()));
- return;
- }
-
- std::vector<std::unique_ptr<CFXJSE_Value>> values;
- for (int32_t i = 0; i < iCounter + 2; i++)
- values.push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- values[0]->SetInteger(1);
- if (bAttribute)
- values[1]->SetString(szName.AsStringView());
- else
- values[1]->SetNull();
-
- int32_t iIndex = 2;
- for (int32_t i = 0; i < iLength - 2; i++) {
- for (size_t j = 0; j < resolveValues[i].size(); j++) {
- values[iIndex]->Assign(resolveValues[i][j].get());
- iIndex++;
- }
- }
- args.GetReturnValue()->SetArray(values);
- return;
- }
-
- XFA_RESOLVENODE_RS resoveNodeRS;
- int32_t iRet = 0;
- ByteString bsAccessorName = args.GetUTF8String(1);
- if (argAccessor->IsObject() ||
- (argAccessor->IsNull() && bsAccessorName.IsEmpty())) {
- iRet = ResolveObjects(pThis, argAccessor.get(), szSomExp.AsStringView(),
- resoveNodeRS, true, szName.IsEmpty());
- } else if (!argAccessor->IsObject() && !bsAccessorName.IsEmpty() &&
- GetObjectForName(pThis, argAccessor.get(),
- bsAccessorName.AsStringView())) {
- iRet = ResolveObjects(pThis, argAccessor.get(), szSomExp.AsStringView(),
- resoveNodeRS, true, szName.IsEmpty());
- }
- if (iRet < 1) {
- pContext->ThrowPropertyNotInObjectException(
- WideString::FromUTF8(szName.AsStringView()),
- WideString::FromUTF8(szSomExp.AsStringView()));
- return;
- }
-
- std::vector<std::unique_ptr<CFXJSE_Value>> resolveValues;
- bool bAttribute = false;
- ParseResolveResult(pThis, resoveNodeRS, argAccessor.get(), &resolveValues,
- &bAttribute);
-
- std::vector<std::unique_ptr<CFXJSE_Value>> values;
- for (size_t i = 0; i < resolveValues.size() + 2; i++)
- values.push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- values[0]->SetInteger(1);
- if (bAttribute)
- values[1]->SetString(szName.AsStringView());
- else
- values[1]->SetNull();
-
- for (size_t i = 0; i < resolveValues.size(); i++)
- values[i + 2]->Assign(resolveValues[i].get());
-
- args.GetReturnValue()->SetArray(values);
-}
-
-// static
-void CXFA_FM2JSContext::dotdot_accessor(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- int32_t argc = args.GetLength();
- if (argc < 4 || argc > 5) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- bool bIsStar = true;
- int32_t iIndexValue = 0;
- if (argc > 4) {
- bIsStar = false;
- iIndexValue = ValueToInteger(pThis, args.GetValue(4).get());
- }
-
- ByteString szName = args.GetUTF8String(2);
- ByteString szSomExp = GenerateSomExpression(
- szName.AsStringView(), args.GetInt32(3), iIndexValue, bIsStar);
-
- std::unique_ptr<CFXJSE_Value> argAccessor = args.GetValue(0);
- if (argAccessor->IsArray()) {
- auto pLengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argAccessor->GetObjectProperty("length", pLengthValue.get());
- int32_t iLength = pLengthValue->ToInteger();
- if (iLength < 3) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- int32_t iCounter = 0;
-
- std::vector<std::vector<std::unique_ptr<CFXJSE_Value>>> resolveValues(
- iLength - 2);
- auto hJSObjValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- bool bAttribute = false;
- for (int32_t i = 2; i < iLength; i++) {
- argAccessor->GetObjectPropertyByIdx(i, hJSObjValue.get());
- XFA_RESOLVENODE_RS resoveNodeRS;
- if (ResolveObjects(pThis, hJSObjValue.get(), szSomExp.AsStringView(),
- resoveNodeRS, false) > 0) {
- ParseResolveResult(pThis, resoveNodeRS, hJSObjValue.get(),
- &resolveValues[i - 2], &bAttribute);
- iCounter += resolveValues[i - 2].size();
- }
- }
- if (iCounter < 1) {
- pContext->ThrowPropertyNotInObjectException(
- WideString::FromUTF8(szName.AsStringView()),
- WideString::FromUTF8(szSomExp.AsStringView()));
- return;
- }
-
- std::vector<std::unique_ptr<CFXJSE_Value>> values;
- for (int32_t i = 0; i < iCounter + 2; i++)
- values.push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- values[0]->SetInteger(1);
- if (bAttribute)
- values[1]->SetString(szName.AsStringView());
- else
- values[1]->SetNull();
-
- int32_t iIndex = 2;
- for (int32_t i = 0; i < iLength - 2; i++) {
- for (size_t j = 0; j < resolveValues[i].size(); j++) {
- values[iIndex]->Assign(resolveValues[i][j].get());
- iIndex++;
- }
- }
- args.GetReturnValue()->SetArray(values);
- return;
- }
-
- XFA_RESOLVENODE_RS resoveNodeRS;
- int32_t iRet = 0;
- ByteString bsAccessorName = args.GetUTF8String(1);
- if (argAccessor->IsObject() ||
- (argAccessor->IsNull() && bsAccessorName.IsEmpty())) {
- iRet = ResolveObjects(pThis, argAccessor.get(), szSomExp.AsStringView(),
- resoveNodeRS, false);
- } else if (!argAccessor->IsObject() && !bsAccessorName.IsEmpty() &&
- GetObjectForName(pThis, argAccessor.get(),
- bsAccessorName.AsStringView())) {
- iRet = ResolveObjects(pThis, argAccessor.get(), szSomExp.AsStringView(),
- resoveNodeRS, false);
- }
- if (iRet < 1) {
- pContext->ThrowPropertyNotInObjectException(
- WideString::FromUTF8(szName.AsStringView()),
- WideString::FromUTF8(szSomExp.AsStringView()));
- return;
- }
-
- std::vector<std::unique_ptr<CFXJSE_Value>> resolveValues;
- bool bAttribute = false;
- ParseResolveResult(pThis, resoveNodeRS, argAccessor.get(), &resolveValues,
- &bAttribute);
-
- std::vector<std::unique_ptr<CFXJSE_Value>> values;
- for (size_t i = 0; i < resolveValues.size() + 2; i++)
- values.push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- values[0]->SetInteger(1);
- if (bAttribute)
- values[1]->SetString(szName.AsStringView());
- else
- values[1]->SetNull();
-
- for (size_t i = 0; i < resolveValues.size(); i++)
- values[i + 2]->Assign(resolveValues[i].get());
-
- args.GetReturnValue()->SetArray(values);
-}
-
-// static
-void CXFA_FM2JSContext::eval_translation(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 1) {
- pContext->ThrowParamCountMismatchException(L"Eval");
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = GetSimpleValue(pThis, args, 0);
- ByteString argString = ValueToUTF8String(argOne.get());
- if (argString.IsEmpty()) {
- pContext->ThrowArgumentMismatchException();
- return;
- }
-
- WideString scriptString = WideString::FromUTF8(argString.AsStringView());
- CFX_WideTextBuf wsJavaScriptBuf;
- if (!CXFA_FM2JSContext::Translate(scriptString.AsStringView(),
- &wsJavaScriptBuf)) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- args.GetReturnValue()->SetString(
- FX_UTF8Encode(wsJavaScriptBuf.AsStringView()).AsStringView());
-}
-
-// static
-void CXFA_FM2JSContext::is_fm_object(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- args.GetReturnValue()->SetBoolean(false);
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- args.GetReturnValue()->SetBoolean(argOne->IsObject());
-}
-
-// static
-void CXFA_FM2JSContext::is_fm_array(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- args.GetReturnValue()->SetBoolean(false);
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- args.GetReturnValue()->SetBoolean(argOne->IsArray());
-}
-
-// static
-void CXFA_FM2JSContext::get_fm_value(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 1) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- if (argOne->IsArray()) {
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argOne->GetObjectPropertyByIdx(1, propertyValue.get());
- argOne->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- GetObjectDefaultValue(jsObjectValue.get(), args.GetReturnValue());
- return;
- }
-
- jsObjectValue->GetObjectProperty(propertyValue->ToString().AsStringView(),
- args.GetReturnValue());
- return;
- }
-
- if (argOne->IsObject()) {
- GetObjectDefaultValue(argOne.get(), args.GetReturnValue());
- return;
- }
-
- args.GetReturnValue()->Assign(argOne.get());
-}
-
-// static
-void CXFA_FM2JSContext::get_fm_jsobj(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- if (args.GetLength() != 1) {
- ToJSContext(pThis, nullptr)->ThrowCompilerErrorException();
- return;
- }
-
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- if (!argOne->IsArray()) {
- args.GetReturnValue()->Assign(argOne.get());
- return;
- }
-
-#ifndef NDEBUG
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argOne->GetObjectProperty("length", lengthValue.get());
- ASSERT(lengthValue->ToInteger() >= 3);
-#endif
-
- argOne->GetObjectPropertyByIdx(2, args.GetReturnValue());
-}
-
-// static
-void CXFA_FM2JSContext::fm_var_filter(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- if (args.GetLength() != 1) {
- pContext->ThrowCompilerErrorException();
- return;
- }
-
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
- std::unique_ptr<CFXJSE_Value> argOne = args.GetValue(0);
- if (!argOne->IsArray()) {
- std::unique_ptr<CFXJSE_Value> simpleValue = GetSimpleValue(pThis, args, 0);
- args.GetReturnValue()->Assign(simpleValue.get());
- return;
- }
-
-#ifndef NDEBUG
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argOne->GetObjectProperty("length", lengthValue.get());
- ASSERT(lengthValue->ToInteger() >= 3);
-#endif
-
- auto flagsValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argOne->GetObjectPropertyByIdx(0, flagsValue.get());
- int32_t iFlags = flagsValue->ToInteger();
- if (iFlags != 3 && iFlags != 4) {
- std::unique_ptr<CFXJSE_Value> simpleValue = GetSimpleValue(pThis, args, 0);
- args.GetReturnValue()->Assign(simpleValue.get());
- return;
- }
-
- if (iFlags == 4) {
- std::vector<std::unique_ptr<CFXJSE_Value>> values;
- for (int32_t i = 0; i < 3; i++)
- values.push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- values[0]->SetInteger(3);
- values[1]->SetNull();
- values[2]->SetNull();
- args.GetReturnValue()->SetArray(values);
- return;
- }
-
- auto objectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argOne->GetObjectPropertyByIdx(2, objectValue.get());
- if (objectValue->IsNull()) {
- pContext->ThrowCompilerErrorException();
- return;
- }
- args.GetReturnValue()->Assign(argOne.get());
-}
-
-// static
-void CXFA_FM2JSContext::concat_fm_object(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args) {
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- uint32_t iLength = 0;
- int32_t argc = args.GetLength();
- std::vector<std::unique_ptr<CFXJSE_Value>> argValues;
- for (int32_t i = 0; i < argc; i++) {
- argValues.push_back(args.GetValue(i));
- if (argValues[i]->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValues[i]->GetObjectProperty("length", lengthValue.get());
- int32_t length = lengthValue->ToInteger();
- iLength = iLength + ((length > 2) ? (length - 2) : 0);
- }
- iLength += 1;
- }
-
- std::vector<std::unique_ptr<CFXJSE_Value>> returnValues;
- for (int32_t i = 0; i < (int32_t)iLength; i++)
- returnValues.push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- int32_t index = 0;
- for (int32_t i = 0; i < argc; i++) {
- if (argValues[i]->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argValues[i]->GetObjectProperty("length", lengthValue.get());
-
- int32_t length = lengthValue->ToInteger();
- for (int32_t j = 2; j < length; j++) {
- argValues[i]->GetObjectPropertyByIdx(j, returnValues[index].get());
- index++;
- }
- }
- returnValues[index]->Assign(argValues[i].get());
- index++;
- }
- args.GetReturnValue()->SetArray(returnValues);
-}
-
-// static
-std::unique_ptr<CFXJSE_Value> CXFA_FM2JSContext::GetSimpleValue(
- CFXJSE_Value* pThis,
- CFXJSE_Arguments& args,
- uint32_t index) {
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- ASSERT(index < (uint32_t)args.GetLength());
-
- std::unique_ptr<CFXJSE_Value> argIndex = args.GetValue(index);
- if (!argIndex->IsArray() && !argIndex->IsObject())
- return argIndex;
-
- if (argIndex->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argIndex->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- auto simpleValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- if (iLength < 3) {
- simpleValue.get()->SetUndefined();
- return simpleValue;
- }
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argIndex->GetObjectPropertyByIdx(1, propertyValue.get());
- argIndex->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- GetObjectDefaultValue(jsObjectValue.get(), simpleValue.get());
- return simpleValue;
- }
-
- jsObjectValue->GetObjectProperty(propertyValue->ToString().AsStringView(),
- simpleValue.get());
- return simpleValue;
- }
-
- auto defaultValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(argIndex.get(), defaultValue.get());
- return defaultValue;
-}
-
-// static
-bool CXFA_FM2JSContext::ValueIsNull(CFXJSE_Value* pThis, CFXJSE_Value* arg) {
- if (!arg || arg->IsNull())
- return true;
-
- if (!arg->IsArray() && !arg->IsObject())
- return false;
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- if (arg->IsArray()) {
- int32_t iLength = hvalue_get_array_length(pThis, arg);
- if (iLength < 3)
- return true;
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- arg->GetObjectPropertyByIdx(1, propertyValue.get());
- arg->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- auto defaultValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(jsObjectValue.get(), defaultValue.get());
- return defaultValue->IsNull();
- }
-
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- jsObjectValue->GetObjectProperty(propertyValue->ToString().AsStringView(),
- newPropertyValue.get());
- return newPropertyValue->IsNull();
- }
-
- auto defaultValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(arg, defaultValue.get());
- return defaultValue->IsNull();
-}
-
-// static
-int32_t CXFA_FM2JSContext::hvalue_get_array_length(CFXJSE_Value* pThis,
- CFXJSE_Value* arg) {
- if (!arg || !arg->IsArray())
- return 0;
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- arg->GetObjectProperty("length", lengthValue.get());
- return lengthValue->ToInteger();
-}
-
-// static
-bool CXFA_FM2JSContext::simpleValueCompare(CFXJSE_Value* pThis,
- CFXJSE_Value* firstValue,
- CFXJSE_Value* secondValue) {
- if (!firstValue)
- return false;
-
- if (firstValue->IsString()) {
- ByteString firstString = ValueToUTF8String(firstValue);
- ByteString secondString = ValueToUTF8String(secondValue);
- return firstString == secondString;
- }
- if (firstValue->IsNumber()) {
- float first = ValueToFloat(pThis, firstValue);
- float second = ValueToFloat(pThis, secondValue);
- return first == second;
- }
- if (firstValue->IsBoolean())
- return firstValue->ToBoolean() == secondValue->ToBoolean();
-
- return firstValue->IsNull() && secondValue && secondValue->IsNull();
-}
-
-// static
-void CXFA_FM2JSContext::unfoldArgs(
- CFXJSE_Value* pThis,
- CFXJSE_Arguments& args,
- std::vector<std::unique_ptr<CFXJSE_Value>>* resultValues,
- int32_t iStart) {
- resultValues->clear();
-
- int32_t iCount = 0;
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- int32_t argc = args.GetLength();
- std::vector<std::unique_ptr<CFXJSE_Value>> argsValue;
- for (int32_t i = 0; i < argc - iStart; i++) {
- argsValue.push_back(args.GetValue(i + iStart));
- if (argsValue[i]->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argsValue[i]->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- iCount += ((iLength > 2) ? (iLength - 2) : 0);
- } else {
- iCount += 1;
- }
- }
-
- for (int32_t i = 0; i < iCount; i++)
- resultValues->push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
-
- int32_t index = 0;
- for (int32_t i = 0; i < argc - iStart; i++) {
- if (argsValue[i]->IsArray()) {
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argsValue[i]->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- if (iLength < 3)
- continue;
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- argsValue[i]->GetObjectPropertyByIdx(1, propertyValue.get());
- if (propertyValue->IsNull()) {
- for (int32_t j = 2; j < iLength; j++) {
- argsValue[i]->GetObjectPropertyByIdx(j, jsObjectValue.get());
- GetObjectDefaultValue(jsObjectValue.get(),
- (*resultValues)[index].get());
- index++;
- }
- } else {
- for (int32_t j = 2; j < iLength; j++) {
- argsValue[i]->GetObjectPropertyByIdx(j, jsObjectValue.get());
- jsObjectValue->GetObjectProperty(
- propertyValue->ToString().AsStringView(),
- (*resultValues)[index].get());
- index++;
- }
- }
- } else if (argsValue[i]->IsObject()) {
- GetObjectDefaultValue(argsValue[i].get(), (*resultValues)[index].get());
- index++;
- } else {
- (*resultValues)[index]->Assign(argsValue[i].get());
- index++;
- }
- }
-}
-
-// static
-void CXFA_FM2JSContext::GetObjectDefaultValue(CFXJSE_Value* pValue,
- CFXJSE_Value* pDefaultValue) {
- CXFA_Node* pNode = ToNode(CXFA_ScriptContext::ToObject(pValue, nullptr));
- if (!pNode) {
- pDefaultValue->SetNull();
- return;
- }
- pNode->Script_Som_DefaultValue(pDefaultValue, false, (XFA_ATTRIBUTE)-1);
-}
-
-// static
-bool CXFA_FM2JSContext::SetObjectDefaultValue(CFXJSE_Value* pValue,
- CFXJSE_Value* hNewValue) {
- CXFA_Node* pNode = ToNode(CXFA_ScriptContext::ToObject(pValue, nullptr));
- if (!pNode)
- return false;
-
- pNode->Script_Som_DefaultValue(hNewValue, true, (XFA_ATTRIBUTE)-1);
- return true;
-}
-
-// static
-ByteString CXFA_FM2JSContext::GenerateSomExpression(
- const ByteStringView& szName,
- int32_t iIndexFlags,
- int32_t iIndexValue,
- bool bIsStar) {
- if (bIsStar)
- return ByteString(szName, "[*]");
-
- if (iIndexFlags == 0)
- return ByteString(szName);
-
- if (iIndexFlags == 1 || iIndexValue == 0) {
- return ByteString(szName, "[") + ByteString::FormatInteger(iIndexValue) +
- "]";
- }
- ByteString szSomExp;
- if (iIndexFlags == 2) {
- szSomExp = (iIndexValue < 0) ? (szName + "[-") : (szName + "[+");
- iIndexValue = (iIndexValue < 0) ? (0 - iIndexValue) : iIndexValue;
- szSomExp += ByteString::FormatInteger(iIndexValue);
- szSomExp += "]";
- } else {
- szSomExp = (iIndexValue < 0) ? (szName + "[") : (szName + "[-");
- iIndexValue = (iIndexValue < 0) ? (0 - iIndexValue) : iIndexValue;
- szSomExp += ByteString::FormatInteger(iIndexValue);
- szSomExp += "]";
- }
- return szSomExp;
-}
-
-// static
-bool CXFA_FM2JSContext::GetObjectForName(CFXJSE_Value* pThis,
- CFXJSE_Value* accessorValue,
- const ByteStringView& szAccessorName) {
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- if (!pDoc)
- return false;
-
- CXFA_ScriptContext* pScriptContext = pDoc->GetScriptContext();
- XFA_RESOLVENODE_RS resoveNodeRS;
- uint32_t dwFlags = XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Properties |
- XFA_RESOLVENODE_Siblings | XFA_RESOLVENODE_Parent;
- int32_t iRet = pScriptContext->ResolveObjects(
- pScriptContext->GetThisObject(),
- WideString::FromUTF8(szAccessorName).AsStringView(), resoveNodeRS,
- dwFlags);
- if (iRet >= 1 && resoveNodeRS.dwFlags == XFA_RESOVENODE_RSTYPE_Nodes) {
- accessorValue->Assign(
- pScriptContext->GetJSValueFromMap(resoveNodeRS.objects.front()));
- return true;
- }
- return false;
-}
-
-// static
-int32_t CXFA_FM2JSContext::ResolveObjects(CFXJSE_Value* pThis,
- CFXJSE_Value* pRefValue,
- const ByteStringView& bsSomExp,
- XFA_RESOLVENODE_RS& resoveNodeRS,
- bool bdotAccessor,
- bool bHasNoResolveName) {
- CXFA_Document* pDoc = ToJSContext(pThis, nullptr)->GetDocument();
- if (!pDoc)
- return -1;
-
- WideString wsSomExpression = WideString::FromUTF8(bsSomExp);
- CXFA_ScriptContext* pScriptContext = pDoc->GetScriptContext();
- CXFA_Object* pNode = nullptr;
- uint32_t dFlags = 0UL;
- if (bdotAccessor) {
- if (pRefValue && pRefValue->IsNull()) {
- pNode = pScriptContext->GetThisObject();
- dFlags = XFA_RESOLVENODE_Siblings | XFA_RESOLVENODE_Parent;
- } else {
- pNode = CXFA_ScriptContext::ToObject(pRefValue, nullptr);
- ASSERT(pNode);
- if (bHasNoResolveName) {
- WideString wsName;
- if (CXFA_Node* pXFANode = pNode->AsNode())
- pXFANode->GetAttribute(XFA_ATTRIBUTE_Name, wsName, false);
- if (wsName.IsEmpty())
- wsName = L"#" + pNode->GetClassName();
-
- wsSomExpression = wsName + wsSomExpression;
- dFlags = XFA_RESOLVENODE_Siblings;
- } else {
- dFlags = (bsSomExp == "*")
- ? (XFA_RESOLVENODE_Children)
- : (XFA_RESOLVENODE_Children | XFA_RESOLVENODE_Attributes |
- XFA_RESOLVENODE_Properties);
- }
- }
- } else {
- pNode = CXFA_ScriptContext::ToObject(pRefValue, nullptr);
- dFlags = XFA_RESOLVENODE_AnyChild;
- }
- return pScriptContext->ResolveObjects(pNode, wsSomExpression.AsStringView(),
- resoveNodeRS, dFlags);
-}
-
-// static
-void CXFA_FM2JSContext::ParseResolveResult(
- CFXJSE_Value* pThis,
- const XFA_RESOLVENODE_RS& resoveNodeRS,
- CFXJSE_Value* pParentValue,
- std::vector<std::unique_ptr<CFXJSE_Value>>* resultValues,
- bool* bAttribute) {
- ASSERT(bAttribute);
-
- resultValues->clear();
-
- CXFA_FM2JSContext* pContext = ToJSContext(pThis, nullptr);
- v8::Isolate* pIsolate = pContext->GetScriptRuntime();
-
- if (resoveNodeRS.dwFlags == XFA_RESOVENODE_RSTYPE_Nodes) {
- *bAttribute = false;
- CXFA_ScriptContext* pScriptContext =
- pContext->GetDocument()->GetScriptContext();
- for (CXFA_Object* pObject : resoveNodeRS.objects) {
- resultValues->push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
- resultValues->back()->Assign(pScriptContext->GetJSValueFromMap(pObject));
- }
- return;
- }
-
- CXFA_ValueArray objectProperties(pIsolate);
- int32_t iRet = resoveNodeRS.GetAttributeResult(&objectProperties);
- *bAttribute = true;
- if (iRet != 0) {
- *bAttribute = false;
- for (int32_t i = 0; i < iRet; i++) {
- resultValues->push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
- resultValues->back()->Assign(objectProperties.m_Values[i].get());
- }
- return;
- }
-
- if (!pParentValue || !pParentValue->IsObject())
- return;
-
- resultValues->push_back(pdfium::MakeUnique<CFXJSE_Value>(pIsolate));
- resultValues->back()->Assign(pParentValue);
-}
-
-// static
-int32_t CXFA_FM2JSContext::ValueToInteger(CFXJSE_Value* pThis,
- CFXJSE_Value* pValue) {
- if (!pValue)
- return 0;
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- if (pValue->IsArray()) {
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- pValue->GetObjectPropertyByIdx(1, propertyValue.get());
- pValue->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- return ValueToInteger(pThis, newPropertyValue.get());
- }
-
- jsObjectValue->GetObjectProperty(propertyValue->ToString().AsStringView(),
- newPropertyValue.get());
- return ValueToInteger(pThis, newPropertyValue.get());
- }
- if (pValue->IsObject()) {
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(pValue, newPropertyValue.get());
- return ValueToInteger(pThis, newPropertyValue.get());
- }
- if (pValue->IsString())
- return FXSYS_atoi(pValue->ToString().c_str());
- return pValue->ToInteger();
-}
-
-// static
-float CXFA_FM2JSContext::ValueToFloat(CFXJSE_Value* pThis, CFXJSE_Value* arg) {
- if (!arg)
- return 0.0f;
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- if (arg->IsArray()) {
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- arg->GetObjectPropertyByIdx(1, propertyValue.get());
- arg->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- return ValueToFloat(pThis, newPropertyValue.get());
- }
- jsObjectValue->GetObjectProperty(propertyValue->ToString().AsStringView(),
- newPropertyValue.get());
- return ValueToFloat(pThis, newPropertyValue.get());
- }
- if (arg->IsObject()) {
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(arg, newPropertyValue.get());
- return ValueToFloat(pThis, newPropertyValue.get());
- }
- if (arg->IsString())
- return (float)XFA_ByteStringToDouble(arg->ToString().AsStringView());
- if (arg->IsUndefined())
- return 0;
-
- return arg->ToFloat();
-}
-
-// static
-double CXFA_FM2JSContext::ValueToDouble(CFXJSE_Value* pThis,
- CFXJSE_Value* arg) {
- if (!arg)
- return 0;
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- if (arg->IsArray()) {
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- arg->GetObjectPropertyByIdx(1, propertyValue.get());
- arg->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull()) {
- GetObjectDefaultValue(jsObjectValue.get(), newPropertyValue.get());
- return ValueToDouble(pThis, newPropertyValue.get());
- }
- jsObjectValue->GetObjectProperty(propertyValue->ToString().AsStringView(),
- newPropertyValue.get());
- return ValueToDouble(pThis, newPropertyValue.get());
- }
- if (arg->IsObject()) {
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- GetObjectDefaultValue(arg, newPropertyValue.get());
- return ValueToDouble(pThis, newPropertyValue.get());
- }
- if (arg->IsString())
- return XFA_ByteStringToDouble(arg->ToString().AsStringView());
- if (arg->IsUndefined())
- return 0;
- return arg->ToDouble();
-}
-
-// static.
-double CXFA_FM2JSContext::ExtractDouble(CFXJSE_Value* pThis,
- CFXJSE_Value* src,
- bool* ret) {
- ASSERT(ret);
- *ret = true;
-
- if (!src)
- return 0;
-
- if (!src->IsArray())
- return ValueToDouble(pThis, src);
-
- v8::Isolate* pIsolate = ToJSContext(pThis, nullptr)->GetScriptRuntime();
- auto lengthValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- src->GetObjectProperty("length", lengthValue.get());
- int32_t iLength = lengthValue->ToInteger();
- if (iLength <= 2) {
- *ret = false;
- return 0.0;
- }
-
- auto propertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- auto jsObjectValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- src->GetObjectPropertyByIdx(1, propertyValue.get());
- src->GetObjectPropertyByIdx(2, jsObjectValue.get());
- if (propertyValue->IsNull())
- return ValueToDouble(pThis, jsObjectValue.get());
-
- auto newPropertyValue = pdfium::MakeUnique<CFXJSE_Value>(pIsolate);
- jsObjectValue->GetObjectProperty(propertyValue->ToString().AsStringView(),
- newPropertyValue.get());
- return ValueToDouble(pThis, newPropertyValue.get());
-}
-
-// static
-ByteString CXFA_FM2JSContext::ValueToUTF8String(CFXJSE_Value* arg) {
- if (!arg || arg->IsNull() || arg->IsUndefined())
- return ByteString();
- if (arg->IsBoolean())
- return arg->ToBoolean() ? "1" : "0";
- return arg->ToString();
-}
-
-// static.
-bool CXFA_FM2JSContext::Translate(const WideStringView& wsFormcalc,
- CFX_WideTextBuf* wsJavascript) {
- if (wsFormcalc.IsEmpty()) {
- wsJavascript->Clear();
- return true;
- }
-
- CXFA_FMParser parser(wsFormcalc);
- std::unique_ptr<CXFA_FMFunctionDefinition> func = parser.Parse();
- if (!func || parser.HasError())
- return false;
-
- CXFA_FMToJavaScriptDepth::Reset();
- if (!func->ToJavaScript(*wsJavascript))
- return false;
-
- wsJavascript->AppendChar(0);
-
- return !CXFA_IsTooBig(*wsJavascript);
-}
-
-CXFA_FM2JSContext::CXFA_FM2JSContext(v8::Isolate* pScriptIsolate,
- CFXJSE_Context* pScriptContext,
- CXFA_Document* pDoc)
- : CFXJSE_HostObject(kFM2JS),
- m_pIsolate(pScriptIsolate),
- m_pFMClass(CFXJSE_Class::Create(pScriptContext,
- &formcalc_fm2js_descriptor,
- false)),
- m_pValue(pdfium::MakeUnique<CFXJSE_Value>(pScriptIsolate)),
- m_pDocument(pDoc) {
- m_pValue.get()->SetObject(this, m_pFMClass);
-}
-
-CXFA_FM2JSContext::~CXFA_FM2JSContext() {}
-
-void CXFA_FM2JSContext::GlobalPropertyGetter(CFXJSE_Value* pValue) {
- pValue->Assign(m_pValue.get());
-}
-
-void CXFA_FM2JSContext::ThrowNoDefaultPropertyException(
- const ByteStringView& name) const {
- // TODO(tsepez): check usage of c_str() below.
- ThrowException(L"%.16S doesn't have a default property.",
- name.unterminated_c_str());
-}
-
-void CXFA_FM2JSContext::ThrowCompilerErrorException() const {
- ThrowException(L"Compiler error.");
-}
-
-void CXFA_FM2JSContext::ThrowDivideByZeroException() const {
- ThrowException(L"Divide by zero.");
-}
-
-void CXFA_FM2JSContext::ThrowServerDeniedException() const {
- ThrowException(L"Server does not permit operation.");
-}
-
-void CXFA_FM2JSContext::ThrowPropertyNotInObjectException(
- const WideString& name,
- const WideString& exp) const {
- ThrowException(
- L"An attempt was made to reference property '%.16s' of a non-object "
- L"in SOM expression %.16s.",
- name.c_str(), exp.c_str());
-}
-
-void CXFA_FM2JSContext::ThrowParamCountMismatchException(
- const WideString& method) const {
- ThrowException(L"Incorrect number of parameters calling method '%.16s'.",
- method.c_str());
-}
-
-void CXFA_FM2JSContext::ThrowArgumentMismatchException() const {
- ThrowException(L"Argument mismatch in property or function argument.");
-}
-
-void CXFA_FM2JSContext::ThrowException(const wchar_t* str, ...) const {
- WideString wsMessage;
- va_list arg_ptr;
- va_start(arg_ptr, str);
- wsMessage.FormatV(str, arg_ptr);
- va_end(arg_ptr);
- ASSERT(!wsMessage.IsEmpty());
- FXJSE_ThrowMessage(wsMessage.UTF8Encode().AsStringView());
-}
diff --git a/xfa/fxfa/fm2js/cxfa_fm2jscontext.h b/xfa/fxfa/fm2js/cxfa_fm2jscontext.h
deleted file mode 100644
index 8d1b33bf81..0000000000
--- a/xfa/fxfa/fm2js/cxfa_fm2jscontext.h
+++ /dev/null
@@ -1,445 +0,0 @@
-// 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
-
-#ifndef XFA_FXFA_FM2JS_CXFA_FM2JSCONTEXT_H_
-#define XFA_FXFA_FM2JS_CXFA_FM2JSCONTEXT_H_
-
-#include <memory>
-#include <vector>
-
-#include "fxjs/cfxjse_arguments.h"
-#include "fxjs/cfxjse_context.h"
-#include "xfa/fxfa/parser/xfa_resolvenode_rs.h"
-
-class CFX_WideTextBuf;
-class CXFA_Document;
-
-class CXFA_FM2JSContext : public CFXJSE_HostObject {
- public:
- CXFA_FM2JSContext(v8::Isolate* pScriptIsolate,
- CFXJSE_Context* pScriptContext,
- CXFA_Document* pDoc);
- ~CXFA_FM2JSContext() override;
-
- static void Abs(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Avg(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Ceil(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Count(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Floor(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Max(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Min(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Mod(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Round(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Sum(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Date(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Date2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void DateFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void IsoDate2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void IsoTime2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void LocalDateFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void LocalTimeFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Num2Date(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Num2GMTime(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Num2Time(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Time(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Time2Num(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void TimeFmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
-
- static bool IsIsoDateFormat(const char* pData,
- int32_t iLength,
- int32_t& iStyle,
- int32_t& iYear,
- int32_t& iMonth,
- int32_t& iDay);
- static bool IsIsoTimeFormat(const char* pData,
- int32_t iLength,
- int32_t& iHour,
- int32_t& iMinute,
- int32_t& iSecond,
- int32_t& iMilliSecond,
- int32_t& iZoneHour,
- int32_t& iZoneMinute);
- static bool IsIsoDateTimeFormat(const char* pData,
- int32_t iLength,
- int32_t& iYear,
- int32_t& iMonth,
- int32_t& iDay,
- int32_t& iHour,
- int32_t& iMinute,
- int32_t& iSecond,
- int32_t& iMillionSecond,
- int32_t& iZoneHour,
- int32_t& iZoneMinute);
- static ByteString Local2IsoDate(CFXJSE_Value* pThis,
- const ByteStringView& szDate,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale);
- static ByteString IsoDate2Local(CFXJSE_Value* pThis,
- const ByteStringView& szDate,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale);
- static ByteString IsoTime2Local(CFXJSE_Value* pThis,
- const ByteStringView& szTime,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale);
- static int32_t DateString2Num(const ByteStringView& szDateString);
- static ByteString GetLocalDateFormat(CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocalStr,
- bool bStandard);
- static ByteString GetLocalTimeFormat(CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocalStr,
- bool bStandard);
- static ByteString GetStandardDateFormat(CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocalStr);
- static ByteString GetStandardTimeFormat(CFXJSE_Value* pThis,
- int32_t iStyle,
- const ByteStringView& szLocalStr);
- static ByteString Num2AllTime(CFXJSE_Value* pThis,
- int32_t iTime,
- const ByteStringView& szFormat,
- const ByteStringView& szLocale,
- bool bGM);
- static void GetLocalTimeZone(int32_t& iHour, int32_t& iMin, int32_t& iSec);
-
- static void Apr(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void CTerm(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void FV(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void IPmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void NPV(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Pmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void PPmt(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void PV(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Rate(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Term(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Choose(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Exists(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void HasValue(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Oneof(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Within(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void If(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Eval(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Ref(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void UnitType(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void UnitValue(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
-
- static void At(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Concat(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Decode(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static WideString DecodeURL(const WideString& wsURLString);
- static WideString DecodeHTML(const WideString& wsHTMLString);
- static WideString DecodeXML(const WideString& wsXMLString);
- static void Encode(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static WideString EncodeURL(const ByteString& szURLString);
- static WideString EncodeHTML(const ByteString& szHTMLString);
- static WideString EncodeXML(const ByteString& szXMLString);
- static bool HTMLSTR2Code(const WideStringView& pData, uint32_t* iCode);
- static bool HTMLCode2STR(uint32_t iCode, WideString* wsHTMLReserve);
- static void Format(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Left(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Len(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Lower(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Ltrim(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Parse(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Replace(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Right(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Rtrim(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Space(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Str(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Stuff(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Substr(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Uuid(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Upper(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void WordNum(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static ByteString TrillionUS(const ByteStringView& szData);
- static ByteString WordUS(const ByteString& szData, int32_t iStyle);
-
- static void Get(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Post(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void Put(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void assign_value_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void logical_or_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void logical_and_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void equality_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void notequality_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static bool fm_ref_equal(CFXJSE_Value* pThis, CFXJSE_Arguments& args);
- static void less_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void lessequal_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void greater_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void greaterequal_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void plus_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void minus_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void multiple_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void divide_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void positive_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void negative_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void logical_not_operator(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void dot_accessor(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void dotdot_accessor(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void eval_translation(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void is_fm_object(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void is_fm_array(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void get_fm_value(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void get_fm_jsobj(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void fm_var_filter(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
- static void concat_fm_object(CFXJSE_Value* pThis,
- const ByteStringView& szFuncName,
- CFXJSE_Arguments& args);
-
- static int32_t hvalue_get_array_length(CFXJSE_Value* pThis,
- CFXJSE_Value* arg);
- static bool simpleValueCompare(CFXJSE_Value* pThis,
- CFXJSE_Value* firstValue,
- CFXJSE_Value* secondValue);
- static void unfoldArgs(
- CFXJSE_Value* pThis,
- CFXJSE_Arguments& args,
- std::vector<std::unique_ptr<CFXJSE_Value>>* resultValues,
- int32_t iStart = 0);
- static void GetObjectDefaultValue(CFXJSE_Value* pObjectValue,
- CFXJSE_Value* pDefaultValue);
- static bool SetObjectDefaultValue(CFXJSE_Value* pObjectValue,
- CFXJSE_Value* pNewValue);
- static ByteString GenerateSomExpression(const ByteStringView& szName,
- int32_t iIndexFlags,
- int32_t iIndexValue,
- bool bIsStar);
- static bool GetObjectForName(CFXJSE_Value* pThis,
- CFXJSE_Value* accessorValue,
- const ByteStringView& szAccessorName);
- static int32_t ResolveObjects(CFXJSE_Value* pThis,
- CFXJSE_Value* pParentValue,
- const ByteStringView& bsSomExp,
- XFA_RESOLVENODE_RS& resoveNodeRS,
- bool bdotAccessor = true,
- bool bHasNoResolveName = false);
- static void ParseResolveResult(
- CFXJSE_Value* pThis,
- const XFA_RESOLVENODE_RS& resoveNodeRS,
- CFXJSE_Value* pParentValue,
- std::vector<std::unique_ptr<CFXJSE_Value>>* resultValues,
- bool* bAttribute);
-
- static std::unique_ptr<CFXJSE_Value> GetSimpleValue(CFXJSE_Value* pThis,
- CFXJSE_Arguments& args,
- uint32_t index);
- static bool ValueIsNull(CFXJSE_Value* pThis, CFXJSE_Value* pValue);
- static int32_t ValueToInteger(CFXJSE_Value* pThis, CFXJSE_Value* pValue);
- static float ValueToFloat(CFXJSE_Value* pThis, CFXJSE_Value* pValue);
- static double ValueToDouble(CFXJSE_Value* pThis, CFXJSE_Value* pValue);
- static ByteString ValueToUTF8String(CFXJSE_Value* pValue);
- static double ExtractDouble(CFXJSE_Value* pThis,
- CFXJSE_Value* src,
- bool* ret);
-
- static bool Translate(const WideStringView& wsFormcalc,
- CFX_WideTextBuf* wsJavascript);
-
- void GlobalPropertyGetter(CFXJSE_Value* pValue);
-
- private:
- v8::Isolate* GetScriptRuntime() const { return m_pIsolate; }
- CXFA_Document* GetDocument() const { return m_pDocument.Get(); }
-
- void ThrowNoDefaultPropertyException(const ByteStringView& name) const;
- void ThrowCompilerErrorException() const;
- void ThrowDivideByZeroException() const;
- void ThrowServerDeniedException() const;
- void ThrowPropertyNotInObjectException(const WideString& name,
- const WideString& exp) const;
- void ThrowArgumentMismatchException() const;
- void ThrowParamCountMismatchException(const WideString& method) const;
- void ThrowException(const wchar_t* str, ...) const;
-
- v8::Isolate* m_pIsolate;
- CFXJSE_Class* m_pFMClass;
- std::unique_ptr<CFXJSE_Value> m_pValue;
- UnownedPtr<CXFA_Document> const m_pDocument;
-};
-
-#endif // XFA_FXFA_FM2JS_CXFA_FM2JSCONTEXT_H_
diff --git a/xfa/fxfa/fm2js/cxfa_fm2jscontext_embeddertest.cpp b/xfa/fxfa/fm2js/cxfa_fm2jscontext_embeddertest.cpp
deleted file mode 100644
index ff2e200186..0000000000
--- a/xfa/fxfa/fm2js/cxfa_fm2jscontext_embeddertest.cpp
+++ /dev/null
@@ -1,1446 +0,0 @@
-// Copyright 2017 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.
-
-#include "testing/gtest/include/gtest/gtest.h"
-#include "testing/xfa_js_embedder_test.h"
-
-class FM2JSContextEmbedderTest : public XFAJSEmbedderTest {};
-
-// TODO(dsinclair): Comment out tests are broken and need to be fixed.
-
-TEST_F(FM2JSContextEmbedderTest, TranslateEmpty) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- const char input[] = "";
- EXPECT_TRUE(Execute(input));
- // TODO(dsinclair): This should probably throw as a blank formcalc script
- // is invalid.
-}
-
-TEST_F(FM2JSContextEmbedderTest, TranslateNumber) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- const char input[] = "123";
- EXPECT_TRUE(Execute(input));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(123, value->ToInteger()) << "Program: " << input;
-}
-
-TEST_F(FM2JSContextEmbedderTest, Numeric) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"123 + 456", 579},
- {"2 - 3 * 10 / 2 + 7", -6},
- {"10 * 3 + 5 * 4", 50},
- {"(5 - \"abc\") * 3", 15},
- {"\"100\" / 10e1", 1},
- {"5 + null + 3", 8},
- // {"if (\"abc\") then\n"
- // " 10\n"
- // "else\n"
- // " 20\n"
- // "endif",
- // 20},
- // {"3 / 0 + 1", 0},
- {"-(17)", -17},
- {"-(-17)", 17},
- {"+(17)", 17},
- {"+(-17)", -17},
- {"if (1 < 2) then\n1\nendif", 1},
- {"if (\"abc\" > \"def\") then\n"
- " 1 and 0\n"
- "else\n"
- " 0\n"
- "endif",
- 0}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Strings) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- {"\"abc\"", "abc"},
- {"concat(\"The total is \", 2, \" dollars and \", 57, \" cents.\")",
- "The total is 2 dollars and 57 cents."}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Booleans) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- bool result;
- } tests[] = {{"0 and 1 or 2 > 1", true},
- {"2 < 3 not 1 == 1", false},
- {"\"abc\" | 2", true},
- {"1 or 0", true},
- {"0 | 0", false},
- {"0 or 1 | 0 or 0", true},
- {"1 and 0", false},
- // {"0 & 0", true}, // TODO(dsinclair) Confirm with Reader.
- {"0 and 1 & 0 and 0", false},
- {"not(\"true\")", true},
- {"not(1)", false},
- {"3 == 3", true},
- {"3 <> 4", true},
- {"\"abc\" eq \"def\"", false},
- {"\"def\" ne \"abc\"", true},
- {"5 + 5 == 10", true},
- {"5 + 5 <> \"10\"", false},
- {"3 < 3", false},
- {"3 > 4", false},
- {"\"abc\" <= \"def\"", true},
- {"\"def\" > \"abc\"", true},
- {"12 >= 12", true},
- {"\"true\" < \"false\"", false}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger()) << "Program: " << tests[i].program;
- EXPECT_EQ(tests[i].result, value->ToBoolean())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Abs) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"Abs(1.03)", 1.03f}, {"Abs(-1.03)", 1.03f}, {"Abs(0)", 0.0f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Avg) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"Avg(0, 32, 16)", 16.0f}, {"Avg(2.5, 17, null)", 9.75f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Ceil) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"Ceil(2.5875)", 3}, {"Ceil(-5.9)", -5}, {"Ceil(\"abc\")", 0}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Count) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"Count(\"Tony\", \"Blue\", 41)", 3}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Floor) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"Floor(21.3409873)", 21},
- {"Floor(5.999965342)", 5},
- {"Floor(3.2 * 15)", 48}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Max) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"Max(234, 15, 107)", 234},
- {"Max(\"abc\", 15, \"Tony Blue\")", 15},
- {"Max(\"abc\")", 0}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Min) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"Min(234, 15, 107)", 15},
- // TODO(dsinclair): Verify with Reader; I believe this should
- // have a return of 0.
- // {"Min(\"abc\", 15, \"Tony Blue\")", 15},
- {"Min(\"abc\")", 0}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Mod) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"Mod(64, -3)", 1}, {"Mod(-13, 3)", -1}, {"Mod(\"abc\", 2)", 0}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Round) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"Round(12.389764537, 4)", 12.3898f},
- {"Round(20/3, 2)", 6.67f},
- {"Round(8.9897, \"abc\")", 9.0f},
- {"Round(FV(400, 0.10/12, 30*12), 2)", 904195.17f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber()) << "Program: " << tests[i].program;
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Sum) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"Sum(2, 4, 6, 8)", 20},
- {"Sum(-2, 4, -6, 8)", 4},
- {"Sum(4, 16, \"abc\", 19)", 39}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-// TEST_F(FM2JSContextEmbedderTest, DISABLED_Date) {
-// ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-//
-// TODO(dsinclair): Make compatible with windows.
-// time_t seconds = time(nullptr);
-// int days = seconds / (60 * 60 * 24);
-
-// EXPECT_TRUE(Execute("Date()"));
-
-// CFXJSE_Value* value = GetValue();
-// EXPECT_TRUE(value->IsNumber());
-// EXPECT_EQ(days, value->ToInteger());
-// }
-
-TEST_F(FM2JSContextEmbedderTest, Date2Num) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {
- // {"Date2Num(\"Mar 15, 1996\")", 35138},
- {"Date2Num(\"1/1/1900\", \"D/M/YYYY\")", 1},
- {"Date2Num(\"03/15/96\", \"MM/DD/YY\")", 35138},
- // {"Date2Num(\"Aug 1, 1996\", \"MMM D, YYYY\")", 35277},
- {"Date2Num(\"96-08-20\", \"YY-MM-DD\", \"fr_FR\")", 35296},
- {"Date2Num(\"1/3/00\", \"D/M/YY\") - Date2Num(\"1/2/00\", \"D/M/YY\")",
- 29}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DateFmt) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- // {"DateFmt(1)", "M/D/YY"},
- // {"DateFmt(2, \"fr_CA\")", "YY-MM-DD"},
- {"DateFmt(3, \"de_DE\")", "D. MMMM YYYY"},
- // {"DateFmt(4, \"fr_FR\")", "EEE D' MMMM YYYY"}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, IsoDate2Num) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"IsoDate2Num(\"1900\")", 1},
- {"IsoDate2Num(\"1900-01\")", 1},
- {"IsoDate2Num(\"1900-01-01\")", 1},
- {"IsoDate2Num(\"19960315T20:20:20\")", 35138},
- {"IsoDate2Num(\"2000-03-01\") - IsoDate2Num(\"20000201\")", 29}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_IsoTime2Num) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"IsoTime2Num(\"00:00:00Z\")", 1}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, LocalDateFmt) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {// {"LocalDateFmt(1, \"de_DE\")", "tt.MM.uu"},
- // {"LocalDateFmt(2, \"fr_CA\")", "aa-MM-jj"},
- {"LocalDateFmt(3, \"de_CH\")", "t. MMMM jjjj"},
- {"LocalDateFmt(4, \"fr_FR\")", "EEEE j MMMM aaaa"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_LocalTimeFmt) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"LocalTimeFmt(1, \"de_DE\")", "HH:mm"},
- {"LocalTimeFmt(2, \"fr_CA\")", "HH:mm::ss"},
- {"LocalTimeFmt(3, \"de_CH\")", "HH:mm:ss z"},
- {"LocalTimeFmt(4, \"fr_FR\")", "HH' h 'mm z"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Num2Date) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- {"Num2Date(1, \"DD/MM/YYYY\")", "01/01/1900"},
- {"Num2Date(35139, \"DD-MMM-YYYY\", \"de_DE\")", "16-Mrz-1996"},
- // {"Num2Date(Date2Num(\"Mar 15, 2000\") - Date2Num(\"98-03-15\", "
- // "\"YY-MM-DD\", \"fr_CA\"))",
- // "Jan 1, 1902"}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString()) << "Program: " << tests[i].program;
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Num2GMTime) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {// Broken on Windows only.
- {"Num2GMTime(1, \"HH:MM:SS\")", "00:00:00"},
- // Below broken on other platforms.
- {"Num2GMTime(65593001, \"HH:MM:SS Z\")", "18:13:13 GMT"},
- {"Num2GMTime(43993001, TimeFmt(4, \"de_DE\"), \"de_DE\")",
- "12.13 Uhr GMT"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-// TODO(dsinclair): Broken on Mac ...
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Num2Time) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Num2Time(1, \"HH:MM:SS\")", "00:00:00"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-// TEST_F(FM2JSContextEmbedderTest, DISABLED_Time) {
-// ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-// TODO(dsinclair): Make compatible with windows.
-// struct timeval tp;
-// gettimeofday(&tp, nullptr);
-
-// EXPECT_TRUE(Execute("Time()"));
-
-// CFXJSE_Value* value = GetValue();
-// EXPECT_TRUE(value->IsInteger());
-// EXPECT_EQ(tp.tv_sec * 1000L + tp.tv_usec / 1000, value->ToInteger())
-// << "Program: Time()";
-// }
-
-TEST_F(FM2JSContextEmbedderTest, Time2Num) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {
- // {"Time2Num(\"00:00:00 GMT\", \"HH:MM:SS Z\")", 1},
- {"Time2Num(\"13:13:13 GMT\", \"HH:MM:SS Z\", \"fr_FR\")", 47593001}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, TimeFmt) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- // {"TimeFmt(1)", "h::MM A"},
- {"TimeFmt(2, \"fr_CA\")", "HH:MM:SS"},
- {"TimeFmt(3, \"fr_FR\")", "HH:MM:SS Z"},
- // {"TimeFmt(4, \"de_DE\")", "H.MM' Uhr 'Z"}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Apr) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"Apr(35000, 269.50, 360)", 0.08515404566f},
- {"Apr(210000 * 0.75, 850 + 110, 25 * 26)", 0.07161332404f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, CTerm) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {
- // {"CTerm(0.02, 1000, 100)", 116.2767474515f},
- {"CTerm(0.10, 500000, 12000)", 39.13224648502f},
- // {"CTerm(0.0275 + 0.0025, 1000000, 55000 * 0.10)", 176.02226044975f}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, FV) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"FV(400, 0.10 / 12, 30 * 12)", 904195.16991842445f},
- {"FV(1000, 0.075 / 4, 10 * 4)", 58791.96145535981f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, IPmt) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"IPmt(30000, 0.085, 295.50, 7, 3)", 624.8839283142f},
- {"IPmt(160000, 0.0475, 980, 24, 12)", 7103.80833569485f},
- {"IPmt(15000, 0.065, 65.50, 15, 1)", 0.0f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_NPV) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"NPV(0.065, 5000)", 4694.83568075117f},
- {"NPV(0.10, 500, 1500, 4000, 10000)", 11529.60863329007f},
- {"NPV(0.0275 / 12, 50, 60, 40, 100, 25)", 273.14193838457f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber()) << "Program: " << tests[i].program;
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Pmt) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {// {"Pmt(150000, 0.0475 / 12, 25 * 12)", 855.17604207164f},
- {"Pmt(25000, 0.085, 12)", 3403.82145169876f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, PPmt) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {
- {"PPmt(30000, 0.085, 295.50, 7, 3)", 261.6160716858f},
- {"PPmt(160000, 0.0475, 980, 24, 12)", 4656.19166430515f},
- // {"PPmt(15000, 0.065, 65.50, 15, 1)", 0.0f}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, PV) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {
- {"PV(400, 0.10 / 12, 30 * 12)", 45580.32799074439f},
- // {"PV(1000, 0.075 / 4, 10 * 4)", 58791.96145535981f}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Rate) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {{"Rate(12000, 8000, 5)", 0.0844717712f},
- {"Rate(10000, 0.25 * 5000, 4 * 12)", 0.04427378243f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Term) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {// {"Term(475, .05, 1500)", 3.00477517728f},
- {"Term(2500, 0.0275 + 0.0025, 5000)", 1.97128786369f}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Choose) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- {"Choose(3, \"Taxes\", \"Price\", \"Person\", \"Teller\")", "Person"},
- {"Choose(2, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)", "9"},
- {"Choose(20/3, \"A\", \"B\", \"C\", \"D\", \"E\", \"F\", \"G\", \"H\")",
- "F"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Exists) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- EXPECT_TRUE(Execute("Exists(\"hello world\")"));
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_FALSE(value->ToBoolean());
-}
-
-TEST_F(FM2JSContextEmbedderTest, HasValue) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- bool result;
- } tests[] = {{"HasValue(2)", true}, {"HasValue(\" \")", false}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger()) << "Program: " << tests[i].program;
- EXPECT_EQ(tests[i].result, value->ToBoolean())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Oneof) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- bool result;
- } tests[] = {
- {"Oneof(3, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)", true},
- {"Oneof(\"John\", \"Bill\", \"Gary\", \"Joan\", \"John\", \"Lisa\")",
- true},
- {"Oneof(3, 1, 25)", false},
- {"Oneof(3, 3, null)", true},
- {"Oneof(3, null, null)", false},
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger()) << "Program: " << tests[i].program;
- EXPECT_EQ(tests[i].result, value->ToBoolean())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Within) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- bool result;
- } tests[] = {{"Within(\"C\", \"A\", \"D\")", true},
- {"Within(1.5, 0, 2)", true},
- {"Within(-1, 0, 2)", false}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger()) << "Program: " << tests[i].program;
- EXPECT_EQ(tests[i].result, value->ToBoolean())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Eval) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"eval(\"10*3+5*4\")", 50}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Null) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Null()", "null"},
- {"Concat(\"ABC\", Null(), \"DEF\")", "ABCDEF"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-
- EXPECT_TRUE(Execute("Null() + 5"));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(5, value->ToInteger());
-}
-
-TEST_F(FM2JSContextEmbedderTest, Ref) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Ref(\"10*3+5*4\")", "10*3+5*4"}, {"Ref(\"hello\")", "hello"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, UnitType) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"UnitType(\"36 in\")", "in"},
- {"UnitType(\"2.54centimeters\")", "cm"},
- {"UnitType(\"picas\")", "pt"},
- {"UnitType(\"2.cm\")", "cm"},
- {"UnitType(\"2.zero cm\")", "in"},
- {"UnitType(\"kilometers\")", "in"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, UnitValue) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- float result;
- } tests[] = {
- {"UnitValue(\"2in\")", 2.0f}, {"UnitValue(\"2in\", \"cm\")", 5.08f},
- // {"UnitValue(\"6\", \"pt\")", 432f},
- // {"UnitType(\"A\", \"cm\")", 0.0f},
- // {"UnitType(\"5.08cm\", \"kilograms\")", 2.0f}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(tests[i].result, value->ToFloat())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, At) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {{"At(\"ABCDEFGH\", \"AB\")", 1},
- {"At(\"ABCDEFGH\", \"F\")", 6},
- {"At(23412931298471, 29)", 5}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Concat) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Concat(\"ABC\", \"DEF\")", "ABCDEF"},
- {"Concat(\"Tony\", Space(1), \"Blue\")", "Tony Blue"},
- {"Concat(\"You owe \", WordNum(1154.67, 2), \".\")",
- "You owe One Thousand One Hundred Fifty-four Dollars And "
- "Sixty-seven Cents."}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Decode) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- {"Decode(\"&AElig;&Aacute;&Acirc;&Aacute;&Acirc;\", \"html\")", "ÆÁÂÁÂ"},
- // {"Decode(\"~!@#$%%^&amp;*()_+|`{&quot;}[]&lt;&gt;?,./;&apos;:\", "
- // "\"xml\")",
- // "~!@#$%%^&*()_+|`{"
- // "}[]<>?,./;':"}
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Encode) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- {"Encode(\"\"\"hello, world!\"\"\", \"url\")",
- "%%22hello,%%20world!%%22"},
- {"Encode(\"ÁÂÃÄÅÆ\", \"html\")", "&#xc1;&#Xc2;&#Xc3;&#xc4;&#xc5;&#xc6;"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Format) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Format(\"MMM D, YYYY\", \"20020901\")", "Sep 1, 2002"},
- {"Format(\"$9,999,999.99\", 1234567.89)", "$1,234,567.89"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Left) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Left(\"ABCDEFGH\", 3)", "ABC"},
- {"Left(\"Tony Blue\", 5)", "Tony "}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Len) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- int result;
- } tests[] = {
- {"Len(\"ABCDEFGH\")", 8}, {"Len(4)", 1}, {"Len(Str(4.532, 6, 4))", 6}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsInteger());
- EXPECT_EQ(tests[i].result, value->ToInteger())
- << "Program: " << tests[i].program;
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Lower) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Lower(\"ABC\")", "abc"},
- {"Lower(\"21 Main St.\")", "21 main st."},
- {"Lower(15)", "15"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Ltrim) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Ltrim(\" ABCD\")", "ABCD"},
- {"Ltrim(Rtrim(\" Tony Blue \"))", "Tony Blue"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, DISABLED_Parse) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Parse(\"MMM D, YYYY\", \"Sep 1, 2002\")", "2002-09-01"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-
- EXPECT_TRUE(Execute("Parse(\"$9,999,999.99\", \"$1,234,567.89\")"));
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsNumber());
- EXPECT_FLOAT_EQ(1234567.89f, value->ToFloat());
-}
-
-TEST_F(FM2JSContextEmbedderTest, Replace) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Replace(\"Tony Blue\", \"Tony\", \"Chris\")", "Chris Blue"},
- {"Replace(\"ABCDEFGH\", \"D\")", "ABCEFGH"},
- {"Replace(\"ABCDEFGH\", \"d\")", "ABCDEFGH"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Right) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Right(\"ABCDEFGH\", 3)", "FGH"},
- {"Right(\"Tony Blue\", 5)", " Blue"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Rtrim) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Rtrim(\"ABCD \")", "ABCD"},
- {"Rtrim(\"Tony Blue \t\")", "Tony Blue"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Space) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Space(5)", " "},
- {"Concat(\"Tony\", Space(1), \"Blue\")", "Tony Blue"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Str) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Str(2.456)", " 2"},
- {"Str(4.532, 6, 4)", "4.5320"},
- {"Str(234.458, 4)", " 234"},
- {"Str(31.2345, 4, 2)", "****"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Stuff) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Stuff(\"TonyBlue\", 5, 0, \" \")", "Tony Blue"},
- {"Stuff(\"ABCDEFGH\", 4, 2)", "ABCFGH"},
- {"Stuff(\"members-list@myweb.com\", 0, 0, \"cc:\")",
- "cc:members-list@myweb.com"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Substr) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Substr(\"ABCDEFG\", 3, 4)", "CDEF"},
- {"Substr(3214, 2, 1)", "2"},
- {"Substr(\"ABCDEFG\", 5, 0)", ""},
- {"Substr(\"21 Waterloo St.\", 4, 5)", "Water"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Uuid) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- EXPECT_TRUE(Execute("Uuid()"));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
-}
-
-TEST_F(FM2JSContextEmbedderTest, Upper) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {{"Upper(\"abc\")", "ABC"},
- {"Upper(\"21 Main St.\")", "21 MAIN ST."},
- {"Upper(15)", "15"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, WordNum) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- struct {
- const char* program;
- const char* result;
- } tests[] = {
- // {"WordNum(123.45)",
- // "One Hundred and Twenty-three"}, // This looks like it's wrong in the
- // // Formcalc document.
- // {"WordNum(123.45, 1)", "One Hundred and Twenty-three Dollars"},
- {"WordNum(1154.67, 2)",
- "One Thousand One Hundred Fifty-four Dollars And Sixty-seven Cents"},
- {"WordNum(43, 2)", "Forty-three Dollars And Zero Cents"}};
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_TRUE(Execute(tests[i].program));
-
- CFXJSE_Value* value = GetValue();
- EXPECT_TRUE(value->IsString());
- EXPECT_STREQ(tests[i].result, value->ToString().c_str())
- << "Program: " << tests[i].program << " Result: '"
- << value->ToString().c_str() << "'";
- }
-}
-
-TEST_F(FM2JSContextEmbedderTest, Get) {
- // TODO(dsinclair): Is this supported?
-}
-
-TEST_F(FM2JSContextEmbedderTest, Post) {
- // TODO(dsinclair): Is this supported?
-}
-
-TEST_F(FM2JSContextEmbedderTest, Put) {
- // TODO(dsinclair): Is this supported?
-}
-
-TEST_F(FM2JSContextEmbedderTest, InvalidFunctions) {
- ASSERT_TRUE(OpenDocument("simple_xfa.pdf"));
-
- const char* const tests[] = {
- "F()", "()", "()()()", "Round(2.0)()",
- };
-
- for (size_t i = 0; i < FX_ArraySize(tests); ++i) {
- EXPECT_FALSE(ExecuteSilenceFailure(tests[i]));
- }
-}