From 3dd8752283771eccc4df2f2df41fbd6b6d1ac299 Mon Sep 17 00:00:00 2001 From: Calixte Denizet Date: Fri, 3 Jan 2025 19:00:33 +0100 Subject: [PATCH] Let be more tolerant with predefined phone number format The long US phone number format should be '(999) 999-9999' but the parenthesis can be omitted or the space replaced with a dash, ... --- src/scripting_api/aform.js | 48 ++++++++++++++++++++++------- test/unit/scripting_spec.js | 61 +++++++++++++++++++++++++++++++++++++ 2 files changed, 98 insertions(+), 11 deletions(-) diff --git a/src/scripting_api/aform.js b/src/scripting_api/aform.js index 78a9a312df26c..5f9c67c90844a 100644 --- a/src/scripting_api/aform.js +++ b/src/scripting_api/aform.js @@ -465,22 +465,22 @@ class AForm { // specific to the format because the user could enter 1234567 when the // format is 999-9999. const simplifiedFormatStr = cMask.replaceAll(/[^9AOX]/g, ""); - this.#AFSpecial_KeystrokeEx_helper(simplifiedFormatStr, false); + this.#AFSpecial_KeystrokeEx_helper(simplifiedFormatStr, null, false); if (event.rc) { return; } event.rc = true; - this.#AFSpecial_KeystrokeEx_helper(cMask, true); + this.#AFSpecial_KeystrokeEx_helper(cMask, null, true); } - #AFSpecial_KeystrokeEx_helper(cMask, warn) { + #AFSpecial_KeystrokeEx_helper(cMask, value, warn) { if (!cMask) { return; } const event = globalThis.event; - const value = this.AFMergeChange(event); + value ||= this.AFMergeChange(event); if (!value) { return; } @@ -563,7 +563,8 @@ class AForm { const event = globalThis.event; psf = this.AFMakeNumber(psf); - let formatStr; + let value = this.AFMergeChange(event); + let formatStr, secondFormatStr; switch (psf) { case 0: formatStr = "99999"; @@ -572,11 +573,8 @@ class AForm { formatStr = "99999-9999"; break; case 2: - const value = this.AFMergeChange(event); - formatStr = - value.startsWith("(") || (value.length > 7 && /^\p{N}+$/.test(value)) - ? "(999) 999-9999" - : "999-9999"; + formatStr = "999-9999"; + secondFormatStr = "(999) 999-9999"; break; case 3: formatStr = "999-99-9999"; @@ -584,8 +582,36 @@ class AForm { default: throw new Error("Invalid psf in AFSpecial_Keystroke"); } + const formats = secondFormatStr + ? [formatStr, secondFormatStr] + : [formatStr]; + for (const format of formats) { + this.#AFSpecial_KeystrokeEx_helper(format, value, false); + if (event.rc) { + return; + } + event.rc = true; + } + + const re = /([-()]|\s)+/g; + value = value.replaceAll(re, ""); + for (const format of formats) { + this.#AFSpecial_KeystrokeEx_helper( + format.replaceAll(re, ""), + value, + false + ); + if (event.rc) { + return; + } + event.rc = true; + } - this.AFSpecial_KeystrokeEx(formatStr); + this.AFSpecial_KeystrokeEx( + ((secondFormatStr && value.match(/\d/g)) || []).length > 7 + ? secondFormatStr + : formatStr + ); } AFTime_FormatEx(cFormat) { diff --git a/test/unit/scripting_spec.js b/test/unit/scripting_spec.js index 6246120df4169..d570908662a90 100644 --- a/test/unit/scripting_spec.js +++ b/test/unit/scripting_spec.js @@ -1685,6 +1685,67 @@ describe("Scripting", function () { send_queue.delete(refId); }); + it("should validate a US phone number with digits and dashes (long) on a keystroke event", async () => { + const refId = getId(); + const data = { + objects: { + field: [ + { + id: refId, + value: "", + actions: { + Keystroke: [`AFSpecial_Keystroke(2);`], + }, + type: "text", + }, + ], + }, + appInfo: { language: "en-US", platform: "Linux x86_64" }, + calculationOrder: [], + dispatchEventName: "_dispatchMe", + }; + sandbox.createSandbox(data); + + let value = ""; + const changes = "123-456-7890"; + let i = 0; + + for (; i < changes.length; i++) { + const change = changes.charAt(i); + await sandbox.dispatchEventInSandbox({ + id: refId, + value, + change, + name: "Keystroke", + willCommit: false, + selStart: i, + selEnd: i, + }); + expect(send_queue.has(refId)).toEqual(true); + send_queue.delete(refId); + value += change; + } + + await sandbox.dispatchEventInSandbox({ + id: refId, + value, + change: "A", + name: "Keystroke", + willCommit: false, + selStart: i, + selEnd: i, + }); + expect(send_queue.has(refId)).toEqual(true); + expect(send_queue.get(refId)).toEqual({ + id: refId, + siblings: null, + value, + selRange: [i, i], + }); + + send_queue.delete(refId); + }); + it("should validate a US phone number (short) on a keystroke event", async () => { const refId = getId(); const data = {