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