1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6 
7 #include "fpdfsdk/javascript/PublicMethods.h"
8 
9 #include <algorithm>
10 #include <iomanip>
11 #include <limits>
12 #include <sstream>
13 #include <string>
14 #include <vector>
15 
16 #include "core/fpdfdoc/cpdf_interform.h"
17 #include "core/fxcrt/fx_ext.h"
18 #include "fpdfsdk/cpdfsdk_formfillenvironment.h"
19 #include "fpdfsdk/cpdfsdk_interform.h"
20 #include "fpdfsdk/javascript/Field.h"
21 #include "fpdfsdk/javascript/JS_Define.h"
22 #include "fpdfsdk/javascript/JS_EventHandler.h"
23 #include "fpdfsdk/javascript/JS_Object.h"
24 #include "fpdfsdk/javascript/JS_Value.h"
25 #include "fpdfsdk/javascript/cjs_event_context.h"
26 #include "fpdfsdk/javascript/cjs_runtime.h"
27 #include "fpdfsdk/javascript/color.h"
28 #include "fpdfsdk/javascript/resource.h"
29 #include "fpdfsdk/javascript/util.h"
30 
31 #define DOUBLE_CORRECT 0.000000000000001
32 
33 JSMethodSpec CJS_PublicMethods::GlobalFunctionSpecs[] = {
34     {"AFNumber_Format", AFNumber_Format_static},
35     {"AFNumber_Keystroke", AFNumber_Keystroke_static},
36     {"AFPercent_Format", AFPercent_Format_static},
37     {"AFPercent_Keystroke", AFPercent_Keystroke_static},
38     {"AFDate_FormatEx", AFDate_FormatEx_static},
39     {"AFDate_KeystrokeEx", AFDate_KeystrokeEx_static},
40     {"AFDate_Format", AFDate_Format_static},
41     {"AFDate_Keystroke", AFDate_Keystroke_static},
42     {"AFTime_FormatEx", AFTime_FormatEx_static},
43     {"AFTime_KeystrokeEx", AFTime_KeystrokeEx_static},
44     {"AFTime_Format", AFTime_Format_static},
45     {"AFTime_Keystroke", AFTime_Keystroke_static},
46     {"AFSpecial_Format", AFSpecial_Format_static},
47     {"AFSpecial_Keystroke", AFSpecial_Keystroke_static},
48     {"AFSpecial_KeystrokeEx", AFSpecial_KeystrokeEx_static},
49     {"AFSimple", AFSimple_static},
50     {"AFMakeNumber", AFMakeNumber_static},
51     {"AFSimple_Calculate", AFSimple_Calculate_static},
52     {"AFRange_Validate", AFRange_Validate_static},
53     {"AFMergeChange", AFMergeChange_static},
54     {"AFParseDateEx", AFParseDateEx_static},
55     {"AFExtractNums", AFExtractNums_static},
56     {0, 0}};
57 
58 IMPLEMENT_JS_STATIC_GLOBAL_FUN(CJS_PublicMethods)
59 
60 namespace {
61 
62 const FX_WCHAR* const months[] = {L"Jan", L"Feb", L"Mar", L"Apr",
63                                   L"May", L"Jun", L"Jul", L"Aug",
64                                   L"Sep", L"Oct", L"Nov", L"Dec"};
65 
66 const FX_WCHAR* const fullmonths[] = {L"January", L"February", L"March",
67                                       L"April",   L"May",      L"June",
68                                       L"July",    L"August",   L"September",
69                                       L"October", L"November", L"December"};
70 
StrTrim(const CFX_ByteString & pStr)71 CFX_ByteString StrTrim(const CFX_ByteString& pStr) {
72   CFX_ByteString result(pStr);
73   result.TrimLeft(' ');
74   result.TrimRight(' ');
75   return result;
76 }
77 
StrTrim(const CFX_WideString & pStr)78 CFX_WideString StrTrim(const CFX_WideString& pStr) {
79   CFX_WideString result(pStr);
80   result.TrimLeft(' ');
81   result.TrimRight(' ');
82   return result;
83 }
84 
AlertIfPossible(CJS_EventContext * pContext,const FX_WCHAR * swMsg)85 void AlertIfPossible(CJS_EventContext* pContext, const FX_WCHAR* swMsg) {
86   CPDFSDK_FormFillEnvironment* pFormFillEnv = pContext->GetFormFillEnv();
87   if (pFormFillEnv)
88     pFormFillEnv->JS_appAlert(swMsg, nullptr, 0, 3);
89 }
90 
91 #if _FX_OS_ != _FX_ANDROID_
CalculateString(double dValue,int iDec,int * iDec2,bool * bNegative)92 CFX_ByteString CalculateString(double dValue,
93                                int iDec,
94                                int* iDec2,
95                                bool* bNegative) {
96   *bNegative = dValue < 0;
97   if (*bNegative)
98     dValue = -dValue;
99 
100   // Make sure the number of precision characters will fit.
101   if (iDec > std::numeric_limits<double>::digits10)
102     iDec = std::numeric_limits<double>::digits10;
103 
104   std::stringstream ss;
105   ss << std::fixed << std::setprecision(iDec) << dValue;
106   std::string stringValue = ss.str();
107   size_t iDecimalPos = stringValue.find(".");
108   *iDec2 = iDecimalPos == std::string::npos ? stringValue.size()
109                                             : static_cast<int>(iDecimalPos);
110   return CFX_ByteString(stringValue.c_str());
111 }
112 #endif
113 
114 }  // namespace
115 
IsNumber(const CFX_WideString & str)116 bool CJS_PublicMethods::IsNumber(const CFX_WideString& str) {
117   CFX_WideString sTrim = StrTrim(str);
118   const FX_WCHAR* pTrim = sTrim.c_str();
119   const FX_WCHAR* p = pTrim;
120   bool bDot = false;
121   bool bKXJS = false;
122 
123   wchar_t c;
124   while ((c = *p) != L'\0') {
125     if (c == L'.' || c == L',') {
126       if (bDot)
127         return false;
128       bDot = true;
129     } else if (c == L'-' || c == L'+') {
130       if (p != pTrim)
131         return false;
132     } else if (c == L'e' || c == L'E') {
133       if (bKXJS)
134         return false;
135 
136       p++;
137       c = *p;
138       if (c == L'+' || c == L'-') {
139         bKXJS = true;
140       } else {
141         return false;
142       }
143     } else if (!FXSYS_iswdigit(c)) {
144       return false;
145     }
146     p++;
147   }
148 
149   return true;
150 }
151 
maskSatisfied(wchar_t c_Change,wchar_t c_Mask)152 bool CJS_PublicMethods::maskSatisfied(wchar_t c_Change, wchar_t c_Mask) {
153   switch (c_Mask) {
154     case L'9':
155       return FXSYS_iswdigit(c_Change);
156     case L'A':
157       return FXSYS_iswalpha(c_Change);
158     case L'O':
159       return FXSYS_iswalnum(c_Change);
160     case L'X':
161       return true;
162     default:
163       return (c_Change == c_Mask);
164   }
165 }
166 
isReservedMaskChar(wchar_t ch)167 bool CJS_PublicMethods::isReservedMaskChar(wchar_t ch) {
168   return ch == L'9' || ch == L'A' || ch == L'O' || ch == L'X';
169 }
170 
AF_Simple(const FX_WCHAR * sFuction,double dValue1,double dValue2)171 double CJS_PublicMethods::AF_Simple(const FX_WCHAR* sFuction,
172                                     double dValue1,
173                                     double dValue2) {
174   if (FXSYS_wcsicmp(sFuction, L"AVG") == 0 ||
175       FXSYS_wcsicmp(sFuction, L"SUM") == 0) {
176     return dValue1 + dValue2;
177   }
178   if (FXSYS_wcsicmp(sFuction, L"PRD") == 0) {
179     return dValue1 * dValue2;
180   }
181   if (FXSYS_wcsicmp(sFuction, L"MIN") == 0) {
182     return std::min(dValue1, dValue2);
183   }
184   if (FXSYS_wcsicmp(sFuction, L"MAX") == 0) {
185     return std::max(dValue1, dValue2);
186   }
187   return dValue1;
188 }
189 
AF_MakeArrayFromList(CJS_Runtime * pRuntime,CJS_Value val)190 CJS_Array CJS_PublicMethods::AF_MakeArrayFromList(CJS_Runtime* pRuntime,
191                                                   CJS_Value val) {
192   CJS_Array StrArray;
193   if (val.IsArrayObject()) {
194     val.ConvertToArray(pRuntime, StrArray);
195     return StrArray;
196   }
197   CFX_WideString wsStr = val.ToCFXWideString(pRuntime);
198   CFX_ByteString t = CFX_ByteString::FromUnicode(wsStr);
199   const char* p = t.c_str();
200 
201   int ch = ',';
202   int nIndex = 0;
203 
204   while (*p) {
205     const char* pTemp = strchr(p, ch);
206     if (!pTemp) {
207       StrArray.SetElement(
208           pRuntime, nIndex,
209           CJS_Value(pRuntime, StrTrim(CFX_ByteString(p)).c_str()));
210       break;
211     }
212 
213     char* pSub = new char[pTemp - p + 1];
214     strncpy(pSub, p, pTemp - p);
215     *(pSub + (pTemp - p)) = '\0';
216 
217     StrArray.SetElement(
218         pRuntime, nIndex,
219         CJS_Value(pRuntime, StrTrim(CFX_ByteString(pSub)).c_str()));
220     delete[] pSub;
221 
222     nIndex++;
223     p = ++pTemp;
224   }
225   return StrArray;
226 }
227 
ParseStringInteger(const CFX_WideString & str,int nStart,int & nSkip,int nMaxStep)228 int CJS_PublicMethods::ParseStringInteger(const CFX_WideString& str,
229                                           int nStart,
230                                           int& nSkip,
231                                           int nMaxStep) {
232   int nRet = 0;
233   nSkip = 0;
234   for (int i = nStart, sz = str.GetLength(); i < sz; i++) {
235     if (i - nStart > 10)
236       break;
237 
238     FX_WCHAR c = str.GetAt(i);
239     if (!FXSYS_iswdigit(c))
240       break;
241 
242     nRet = nRet * 10 + FXSYS_toDecimalDigit(c);
243     nSkip = i - nStart + 1;
244     if (nSkip >= nMaxStep)
245       break;
246   }
247 
248   return nRet;
249 }
250 
ParseStringString(const CFX_WideString & str,int nStart,int & nSkip)251 CFX_WideString CJS_PublicMethods::ParseStringString(const CFX_WideString& str,
252                                                     int nStart,
253                                                     int& nSkip) {
254   CFX_WideString swRet;
255   nSkip = 0;
256   for (int i = nStart, sz = str.GetLength(); i < sz; i++) {
257     FX_WCHAR c = str.GetAt(i);
258     if (!FXSYS_iswdigit(c))
259       break;
260 
261     swRet += c;
262     nSkip = i - nStart + 1;
263   }
264 
265   return swRet;
266 }
267 
ParseNormalDate(const CFX_WideString & value,bool * bWrongFormat)268 double CJS_PublicMethods::ParseNormalDate(const CFX_WideString& value,
269                                           bool* bWrongFormat) {
270   double dt = JS_GetDateTime();
271 
272   int nYear = JS_GetYearFromTime(dt);
273   int nMonth = JS_GetMonthFromTime(dt) + 1;
274   int nDay = JS_GetDayFromTime(dt);
275   int nHour = JS_GetHourFromTime(dt);
276   int nMin = JS_GetMinFromTime(dt);
277   int nSec = JS_GetSecFromTime(dt);
278 
279   int number[3];
280 
281   int nSkip = 0;
282   int nLen = value.GetLength();
283   int nIndex = 0;
284   int i = 0;
285   while (i < nLen) {
286     if (nIndex > 2)
287       break;
288 
289     FX_WCHAR c = value.GetAt(i);
290     if (FXSYS_iswdigit(c)) {
291       number[nIndex++] = ParseStringInteger(value, i, nSkip, 4);
292       i += nSkip;
293     } else {
294       i++;
295     }
296   }
297 
298   if (nIndex == 2) {
299     // case2: month/day
300     // case3: day/month
301     if ((number[0] >= 1 && number[0] <= 12) &&
302         (number[1] >= 1 && number[1] <= 31)) {
303       nMonth = number[0];
304       nDay = number[1];
305     } else if ((number[0] >= 1 && number[0] <= 31) &&
306                (number[1] >= 1 && number[1] <= 12)) {
307       nDay = number[0];
308       nMonth = number[1];
309     }
310 
311     if (bWrongFormat)
312       *bWrongFormat = false;
313   } else if (nIndex == 3) {
314     // case1: year/month/day
315     // case2: month/day/year
316     // case3: day/month/year
317 
318     if (number[0] > 12 && (number[1] >= 1 && number[1] <= 12) &&
319         (number[2] >= 1 && number[2] <= 31)) {
320       nYear = number[0];
321       nMonth = number[1];
322       nDay = number[2];
323     } else if ((number[0] >= 1 && number[0] <= 12) &&
324                (number[1] >= 1 && number[1] <= 31) && number[2] > 31) {
325       nMonth = number[0];
326       nDay = number[1];
327       nYear = number[2];
328     } else if ((number[0] >= 1 && number[0] <= 31) &&
329                (number[1] >= 1 && number[1] <= 12) && number[2] > 31) {
330       nDay = number[0];
331       nMonth = number[1];
332       nYear = number[2];
333     }
334 
335     if (bWrongFormat)
336       *bWrongFormat = false;
337   } else {
338     if (bWrongFormat)
339       *bWrongFormat = true;
340     return dt;
341   }
342 
343   CFX_WideString swTemp;
344   swTemp.Format(L"%d/%d/%d %d:%d:%d", nMonth, nDay, nYear, nHour, nMin, nSec);
345   return JS_DateParse(swTemp);
346 }
347 
MakeRegularDate(const CFX_WideString & value,const CFX_WideString & format,bool * bWrongFormat)348 double CJS_PublicMethods::MakeRegularDate(const CFX_WideString& value,
349                                           const CFX_WideString& format,
350                                           bool* bWrongFormat) {
351   double dt = JS_GetDateTime();
352 
353   if (format.IsEmpty() || value.IsEmpty())
354     return dt;
355 
356   int nYear = JS_GetYearFromTime(dt);
357   int nMonth = JS_GetMonthFromTime(dt) + 1;
358   int nDay = JS_GetDayFromTime(dt);
359   int nHour = JS_GetHourFromTime(dt);
360   int nMin = JS_GetMinFromTime(dt);
361   int nSec = JS_GetSecFromTime(dt);
362 
363   int nYearSub = 99;  // nYear - 2000;
364 
365   bool bPm = false;
366   bool bExit = false;
367   bool bBadFormat = false;
368 
369   int i = 0;
370   int j = 0;
371 
372   while (i < format.GetLength()) {
373     if (bExit)
374       break;
375 
376     FX_WCHAR c = format.GetAt(i);
377     switch (c) {
378       case ':':
379       case '.':
380       case '-':
381       case '\\':
382       case '/':
383         i++;
384         j++;
385         break;
386 
387       case 'y':
388       case 'm':
389       case 'd':
390       case 'H':
391       case 'h':
392       case 'M':
393       case 's':
394       case 't': {
395         int oldj = j;
396         int nSkip = 0;
397         int remaining = format.GetLength() - i - 1;
398 
399         if (remaining == 0 || format.GetAt(i + 1) != c) {
400           switch (c) {
401             case 'y':
402               i++;
403               j++;
404               break;
405             case 'm':
406               nMonth = ParseStringInteger(value, j, nSkip, 2);
407               i++;
408               j += nSkip;
409               break;
410             case 'd':
411               nDay = ParseStringInteger(value, j, nSkip, 2);
412               i++;
413               j += nSkip;
414               break;
415             case 'H':
416               nHour = ParseStringInteger(value, j, nSkip, 2);
417               i++;
418               j += nSkip;
419               break;
420             case 'h':
421               nHour = ParseStringInteger(value, j, nSkip, 2);
422               i++;
423               j += nSkip;
424               break;
425             case 'M':
426               nMin = ParseStringInteger(value, j, nSkip, 2);
427               i++;
428               j += nSkip;
429               break;
430             case 's':
431               nSec = ParseStringInteger(value, j, nSkip, 2);
432               i++;
433               j += nSkip;
434               break;
435             case 't':
436               bPm = (j < value.GetLength() && value.GetAt(j) == 'p');
437               i++;
438               j++;
439               break;
440           }
441         } else if (remaining == 1 || format.GetAt(i + 2) != c) {
442           switch (c) {
443             case 'y':
444               nYear = ParseStringInteger(value, j, nSkip, 4);
445               i += 2;
446               j += nSkip;
447               break;
448             case 'm':
449               nMonth = ParseStringInteger(value, j, nSkip, 2);
450               i += 2;
451               j += nSkip;
452               break;
453             case 'd':
454               nDay = ParseStringInteger(value, j, nSkip, 2);
455               i += 2;
456               j += nSkip;
457               break;
458             case 'H':
459               nHour = ParseStringInteger(value, j, nSkip, 2);
460               i += 2;
461               j += nSkip;
462               break;
463             case 'h':
464               nHour = ParseStringInteger(value, j, nSkip, 2);
465               i += 2;
466               j += nSkip;
467               break;
468             case 'M':
469               nMin = ParseStringInteger(value, j, nSkip, 2);
470               i += 2;
471               j += nSkip;
472               break;
473             case 's':
474               nSec = ParseStringInteger(value, j, nSkip, 2);
475               i += 2;
476               j += nSkip;
477               break;
478             case 't':
479               bPm = (j + 1 < value.GetLength() && value.GetAt(j) == 'p' &&
480                      value.GetAt(j + 1) == 'm');
481               i += 2;
482               j += 2;
483               break;
484           }
485         } else if (remaining == 2 || format.GetAt(i + 3) != c) {
486           switch (c) {
487             case 'm': {
488               CFX_WideString sMonth = ParseStringString(value, j, nSkip);
489               bool bFind = false;
490               for (int m = 0; m < 12; m++) {
491                 if (sMonth.CompareNoCase(months[m]) == 0) {
492                   nMonth = m + 1;
493                   i += 3;
494                   j += nSkip;
495                   bFind = true;
496                   break;
497                 }
498               }
499 
500               if (!bFind) {
501                 nMonth = ParseStringInteger(value, j, nSkip, 3);
502                 i += 3;
503                 j += nSkip;
504               }
505             } break;
506             case 'y':
507               break;
508             default:
509               i += 3;
510               j += 3;
511               break;
512           }
513         } else if (remaining == 3 || format.GetAt(i + 4) != c) {
514           switch (c) {
515             case 'y':
516               nYear = ParseStringInteger(value, j, nSkip, 4);
517               j += nSkip;
518               i += 4;
519               break;
520             case 'm': {
521               bool bFind = false;
522 
523               CFX_WideString sMonth = ParseStringString(value, j, nSkip);
524               sMonth.MakeLower();
525 
526               for (int m = 0; m < 12; m++) {
527                 CFX_WideString sFullMonths = fullmonths[m];
528                 sFullMonths.MakeLower();
529 
530                 if (sFullMonths.Find(sMonth.c_str(), 0) != -1) {
531                   nMonth = m + 1;
532                   i += 4;
533                   j += nSkip;
534                   bFind = true;
535                   break;
536                 }
537               }
538 
539               if (!bFind) {
540                 nMonth = ParseStringInteger(value, j, nSkip, 4);
541                 i += 4;
542                 j += nSkip;
543               }
544             } break;
545             default:
546               i += 4;
547               j += 4;
548               break;
549           }
550         } else {
551           if (j >= value.GetLength() || format.GetAt(i) != value.GetAt(j)) {
552             bBadFormat = true;
553             bExit = true;
554           }
555           i++;
556           j++;
557         }
558 
559         if (oldj == j) {
560           bBadFormat = true;
561           bExit = true;
562         }
563       }
564 
565       break;
566       default:
567         if (value.GetLength() <= j) {
568           bExit = true;
569         } else if (format.GetAt(i) != value.GetAt(j)) {
570           bBadFormat = true;
571           bExit = true;
572         }
573 
574         i++;
575         j++;
576         break;
577     }
578   }
579 
580   if (bPm)
581     nHour += 12;
582 
583   if (nYear >= 0 && nYear <= nYearSub)
584     nYear += 2000;
585 
586   if (nMonth < 1 || nMonth > 12)
587     bBadFormat = true;
588 
589   if (nDay < 1 || nDay > 31)
590     bBadFormat = true;
591 
592   if (nHour < 0 || nHour > 24)
593     bBadFormat = true;
594 
595   if (nMin < 0 || nMin > 60)
596     bBadFormat = true;
597 
598   if (nSec < 0 || nSec > 60)
599     bBadFormat = true;
600 
601   double dRet = 0;
602   if (bBadFormat) {
603     dRet = ParseNormalDate(value, &bBadFormat);
604   } else {
605     dRet = JS_MakeDate(JS_MakeDay(nYear, nMonth - 1, nDay),
606                        JS_MakeTime(nHour, nMin, nSec, 0));
607     if (JS_PortIsNan(dRet))
608       dRet = JS_DateParse(value);
609   }
610 
611   if (JS_PortIsNan(dRet))
612     dRet = ParseNormalDate(value, &bBadFormat);
613 
614   if (bWrongFormat)
615     *bWrongFormat = bBadFormat;
616 
617   return dRet;
618 }
619 
MakeFormatDate(double dDate,const CFX_WideString & format)620 CFX_WideString CJS_PublicMethods::MakeFormatDate(double dDate,
621                                                  const CFX_WideString& format) {
622   CFX_WideString sRet = L"", sPart = L"";
623 
624   int nYear = JS_GetYearFromTime(dDate);
625   int nMonth = JS_GetMonthFromTime(dDate) + 1;
626   int nDay = JS_GetDayFromTime(dDate);
627   int nHour = JS_GetHourFromTime(dDate);
628   int nMin = JS_GetMinFromTime(dDate);
629   int nSec = JS_GetSecFromTime(dDate);
630 
631   int i = 0;
632   while (i < format.GetLength()) {
633     FX_WCHAR c = format.GetAt(i);
634     int remaining = format.GetLength() - i - 1;
635     sPart = L"";
636     switch (c) {
637       case 'y':
638       case 'm':
639       case 'd':
640       case 'H':
641       case 'h':
642       case 'M':
643       case 's':
644       case 't':
645         if (remaining == 0 || format.GetAt(i + 1) != c) {
646           switch (c) {
647             case 'y':
648               sPart += c;
649               break;
650             case 'm':
651               sPart.Format(L"%d", nMonth);
652               break;
653             case 'd':
654               sPart.Format(L"%d", nDay);
655               break;
656             case 'H':
657               sPart.Format(L"%d", nHour);
658               break;
659             case 'h':
660               sPart.Format(L"%d", nHour > 12 ? nHour - 12 : nHour);
661               break;
662             case 'M':
663               sPart.Format(L"%d", nMin);
664               break;
665             case 's':
666               sPart.Format(L"%d", nSec);
667               break;
668             case 't':
669               sPart += nHour > 12 ? 'p' : 'a';
670               break;
671           }
672           i++;
673         } else if (remaining == 1 || format.GetAt(i + 2) != c) {
674           switch (c) {
675             case 'y':
676               sPart.Format(L"%02d", nYear - (nYear / 100) * 100);
677               break;
678             case 'm':
679               sPart.Format(L"%02d", nMonth);
680               break;
681             case 'd':
682               sPart.Format(L"%02d", nDay);
683               break;
684             case 'H':
685               sPart.Format(L"%02d", nHour);
686               break;
687             case 'h':
688               sPart.Format(L"%02d", nHour > 12 ? nHour - 12 : nHour);
689               break;
690             case 'M':
691               sPart.Format(L"%02d", nMin);
692               break;
693             case 's':
694               sPart.Format(L"%02d", nSec);
695               break;
696             case 't':
697               sPart = nHour > 12 ? L"pm" : L"am";
698               break;
699           }
700           i += 2;
701         } else if (remaining == 2 || format.GetAt(i + 3) != c) {
702           switch (c) {
703             case 'm':
704               i += 3;
705               if (nMonth > 0 && nMonth <= 12)
706                 sPart += months[nMonth - 1];
707               break;
708             default:
709               i += 3;
710               sPart += c;
711               sPart += c;
712               sPart += c;
713               break;
714           }
715         } else if (remaining == 3 || format.GetAt(i + 4) != c) {
716           switch (c) {
717             case 'y':
718               sPart.Format(L"%04d", nYear);
719               i += 4;
720               break;
721             case 'm':
722               i += 4;
723               if (nMonth > 0 && nMonth <= 12)
724                 sPart += fullmonths[nMonth - 1];
725               break;
726             default:
727               i += 4;
728               sPart += c;
729               sPart += c;
730               sPart += c;
731               sPart += c;
732               break;
733           }
734         } else {
735           i++;
736           sPart += c;
737         }
738         break;
739       default:
740         i++;
741         sPart += c;
742         break;
743     }
744 
745     sRet += sPart;
746   }
747 
748   return sRet;
749 }
750 
751 // function AFNumber_Format(nDec, sepStyle, negStyle, currStyle, strCurrency,
752 // bCurrencyPrepend)
AFNumber_Format(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)753 bool CJS_PublicMethods::AFNumber_Format(CJS_Runtime* pRuntime,
754                                         const std::vector<CJS_Value>& params,
755                                         CJS_Value& vRet,
756                                         CFX_WideString& sError) {
757 #if _FX_OS_ != _FX_ANDROID_
758   if (params.size() != 6) {
759     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
760     return false;
761   }
762 
763   CJS_EventHandler* pEvent =
764       pRuntime->GetCurrentEventContext()->GetEventHandler();
765   if (!pEvent->m_pValue)
766     return false;
767 
768   CFX_WideString& Value = pEvent->Value();
769   CFX_ByteString strValue = StrTrim(CFX_ByteString::FromUnicode(Value));
770   if (strValue.IsEmpty())
771     return true;
772 
773   int iDec = params[0].ToInt(pRuntime);
774   int iSepStyle = params[1].ToInt(pRuntime);
775   int iNegStyle = params[2].ToInt(pRuntime);
776   // params[3] is iCurrStyle, it's not used.
777   CFX_WideString wstrCurrency = params[4].ToCFXWideString(pRuntime);
778   bool bCurrencyPrepend = params[5].ToBool(pRuntime);
779 
780   if (iDec < 0)
781     iDec = -iDec;
782 
783   if (iSepStyle < 0 || iSepStyle > 3)
784     iSepStyle = 0;
785 
786   if (iNegStyle < 0 || iNegStyle > 3)
787     iNegStyle = 0;
788 
789   // Processing decimal places
790   strValue.Replace(",", ".");
791   double dValue = atof(strValue.c_str());
792   if (iDec > 0)
793     dValue += DOUBLE_CORRECT;
794 
795   // Calculating number string
796   bool bNegative;
797   int iDec2;
798   strValue = CalculateString(dValue, iDec, &iDec2, &bNegative);
799   if (strValue.IsEmpty()) {
800     dValue = 0;
801     strValue = CalculateString(dValue, iDec, &iDec2, &bNegative);
802     if (strValue.IsEmpty()) {
803       strValue = "0";
804       iDec2 = 1;
805     }
806   }
807 
808   // Processing separator style
809   if (iDec2 < strValue.GetLength()) {
810     if (iSepStyle == 2 || iSepStyle == 3)
811       strValue.Replace(".", ",");
812 
813     if (iDec2 == 0)
814       strValue.Insert(iDec2, '0');
815   }
816   if (iSepStyle == 0 || iSepStyle == 2) {
817     char cSeparator;
818     if (iSepStyle == 0)
819       cSeparator = ',';
820     else
821       cSeparator = '.';
822 
823     for (int iDecPositive = iDec2 - 3; iDecPositive > 0; iDecPositive -= 3)
824       strValue.Insert(iDecPositive, cSeparator);
825   }
826 
827   // Processing currency string
828   Value = CFX_WideString::FromLocal(strValue.AsStringC());
829 
830   if (bCurrencyPrepend)
831     Value = wstrCurrency + Value;
832   else
833     Value = Value + wstrCurrency;
834 
835   // Processing negative style
836   if (bNegative) {
837     if (iNegStyle == 0)
838       Value = L"-" + Value;
839     else if (iNegStyle == 2 || iNegStyle == 3)
840       Value = L"(" + Value + L")";
841     if (iNegStyle == 1 || iNegStyle == 3) {
842       if (Field* fTarget = pEvent->Target_Field()) {
843         CJS_Array arColor;
844         CJS_Value vColElm(pRuntime);
845         vColElm = CJS_Value(pRuntime, L"RGB");
846         arColor.SetElement(pRuntime, 0, vColElm);
847         vColElm = CJS_Value(pRuntime, 1);
848         arColor.SetElement(pRuntime, 1, vColElm);
849         vColElm = CJS_Value(pRuntime, 0);
850         arColor.SetElement(pRuntime, 2, vColElm);
851         arColor.SetElement(pRuntime, 3, vColElm);
852 
853         CJS_PropValue vProp(pRuntime);
854         vProp.StartGetting();
855         vProp << arColor;
856         vProp.StartSetting();
857         fTarget->textColor(pRuntime, vProp, sError);  // red
858       }
859     }
860   } else {
861     if (iNegStyle == 1 || iNegStyle == 3) {
862       if (Field* fTarget = pEvent->Target_Field()) {
863         CJS_Array arColor;
864         CJS_Value vColElm(pRuntime);
865         vColElm = CJS_Value(pRuntime, L"RGB");
866         arColor.SetElement(pRuntime, 0, vColElm);
867         vColElm = CJS_Value(pRuntime, 0);
868         arColor.SetElement(pRuntime, 1, vColElm);
869         arColor.SetElement(pRuntime, 2, vColElm);
870         arColor.SetElement(pRuntime, 3, vColElm);
871 
872         CJS_PropValue vProp(pRuntime);
873         vProp.StartGetting();
874         fTarget->textColor(pRuntime, vProp, sError);
875 
876         CJS_Array aProp;
877         vProp.GetJSValue()->ConvertToArray(pRuntime, aProp);
878 
879         CPWL_Color crProp;
880         CPWL_Color crColor;
881         color::ConvertArrayToPWLColor(pRuntime, aProp, &crProp);
882         color::ConvertArrayToPWLColor(pRuntime, arColor, &crColor);
883 
884         if (crColor != crProp) {
885           CJS_PropValue vProp2(pRuntime);
886           vProp2.StartGetting();
887           vProp2 << arColor;
888           vProp2.StartSetting();
889           fTarget->textColor(pRuntime, vProp2, sError);
890         }
891       }
892     }
893   }
894 #endif
895   return true;
896 }
897 
898 // function AFNumber_Keystroke(nDec, sepStyle, negStyle, currStyle, strCurrency,
899 // bCurrencyPrepend)
AFNumber_Keystroke(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)900 bool CJS_PublicMethods::AFNumber_Keystroke(CJS_Runtime* pRuntime,
901                                            const std::vector<CJS_Value>& params,
902                                            CJS_Value& vRet,
903                                            CFX_WideString& sError) {
904   if (params.size() < 2)
905     return false;
906 
907   CJS_EventContext* pContext = pRuntime->GetCurrentEventContext();
908   CJS_EventHandler* pEvent = pContext->GetEventHandler();
909   if (!pEvent->m_pValue)
910     return false;
911 
912   CFX_WideString& val = pEvent->Value();
913   CFX_WideString& wstrChange = pEvent->Change();
914   CFX_WideString wstrValue = val;
915 
916   if (pEvent->WillCommit()) {
917     CFX_WideString swTemp = StrTrim(wstrValue);
918     if (swTemp.IsEmpty())
919       return true;
920 
921     swTemp.Replace(L",", L".");
922     if (!IsNumber(swTemp.c_str())) {
923       pEvent->Rc() = false;
924       sError = JSGetStringFromID(IDS_STRING_JSAFNUMBER_KEYSTROKE);
925       AlertIfPossible(pContext, sError.c_str());
926     }
927     return true;  // it happens after the last keystroke and before validating,
928   }
929 
930   CFX_WideString wstrSelected;
931   if (pEvent->SelStart() != -1) {
932     wstrSelected = wstrValue.Mid(pEvent->SelStart(),
933                                  pEvent->SelEnd() - pEvent->SelStart());
934   }
935 
936   bool bHasSign = wstrValue.Find(L'-') != -1 && wstrSelected.Find(L'-') == -1;
937   if (bHasSign) {
938     // can't insert "change" in front to sign postion.
939     if (pEvent->SelStart() == 0) {
940       bool& bRc = pEvent->Rc();
941       bRc = false;
942       return true;
943     }
944   }
945 
946   int iSepStyle = params[1].ToInt(pRuntime);
947   if (iSepStyle < 0 || iSepStyle > 3)
948     iSepStyle = 0;
949   const FX_WCHAR cSep = iSepStyle < 2 ? L'.' : L',';
950 
951   bool bHasSep = wstrValue.Find(cSep) != -1;
952   for (FX_STRSIZE i = 0; i < wstrChange.GetLength(); ++i) {
953     if (wstrChange[i] == cSep) {
954       if (bHasSep) {
955         bool& bRc = pEvent->Rc();
956         bRc = false;
957         return true;
958       }
959       bHasSep = true;
960       continue;
961     }
962     if (wstrChange[i] == L'-') {
963       if (bHasSign) {
964         bool& bRc = pEvent->Rc();
965         bRc = false;
966         return true;
967       }
968       // sign's position is not correct
969       if (i != 0) {
970         bool& bRc = pEvent->Rc();
971         bRc = false;
972         return true;
973       }
974       if (pEvent->SelStart() != 0) {
975         bool& bRc = pEvent->Rc();
976         bRc = false;
977         return true;
978       }
979       bHasSign = true;
980       continue;
981     }
982 
983     if (!FXSYS_iswdigit(wstrChange[i])) {
984       bool& bRc = pEvent->Rc();
985       bRc = false;
986       return true;
987     }
988   }
989 
990   CFX_WideString wprefix = wstrValue.Mid(0, pEvent->SelStart());
991   CFX_WideString wpostfix;
992   if (pEvent->SelEnd() < wstrValue.GetLength())
993     wpostfix = wstrValue.Mid(pEvent->SelEnd());
994   val = wprefix + wstrChange + wpostfix;
995   return true;
996 }
997 
998 // function AFPercent_Format(nDec, sepStyle)
AFPercent_Format(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)999 bool CJS_PublicMethods::AFPercent_Format(CJS_Runtime* pRuntime,
1000                                          const std::vector<CJS_Value>& params,
1001                                          CJS_Value& vRet,
1002                                          CFX_WideString& sError) {
1003 #if _FX_OS_ != _FX_ANDROID_
1004   if (params.size() != 2) {
1005     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1006     return false;
1007   }
1008 
1009   CJS_EventHandler* pEvent =
1010       pRuntime->GetCurrentEventContext()->GetEventHandler();
1011   if (!pEvent->m_pValue)
1012     return false;
1013 
1014   CFX_WideString& Value = pEvent->Value();
1015   CFX_ByteString strValue = StrTrim(CFX_ByteString::FromUnicode(Value));
1016   if (strValue.IsEmpty())
1017     return true;
1018 
1019   int iDec = params[0].ToInt(pRuntime);
1020   if (iDec < 0)
1021     iDec = -iDec;
1022 
1023   int iSepStyle = params[1].ToInt(pRuntime);
1024   if (iSepStyle < 0 || iSepStyle > 3)
1025     iSepStyle = 0;
1026 
1027   // for processing decimal places
1028   double dValue = atof(strValue.c_str());
1029   dValue *= 100;
1030   if (iDec > 0)
1031     dValue += DOUBLE_CORRECT;
1032 
1033   int iDec2;
1034   int iNegative = 0;
1035   strValue = fcvt(dValue, iDec, &iDec2, &iNegative);
1036   if (strValue.IsEmpty()) {
1037     dValue = 0;
1038     strValue = fcvt(dValue, iDec, &iDec2, &iNegative);
1039   }
1040 
1041   if (iDec2 < 0) {
1042     for (int iNum = 0; iNum < abs(iDec2); iNum++) {
1043       strValue = "0" + strValue;
1044     }
1045     iDec2 = 0;
1046   }
1047   int iMax = strValue.GetLength();
1048   if (iDec2 > iMax) {
1049     for (int iNum = 0; iNum <= iDec2 - iMax; iNum++) {
1050       strValue += "0";
1051     }
1052     iMax = iDec2 + 1;
1053   }
1054 
1055   // for processing seperator style
1056   if (iDec2 < iMax) {
1057     if (iSepStyle == 0 || iSepStyle == 1) {
1058       strValue.Insert(iDec2, '.');
1059       iMax++;
1060     } else if (iSepStyle == 2 || iSepStyle == 3) {
1061       strValue.Insert(iDec2, ',');
1062       iMax++;
1063     }
1064 
1065     if (iDec2 == 0)
1066       strValue.Insert(iDec2, '0');
1067   }
1068   if (iSepStyle == 0 || iSepStyle == 2) {
1069     char cSeperator;
1070     if (iSepStyle == 0)
1071       cSeperator = ',';
1072     else
1073       cSeperator = '.';
1074 
1075     for (int iDecPositive = iDec2 - 3; iDecPositive > 0; iDecPositive -= 3) {
1076       strValue.Insert(iDecPositive, cSeperator);
1077       iMax++;
1078     }
1079   }
1080 
1081   // negative mark
1082   if (iNegative)
1083     strValue = "-" + strValue;
1084   strValue += "%";
1085   Value = CFX_WideString::FromLocal(strValue.AsStringC());
1086 #endif
1087   return true;
1088 }
1089 // AFPercent_Keystroke(nDec, sepStyle)
AFPercent_Keystroke(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1090 bool CJS_PublicMethods::AFPercent_Keystroke(
1091     CJS_Runtime* pRuntime,
1092     const std::vector<CJS_Value>& params,
1093     CJS_Value& vRet,
1094     CFX_WideString& sError) {
1095   return AFNumber_Keystroke(pRuntime, params, vRet, sError);
1096 }
1097 
1098 // function AFDate_FormatEx(cFormat)
AFDate_FormatEx(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1099 bool CJS_PublicMethods::AFDate_FormatEx(CJS_Runtime* pRuntime,
1100                                         const std::vector<CJS_Value>& params,
1101                                         CJS_Value& vRet,
1102                                         CFX_WideString& sError) {
1103   if (params.size() != 1) {
1104     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1105     return false;
1106   }
1107 
1108   CJS_EventContext* pContext = pRuntime->GetCurrentEventContext();
1109   CJS_EventHandler* pEvent = pContext->GetEventHandler();
1110   if (!pEvent->m_pValue)
1111     return false;
1112 
1113   CFX_WideString& val = pEvent->Value();
1114   CFX_WideString strValue = val;
1115   if (strValue.IsEmpty())
1116     return true;
1117 
1118   CFX_WideString sFormat = params[0].ToCFXWideString(pRuntime);
1119   double dDate = 0.0f;
1120 
1121   if (strValue.Find(L"GMT") != -1) {
1122     // for GMT format time
1123     // such as "Tue Aug 11 14:24:16 GMT+08002009"
1124     dDate = MakeInterDate(strValue);
1125   } else {
1126     dDate = MakeRegularDate(strValue, sFormat, nullptr);
1127   }
1128 
1129   if (JS_PortIsNan(dDate)) {
1130     CFX_WideString swMsg;
1131     swMsg.Format(JSGetStringFromID(IDS_STRING_JSPARSEDATE).c_str(),
1132                  sFormat.c_str());
1133     AlertIfPossible(pContext, swMsg.c_str());
1134     return false;
1135   }
1136 
1137   val = MakeFormatDate(dDate, sFormat);
1138   return true;
1139 }
1140 
MakeInterDate(const CFX_WideString & strValue)1141 double CJS_PublicMethods::MakeInterDate(const CFX_WideString& strValue) {
1142   std::vector<CFX_WideString> wsArray;
1143   CFX_WideString sTemp = L"";
1144   for (int i = 0; i < strValue.GetLength(); ++i) {
1145     FX_WCHAR c = strValue.GetAt(i);
1146     if (c == L' ' || c == L':') {
1147       wsArray.push_back(sTemp);
1148       sTemp = L"";
1149       continue;
1150     }
1151     sTemp += c;
1152   }
1153   wsArray.push_back(sTemp);
1154   if (wsArray.size() != 8)
1155     return 0;
1156 
1157   int nMonth = 1;
1158   sTemp = wsArray[1];
1159   if (sTemp.Compare(L"Jan") == 0)
1160     nMonth = 1;
1161   else if (sTemp.Compare(L"Feb") == 0)
1162     nMonth = 2;
1163   else if (sTemp.Compare(L"Mar") == 0)
1164     nMonth = 3;
1165   else if (sTemp.Compare(L"Apr") == 0)
1166     nMonth = 4;
1167   else if (sTemp.Compare(L"May") == 0)
1168     nMonth = 5;
1169   else if (sTemp.Compare(L"Jun") == 0)
1170     nMonth = 6;
1171   else if (sTemp.Compare(L"Jul") == 0)
1172     nMonth = 7;
1173   else if (sTemp.Compare(L"Aug") == 0)
1174     nMonth = 8;
1175   else if (sTemp.Compare(L"Sep") == 0)
1176     nMonth = 9;
1177   else if (sTemp.Compare(L"Oct") == 0)
1178     nMonth = 10;
1179   else if (sTemp.Compare(L"Nov") == 0)
1180     nMonth = 11;
1181   else if (sTemp.Compare(L"Dec") == 0)
1182     nMonth = 12;
1183 
1184   int nDay = FX_atof(wsArray[2].AsStringC());
1185   int nHour = FX_atof(wsArray[3].AsStringC());
1186   int nMin = FX_atof(wsArray[4].AsStringC());
1187   int nSec = FX_atof(wsArray[5].AsStringC());
1188   int nYear = FX_atof(wsArray[7].AsStringC());
1189   double dRet = JS_MakeDate(JS_MakeDay(nYear, nMonth - 1, nDay),
1190                             JS_MakeTime(nHour, nMin, nSec, 0));
1191   if (JS_PortIsNan(dRet))
1192     dRet = JS_DateParse(strValue);
1193 
1194   return dRet;
1195 }
1196 
1197 // AFDate_KeystrokeEx(cFormat)
AFDate_KeystrokeEx(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1198 bool CJS_PublicMethods::AFDate_KeystrokeEx(CJS_Runtime* pRuntime,
1199                                            const std::vector<CJS_Value>& params,
1200                                            CJS_Value& vRet,
1201                                            CFX_WideString& sError) {
1202   if (params.size() != 1) {
1203     sError = L"AFDate_KeystrokeEx's parameters' size r not correct";
1204     return false;
1205   }
1206 
1207   CJS_EventContext* pContext = pRuntime->GetCurrentEventContext();
1208   CJS_EventHandler* pEvent = pContext->GetEventHandler();
1209   if (pEvent->WillCommit()) {
1210     if (!pEvent->m_pValue)
1211       return false;
1212 
1213     CFX_WideString strValue = pEvent->Value();
1214     if (strValue.IsEmpty())
1215       return true;
1216 
1217     CFX_WideString sFormat = params[0].ToCFXWideString(pRuntime);
1218     bool bWrongFormat = false;
1219     double dRet = MakeRegularDate(strValue, sFormat, &bWrongFormat);
1220     if (bWrongFormat || JS_PortIsNan(dRet)) {
1221       CFX_WideString swMsg;
1222       swMsg.Format(JSGetStringFromID(IDS_STRING_JSPARSEDATE).c_str(),
1223                    sFormat.c_str());
1224       AlertIfPossible(pContext, swMsg.c_str());
1225       pEvent->Rc() = false;
1226       return true;
1227     }
1228   }
1229   return true;
1230 }
1231 
AFDate_Format(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1232 bool CJS_PublicMethods::AFDate_Format(CJS_Runtime* pRuntime,
1233                                       const std::vector<CJS_Value>& params,
1234                                       CJS_Value& vRet,
1235                                       CFX_WideString& sError) {
1236   if (params.size() != 1) {
1237     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1238     return false;
1239   }
1240 
1241   int iIndex = params[0].ToInt(pRuntime);
1242   const FX_WCHAR* cFormats[] = {L"m/d",
1243                                 L"m/d/yy",
1244                                 L"mm/dd/yy",
1245                                 L"mm/yy",
1246                                 L"d-mmm",
1247                                 L"d-mmm-yy",
1248                                 L"dd-mmm-yy",
1249                                 L"yy-mm-dd",
1250                                 L"mmm-yy",
1251                                 L"mmmm-yy",
1252                                 L"mmm d, yyyy",
1253                                 L"mmmm d, yyyy",
1254                                 L"m/d/yy h:MM tt",
1255                                 L"m/d/yy HH:MM"};
1256 
1257   if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
1258     iIndex = 0;
1259 
1260   std::vector<CJS_Value> newParams;
1261   newParams.push_back(CJS_Value(pRuntime, cFormats[iIndex]));
1262   return AFDate_FormatEx(pRuntime, newParams, vRet, sError);
1263 }
1264 
1265 // AFDate_KeystrokeEx(cFormat)
AFDate_Keystroke(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1266 bool CJS_PublicMethods::AFDate_Keystroke(CJS_Runtime* pRuntime,
1267                                          const std::vector<CJS_Value>& params,
1268                                          CJS_Value& vRet,
1269                                          CFX_WideString& sError) {
1270   if (params.size() != 1) {
1271     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1272     return false;
1273   }
1274 
1275   int iIndex = params[0].ToInt(pRuntime);
1276   const FX_WCHAR* cFormats[] = {L"m/d",
1277                                 L"m/d/yy",
1278                                 L"mm/dd/yy",
1279                                 L"mm/yy",
1280                                 L"d-mmm",
1281                                 L"d-mmm-yy",
1282                                 L"dd-mmm-yy",
1283                                 L"yy-mm-dd",
1284                                 L"mmm-yy",
1285                                 L"mmmm-yy",
1286                                 L"mmm d, yyyy",
1287                                 L"mmmm d, yyyy",
1288                                 L"m/d/yy h:MM tt",
1289                                 L"m/d/yy HH:MM"};
1290 
1291   if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
1292     iIndex = 0;
1293 
1294   std::vector<CJS_Value> newParams;
1295   newParams.push_back(CJS_Value(pRuntime, cFormats[iIndex]));
1296   return AFDate_KeystrokeEx(pRuntime, newParams, vRet, sError);
1297 }
1298 
1299 // function AFTime_Format(ptf)
AFTime_Format(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1300 bool CJS_PublicMethods::AFTime_Format(CJS_Runtime* pRuntime,
1301                                       const std::vector<CJS_Value>& params,
1302                                       CJS_Value& vRet,
1303                                       CFX_WideString& sError) {
1304   if (params.size() != 1) {
1305     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1306     return false;
1307   }
1308 
1309   int iIndex = params[0].ToInt(pRuntime);
1310   const FX_WCHAR* cFormats[] = {L"HH:MM", L"h:MM tt", L"HH:MM:ss",
1311                                 L"h:MM:ss tt"};
1312 
1313   if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
1314     iIndex = 0;
1315 
1316   std::vector<CJS_Value> newParams;
1317   newParams.push_back(CJS_Value(pRuntime, cFormats[iIndex]));
1318   return AFDate_FormatEx(pRuntime, newParams, vRet, sError);
1319 }
1320 
AFTime_Keystroke(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1321 bool CJS_PublicMethods::AFTime_Keystroke(CJS_Runtime* pRuntime,
1322                                          const std::vector<CJS_Value>& params,
1323                                          CJS_Value& vRet,
1324                                          CFX_WideString& sError) {
1325   if (params.size() != 1) {
1326     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1327     return false;
1328   }
1329 
1330   int iIndex = params[0].ToInt(pRuntime);
1331   const FX_WCHAR* cFormats[] = {L"HH:MM", L"h:MM tt", L"HH:MM:ss",
1332                                 L"h:MM:ss tt"};
1333 
1334   if (iIndex < 0 || (static_cast<size_t>(iIndex) >= FX_ArraySize(cFormats)))
1335     iIndex = 0;
1336 
1337   std::vector<CJS_Value> newParams;
1338   newParams.push_back(CJS_Value(pRuntime, cFormats[iIndex]));
1339   return AFDate_KeystrokeEx(pRuntime, newParams, vRet, sError);
1340 }
1341 
AFTime_FormatEx(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1342 bool CJS_PublicMethods::AFTime_FormatEx(CJS_Runtime* pRuntime,
1343                                         const std::vector<CJS_Value>& params,
1344                                         CJS_Value& vRet,
1345                                         CFX_WideString& sError) {
1346   return AFDate_FormatEx(pRuntime, params, vRet, sError);
1347 }
1348 
AFTime_KeystrokeEx(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1349 bool CJS_PublicMethods::AFTime_KeystrokeEx(CJS_Runtime* pRuntime,
1350                                            const std::vector<CJS_Value>& params,
1351                                            CJS_Value& vRet,
1352                                            CFX_WideString& sError) {
1353   return AFDate_KeystrokeEx(pRuntime, params, vRet, sError);
1354 }
1355 
1356 // function AFSpecial_Format(psf)
AFSpecial_Format(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1357 bool CJS_PublicMethods::AFSpecial_Format(CJS_Runtime* pRuntime,
1358                                          const std::vector<CJS_Value>& params,
1359                                          CJS_Value& vRet,
1360                                          CFX_WideString& sError) {
1361   if (params.size() != 1) {
1362     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1363     return false;
1364   }
1365 
1366   CJS_EventHandler* pEvent =
1367       pRuntime->GetCurrentEventContext()->GetEventHandler();
1368   if (!pEvent->m_pValue)
1369     return false;
1370 
1371   CFX_WideString wsSource = pEvent->Value();
1372   CFX_WideString wsFormat;
1373   switch (params[0].ToInt(pRuntime)) {
1374     case 0:
1375       wsFormat = L"99999";
1376       break;
1377     case 1:
1378       wsFormat = L"99999-9999";
1379       break;
1380     case 2:
1381       if (util::printx(L"9999999999", wsSource).GetLength() >= 10)
1382         wsFormat = L"(999) 999-9999";
1383       else
1384         wsFormat = L"999-9999";
1385       break;
1386     case 3:
1387       wsFormat = L"999-99-9999";
1388       break;
1389   }
1390 
1391   pEvent->Value() = util::printx(wsFormat, wsSource);
1392   return true;
1393 }
1394 
1395 // function AFSpecial_KeystrokeEx(mask)
AFSpecial_KeystrokeEx(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1396 bool CJS_PublicMethods::AFSpecial_KeystrokeEx(
1397     CJS_Runtime* pRuntime,
1398     const std::vector<CJS_Value>& params,
1399     CJS_Value& vRet,
1400     CFX_WideString& sError) {
1401   if (params.size() < 1) {
1402     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1403     return false;
1404   }
1405 
1406   CJS_EventContext* pContext = pRuntime->GetCurrentEventContext();
1407   CJS_EventHandler* pEvent = pContext->GetEventHandler();
1408   if (!pEvent->m_pValue)
1409     return false;
1410 
1411   CFX_WideString& valEvent = pEvent->Value();
1412   CFX_WideString wstrMask = params[0].ToCFXWideString(pRuntime);
1413   if (wstrMask.IsEmpty())
1414     return true;
1415 
1416   if (pEvent->WillCommit()) {
1417     if (valEvent.IsEmpty())
1418       return true;
1419 
1420     FX_STRSIZE iIndexMask = 0;
1421     for (; iIndexMask < valEvent.GetLength(); ++iIndexMask) {
1422       if (!maskSatisfied(valEvent[iIndexMask], wstrMask[iIndexMask]))
1423         break;
1424     }
1425 
1426     if (iIndexMask != wstrMask.GetLength() ||
1427         (iIndexMask != valEvent.GetLength() && wstrMask.GetLength() != 0)) {
1428       AlertIfPossible(
1429           pContext, JSGetStringFromID(IDS_STRING_JSAFNUMBER_KEYSTROKE).c_str());
1430       pEvent->Rc() = false;
1431     }
1432     return true;
1433   }
1434 
1435   CFX_WideString& wideChange = pEvent->Change();
1436   if (wideChange.IsEmpty())
1437     return true;
1438 
1439   CFX_WideString wChange = wideChange;
1440   FX_STRSIZE iIndexMask = pEvent->SelStart();
1441   FX_STRSIZE combined_len = valEvent.GetLength() + wChange.GetLength() +
1442                             pEvent->SelStart() - pEvent->SelEnd();
1443   if (combined_len > wstrMask.GetLength()) {
1444     AlertIfPossible(pContext,
1445                     JSGetStringFromID(IDS_STRING_JSPARAM_TOOLONG).c_str());
1446     pEvent->Rc() = false;
1447     return true;
1448   }
1449 
1450   if (iIndexMask >= wstrMask.GetLength() && !wChange.IsEmpty()) {
1451     AlertIfPossible(pContext,
1452                     JSGetStringFromID(IDS_STRING_JSPARAM_TOOLONG).c_str());
1453     pEvent->Rc() = false;
1454     return true;
1455   }
1456 
1457   for (FX_STRSIZE i = 0; i < wChange.GetLength(); ++i) {
1458     if (iIndexMask >= wstrMask.GetLength()) {
1459       AlertIfPossible(pContext,
1460                       JSGetStringFromID(IDS_STRING_JSPARAM_TOOLONG).c_str());
1461       pEvent->Rc() = false;
1462       return true;
1463     }
1464     FX_WCHAR wMask = wstrMask[iIndexMask];
1465     if (!isReservedMaskChar(wMask))
1466       wChange.SetAt(i, wMask);
1467 
1468     if (!maskSatisfied(wChange[i], wMask)) {
1469       pEvent->Rc() = false;
1470       return true;
1471     }
1472     iIndexMask++;
1473   }
1474   wideChange = wChange;
1475   return true;
1476 }
1477 
1478 // function AFSpecial_Keystroke(psf)
AFSpecial_Keystroke(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1479 bool CJS_PublicMethods::AFSpecial_Keystroke(
1480     CJS_Runtime* pRuntime,
1481     const std::vector<CJS_Value>& params,
1482     CJS_Value& vRet,
1483     CFX_WideString& sError) {
1484   if (params.size() != 1) {
1485     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1486     return false;
1487   }
1488 
1489   CJS_EventHandler* pEvent =
1490       pRuntime->GetCurrentEventContext()->GetEventHandler();
1491   if (!pEvent->m_pValue)
1492     return false;
1493 
1494   const char* cFormat = "";
1495   switch (params[0].ToInt(pRuntime)) {
1496     case 0:
1497       cFormat = "99999";
1498       break;
1499     case 1:
1500       cFormat = "999999999";
1501       break;
1502     case 2:
1503       if (pEvent->Value().GetLength() + pEvent->Change().GetLength() > 7)
1504         cFormat = "9999999999";
1505       else
1506         cFormat = "9999999";
1507       break;
1508     case 3:
1509       cFormat = "999999999";
1510       break;
1511   }
1512 
1513   std::vector<CJS_Value> params2;
1514   params2.push_back(CJS_Value(pRuntime, cFormat));
1515   return AFSpecial_KeystrokeEx(pRuntime, params2, vRet, sError);
1516 }
1517 
AFMergeChange(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1518 bool CJS_PublicMethods::AFMergeChange(CJS_Runtime* pRuntime,
1519                                       const std::vector<CJS_Value>& params,
1520                                       CJS_Value& vRet,
1521                                       CFX_WideString& sError) {
1522   if (params.size() != 1) {
1523     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1524     return false;
1525   }
1526 
1527   CJS_EventHandler* pEventHandler =
1528       pRuntime->GetCurrentEventContext()->GetEventHandler();
1529 
1530   CFX_WideString swValue;
1531   if (pEventHandler->m_pValue)
1532     swValue = pEventHandler->Value();
1533 
1534   if (pEventHandler->WillCommit()) {
1535     vRet = CJS_Value(pRuntime, swValue.c_str());
1536     return true;
1537   }
1538 
1539   CFX_WideString prefix, postfix;
1540 
1541   if (pEventHandler->SelStart() >= 0)
1542     prefix = swValue.Mid(0, pEventHandler->SelStart());
1543   else
1544     prefix = L"";
1545 
1546   if (pEventHandler->SelEnd() >= 0 &&
1547       pEventHandler->SelEnd() <= swValue.GetLength())
1548     postfix = swValue.Mid(pEventHandler->SelEnd(),
1549                           swValue.GetLength() - pEventHandler->SelEnd());
1550   else
1551     postfix = L"";
1552 
1553   vRet =
1554       CJS_Value(pRuntime, (prefix + pEventHandler->Change() + postfix).c_str());
1555   return true;
1556 }
1557 
AFParseDateEx(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1558 bool CJS_PublicMethods::AFParseDateEx(CJS_Runtime* pRuntime,
1559                                       const std::vector<CJS_Value>& params,
1560                                       CJS_Value& vRet,
1561                                       CFX_WideString& sError) {
1562   if (params.size() != 2) {
1563     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1564     return false;
1565   }
1566 
1567   CFX_WideString sValue = params[0].ToCFXWideString(pRuntime);
1568   CFX_WideString sFormat = params[1].ToCFXWideString(pRuntime);
1569   double dDate = MakeRegularDate(sValue, sFormat, nullptr);
1570   if (JS_PortIsNan(dDate)) {
1571     CFX_WideString swMsg;
1572     swMsg.Format(JSGetStringFromID(IDS_STRING_JSPARSEDATE).c_str(),
1573                  sFormat.c_str());
1574     AlertIfPossible(pRuntime->GetCurrentEventContext(), swMsg.c_str());
1575     return false;
1576   }
1577 
1578   vRet = CJS_Value(pRuntime, dDate);
1579   return true;
1580 }
1581 
AFSimple(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1582 bool CJS_PublicMethods::AFSimple(CJS_Runtime* pRuntime,
1583                                  const std::vector<CJS_Value>& params,
1584                                  CJS_Value& vRet,
1585                                  CFX_WideString& sError) {
1586   if (params.size() != 3) {
1587     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1588     return false;
1589   }
1590 
1591   vRet = CJS_Value(pRuntime, static_cast<double>(AF_Simple(
1592                                  params[0].ToCFXWideString(pRuntime).c_str(),
1593                                  params[1].ToDouble(pRuntime),
1594                                  params[2].ToDouble(pRuntime))));
1595 
1596   return true;
1597 }
1598 
AFMakeNumber(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1599 bool CJS_PublicMethods::AFMakeNumber(CJS_Runtime* pRuntime,
1600                                      const std::vector<CJS_Value>& params,
1601                                      CJS_Value& vRet,
1602                                      CFX_WideString& sError) {
1603   if (params.size() != 1) {
1604     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1605     return false;
1606   }
1607 
1608   CFX_WideString ws = params[0].ToCFXWideString(pRuntime);
1609   ws.Replace(L",", L".");
1610   vRet = CJS_Value(pRuntime, ws.c_str());
1611   vRet.MaybeCoerceToNumber(pRuntime);
1612   if (vRet.GetType() != CJS_Value::VT_number)
1613     vRet = CJS_Value(pRuntime, 0);
1614   return true;
1615 }
1616 
AFSimple_Calculate(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1617 bool CJS_PublicMethods::AFSimple_Calculate(CJS_Runtime* pRuntime,
1618                                            const std::vector<CJS_Value>& params,
1619                                            CJS_Value& vRet,
1620                                            CFX_WideString& sError) {
1621   if (params.size() != 2) {
1622     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1623     return false;
1624   }
1625 
1626   CJS_Value params1 = params[1];
1627   if (!params1.IsArrayObject() && params1.GetType() != CJS_Value::VT_string) {
1628     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1629     return false;
1630   }
1631 
1632   CPDFSDK_InterForm* pReaderInterForm =
1633       pRuntime->GetFormFillEnv()->GetInterForm();
1634   CPDF_InterForm* pInterForm = pReaderInterForm->GetInterForm();
1635 
1636   CFX_WideString sFunction = params[0].ToCFXWideString(pRuntime);
1637   double dValue = wcscmp(sFunction.c_str(), L"PRD") == 0 ? 1.0 : 0.0;
1638 
1639   CJS_Array FieldNameArray = AF_MakeArrayFromList(pRuntime, params1);
1640   int nFieldsCount = 0;
1641 
1642   for (int i = 0, isz = FieldNameArray.GetLength(pRuntime); i < isz; i++) {
1643     CJS_Value jsValue(pRuntime);
1644     FieldNameArray.GetElement(pRuntime, i, jsValue);
1645     CFX_WideString wsFieldName = jsValue.ToCFXWideString(pRuntime);
1646 
1647     for (int j = 0, jsz = pInterForm->CountFields(wsFieldName); j < jsz; j++) {
1648       if (CPDF_FormField* pFormField = pInterForm->GetField(j, wsFieldName)) {
1649         double dTemp = 0.0;
1650         switch (pFormField->GetFieldType()) {
1651           case FIELDTYPE_TEXTFIELD:
1652           case FIELDTYPE_COMBOBOX: {
1653             CFX_WideString trimmed = pFormField->GetValue();
1654             trimmed.TrimRight();
1655             trimmed.TrimLeft();
1656             dTemp = FX_atof(trimmed.AsStringC());
1657           } break;
1658           case FIELDTYPE_PUSHBUTTON: {
1659             dTemp = 0.0;
1660           } break;
1661           case FIELDTYPE_CHECKBOX:
1662           case FIELDTYPE_RADIOBUTTON: {
1663             dTemp = 0.0;
1664             for (int c = 0, csz = pFormField->CountControls(); c < csz; c++) {
1665               if (CPDF_FormControl* pFormCtrl = pFormField->GetControl(c)) {
1666                 if (pFormCtrl->IsChecked()) {
1667                   CFX_WideString trimmed = pFormCtrl->GetExportValue();
1668                   trimmed.TrimRight();
1669                   trimmed.TrimLeft();
1670                   dTemp = FX_atof(trimmed.AsStringC());
1671                   break;
1672                 }
1673               }
1674             }
1675           } break;
1676           case FIELDTYPE_LISTBOX: {
1677             if (pFormField->CountSelectedItems() <= 1) {
1678               CFX_WideString trimmed = pFormField->GetValue();
1679               trimmed.TrimRight();
1680               trimmed.TrimLeft();
1681               dTemp = FX_atof(trimmed.AsStringC());
1682             }
1683           } break;
1684           default:
1685             break;
1686         }
1687 
1688         if (i == 0 && j == 0 && (wcscmp(sFunction.c_str(), L"MIN") == 0 ||
1689                                  wcscmp(sFunction.c_str(), L"MAX") == 0))
1690           dValue = dTemp;
1691 
1692         dValue = AF_Simple(sFunction.c_str(), dValue, dTemp);
1693 
1694         nFieldsCount++;
1695       }
1696     }
1697   }
1698 
1699   if (wcscmp(sFunction.c_str(), L"AVG") == 0 && nFieldsCount > 0)
1700     dValue /= nFieldsCount;
1701 
1702   dValue = (double)floor(dValue * FXSYS_pow((double)10, (double)6) + 0.49) /
1703            FXSYS_pow((double)10, (double)6);
1704 
1705   CJS_Value jsValue(pRuntime, dValue);
1706   CJS_EventContext* pContext = pRuntime->GetCurrentEventContext();
1707   if (pContext->GetEventHandler()->m_pValue)
1708     pContext->GetEventHandler()->Value() = jsValue.ToCFXWideString(pRuntime);
1709 
1710   return true;
1711 }
1712 
1713 /* This function validates the current event to ensure that its value is
1714 ** within the specified range. */
1715 
AFRange_Validate(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1716 bool CJS_PublicMethods::AFRange_Validate(CJS_Runtime* pRuntime,
1717                                          const std::vector<CJS_Value>& params,
1718                                          CJS_Value& vRet,
1719                                          CFX_WideString& sError) {
1720   if (params.size() != 4) {
1721     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1722     return false;
1723   }
1724   CJS_EventContext* pContext = pRuntime->GetCurrentEventContext();
1725   CJS_EventHandler* pEvent = pContext->GetEventHandler();
1726   if (!pEvent->m_pValue)
1727     return false;
1728 
1729   if (pEvent->Value().IsEmpty())
1730     return true;
1731 
1732   double dEentValue =
1733       atof(CFX_ByteString::FromUnicode(pEvent->Value()).c_str());
1734   bool bGreaterThan = params[0].ToBool(pRuntime);
1735   double dGreaterThan = params[1].ToDouble(pRuntime);
1736   bool bLessThan = params[2].ToBool(pRuntime);
1737   double dLessThan = params[3].ToDouble(pRuntime);
1738   CFX_WideString swMsg;
1739 
1740   if (bGreaterThan && bLessThan) {
1741     if (dEentValue < dGreaterThan || dEentValue > dLessThan)
1742       swMsg.Format(JSGetStringFromID(IDS_STRING_JSRANGE1).c_str(),
1743                    params[1].ToCFXWideString(pRuntime).c_str(),
1744                    params[3].ToCFXWideString(pRuntime).c_str());
1745   } else if (bGreaterThan) {
1746     if (dEentValue < dGreaterThan)
1747       swMsg.Format(JSGetStringFromID(IDS_STRING_JSRANGE2).c_str(),
1748                    params[1].ToCFXWideString(pRuntime).c_str());
1749   } else if (bLessThan) {
1750     if (dEentValue > dLessThan)
1751       swMsg.Format(JSGetStringFromID(IDS_STRING_JSRANGE3).c_str(),
1752                    params[3].ToCFXWideString(pRuntime).c_str());
1753   }
1754 
1755   if (!swMsg.IsEmpty()) {
1756     AlertIfPossible(pContext, swMsg.c_str());
1757     pEvent->Rc() = false;
1758   }
1759   return true;
1760 }
1761 
AFExtractNums(CJS_Runtime * pRuntime,const std::vector<CJS_Value> & params,CJS_Value & vRet,CFX_WideString & sError)1762 bool CJS_PublicMethods::AFExtractNums(CJS_Runtime* pRuntime,
1763                                       const std::vector<CJS_Value>& params,
1764                                       CJS_Value& vRet,
1765                                       CFX_WideString& sError) {
1766   if (params.size() != 1) {
1767     sError = JSGetStringFromID(IDS_STRING_JSPARAMERROR);
1768     return false;
1769   }
1770 
1771   CFX_WideString str = params[0].ToCFXWideString(pRuntime);
1772   if (str.GetAt(0) == L'.' || str.GetAt(0) == L',')
1773     str = L"0" + str;
1774 
1775   CFX_WideString sPart;
1776   CJS_Array nums;
1777   int nIndex = 0;
1778   for (int i = 0, sz = str.GetLength(); i < sz; i++) {
1779     FX_WCHAR wc = str.GetAt(i);
1780     if (FXSYS_iswdigit(wc)) {
1781       sPart += wc;
1782     } else {
1783       if (sPart.GetLength() > 0) {
1784         nums.SetElement(pRuntime, nIndex, CJS_Value(pRuntime, sPart.c_str()));
1785         sPart = L"";
1786         nIndex++;
1787       }
1788     }
1789   }
1790 
1791   if (sPart.GetLength() > 0) {
1792     nums.SetElement(pRuntime, nIndex, CJS_Value(pRuntime, sPart.c_str()));
1793   }
1794 
1795   if (nums.GetLength(pRuntime) > 0)
1796     vRet = CJS_Value(pRuntime, nums);
1797   else
1798     vRet.SetNull(pRuntime);
1799 
1800   return true;
1801 }
1802