diff --git a/api/js/etemplate/Et2Email/Et2Email.ts b/api/js/etemplate/Et2Email/Et2Email.ts index 1070442e3e..ad223dbd3f 100644 --- a/api/js/etemplate/Et2Email/Et2Email.ts +++ b/api/js/etemplate/Et2Email/Et2Email.ts @@ -87,16 +87,7 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI // Parse string into array if(typeof value === 'string' && value.indexOf(',') !== -1) { - let val = value.split(','); - for(let n = 0; n < val.length - 1; n++) - { - while(val[n].indexOf('@') === -1 && n < val.length - 1) - { - val[n] += ',' + val[n + 1]; - val.splice(n + 1, 1); - } - } - return val; + return parseEmailsString(value, false); } return value; }, @@ -670,11 +661,8 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI { selection.deleteFromDocument(); } + let values = parseEmailsString(paste, this.allowPlaceholder); - let preg = this.allowPlaceholder ? IsEmail.EMAIL_PLACEHOLDER_PREG : IsEmail.EMAIL_PREG; - // Trim line start / end anchors off validation regex, make global - let regex = new RegExp(preg.toString().substring(2, preg.toString().length - 3), 'g'); - let values = paste.match(regex); if(values) { values.forEach(v => @@ -685,6 +673,7 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI // Update key to force Lit to redraw tags this._valueUID = this.egw()?.uid() ?? new Date().toISOString(); + this.dispatchEvent(new Event("change", {bubbles: true})); } } @@ -716,6 +705,7 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI if(this.addAddress(this._search.value.trim())) { this._search.value = ""; + this.dispatchEvent(new Event("change", {bubbles: true})); } else if(this._search.value) { @@ -773,6 +763,7 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI { this.open = false; this._search.value = ""; + this.dispatchEvent(new Event("change", {bubbles: true})); } if(event.key == "Tab") { @@ -963,6 +954,7 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI this._search.value = ""; this._search.focus(); this.requestUpdate("value"); + this.dispatchEvent(new Event("change", {bubbles: true})); if(this._close_on_select) { this.open = false; @@ -977,6 +969,7 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI let index = this.value.indexOf(event.originalValue); this.value[index] = event.target.value; this.requestUpdate(); + this.dispatchEvent(new Event("change", {bubbles: true})); } if(event.target.current) { @@ -990,6 +983,7 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI const index = this.value.indexOf(value); this.value.splice(index, 1); this.requestUpdate("value"); + this.dispatchEvent(new Event("change", {bubbles: true})); } tagsTemplate() @@ -1184,4 +1178,19 @@ export class Et2Email extends Et2InputWidget(LitElement) implements SearchMixinI } // @ts-ignore TypeScript is not recognizing that this widget is a LitElement -customElements.define("et2-email", Et2Email); \ No newline at end of file +customElements.define("et2-email", Et2Email); + +/** + * Parse string that may contain multiple comma separated email addresses into an array + * + * @param {string} value + * @returns {string[]} + * @protected + */ +function parseEmailsString(value : string, allowPlaceholder = false) : string[] +{ + let preg = allowPlaceholder ? IsEmail.EMAIL_PLACEHOLDER_PREG : IsEmail.EMAIL_PREG; + // Trim line start / end anchors off validation regex, make global + let regex = new RegExp(preg.toString().substring(2, preg.toString().length - 3), 'g'); + return value.match(regex); +} \ No newline at end of file diff --git a/api/js/etemplate/Et2Email/test/Et2Email.test.ts b/api/js/etemplate/Et2Email/test/Et2Email.test.ts index 49c0c4980a..e4559ed93a 100644 --- a/api/js/etemplate/Et2Email/test/Et2Email.test.ts +++ b/api/js/etemplate/Et2Email/test/Et2Email.test.ts @@ -2,6 +2,8 @@ import {assert, elementUpdated, fixture, html, oneEvent} from '@open-wc/testing' import * as sinon from 'sinon'; import {inputBasicTests} from "../../Et2InputWidget/test/InputBasicTests"; import {Et2Email} from "../Et2Email"; +import {Et2EmailTag} from "../../Et2Select/Tag/Et2EmailTag"; +import {waitForEvent} from "../../Et2Widget/event"; /** * Test file for Etemplate webComponent Select @@ -10,11 +12,24 @@ import {Et2Email} from "../Et2Email"; */ // Stub global egw for cssImage to find // @ts-ignore +let uid = 0; +const testSuggestions = [ + {value: "suggestion.1@example.com", label: "Suggestion 1"}, + {value: "suggestion.2@example.com", label: "Suggestion 2"} +]; window.egw = { + ajaxUrl: () => "", + app: () => "addressbook", + decodePath: (_path : string) => _path, image: () => "", + jsonq: () => Promise.resolve({}), lang: i => i + "*", + link: i => i, + preference: i => "", + request: () => Promise.resolve(testSuggestions), tooltipUnbind: () => {}, - webserverUrl: "" + webserverUrl: "", + uid: () => {return "" + (uid++);} }; let element : Et2Email; @@ -52,7 +67,13 @@ describe("Email widget basics", () => await elementUpdated(element); assert.equal(element.querySelector("[slot='label']").textContent, "Label set"); - }) + }); + + it("textbox gets focus when widget is focused", async() => + { + element.focus(); + assert.equal(element.shadowRoot.activeElement, element._search, "Search textbox did not get focus when widget got focus"); + }); it("closes when losing focus", async() => { @@ -67,8 +88,9 @@ describe("Email widget basics", () => { element.addEventListener("sl-hide", resolve); }); + await elementUpdated(element); - element.focus(); + element.show(); await showPromise; await elementUpdated(element); @@ -82,7 +104,57 @@ describe("Email widget basics", () => // Check that it actually closed dropdown assert.isFalse(element.hasAttribute("open")); - }) + }); + + it("blurring widget accepts current text", async() => + { + const value = "valid@example.com"; + element.focus(); + element._search.value = value; + element.blur(); + await elementUpdated(element); + + assert.sameMembers(element.value, [value], "Valid email was not accepted on blur"); + }); +}); +describe("Suggestions", () => +{ // Setup run before each test + beforeEach(before); + + it("clicking accepts suggestion", async() => + { + await elementUpdated(element); + // Start the search + element.focus(); + element.startSearch(); + debugger; + await waitForEvent(element, "sl-after-show"); + + // Click the first one + element._listbox.querySelector('sl-option').dispatchEvent(new MouseEvent("mouseup", {bubbles: true})) + await elementUpdated(element); + // Check the value + assert.sameMembers(element.value, [testSuggestions[0].value]); + }); + + it("tab accepts top suggestion", async() => + { + element.focus(); + element.startSearch(); + await waitForEvent(element, "sl-after-show"); + + // No match between what they typed and the suggestion - no + element._search.dispatchEvent(new KeyboardEvent("keydown", {key: "Tab"})); + await elementUpdated(element); + assert.sameMembers(element.value, []); + + // Partial match with current suggestion, take it + element.focus(); + element._search.value = "sugg"; + element._search.dispatchEvent(new KeyboardEvent("keydown", {key: "Tab"})); + await elementUpdated(element); + assert.sameMembers(element.value, [testSuggestions[0].value]); + }); }); describe("Tags", () => @@ -107,21 +179,11 @@ describe("Tags", () => { assert.equal(element._tags.length, 2, "Did not find tags"); - // Await tags to render - /* TODO - let tag_updates = [] - element.select.combobox.querySelectorAll("et2-tag").forEach((t : Et2Tag) => tag_updates.push(t.updateComplete)); - await Promise.all(tag_updates); - - assert.equal(tags.length, 2); - assert.equal(tags[0].value, "one"); - assert.equal(tags[1].value, "two"); -*/ // Set up listener const listener = oneEvent(element, "change"); // Click to remove first tag - let removeButton = tags[0].shadowRoot.querySelector("[part='remove-button']"); + let removeButton = element._tags[0].shadowRoot.querySelector("[part='remove-button']"); assert.exists(removeButton, "Could not find tag remove button"); removeButton.dispatchEvent(new Event("click")); @@ -129,14 +191,13 @@ describe("Tags", () => // Wait for widget to update await element.updateComplete; - tag_updates = [] - element.select.combobox.querySelectorAll('et2-tag').forEach((t : Et2Tag) => tag_updates.push(t.updateComplete)); + let tag_updates = [] + element._tags.forEach((t : Et2EmailTag) => tag_updates.push(t.updateComplete)); await Promise.all(tag_updates); // Check - assert.sameMembers(element.value, ["two"], "Removing tag did not remove value"); - tags = element.select.combobox.querySelectorAll('.select__tags et2-tag'); - assert.equal(tags.length, 1, "Removed tag is still there"); + assert.sameMembers(element.value, ["two@example.com"], "Removing tag did not remove value"); + assert.equal(element._tags.length, 1, "Removed tag is still there"); }); }); @@ -146,4 +207,4 @@ inputBasicTests(async() => const element = await before(); element.noLang = true; return element -}, "", "sl-select"); \ No newline at end of file +}, "", "input"); \ No newline at end of file diff --git a/api/js/etemplate/Et2Select/Tag/Et2EmailTag.ts b/api/js/etemplate/Et2Select/Tag/Et2EmailTag.ts index c6376e6405..c9cf2615a3 100644 --- a/api/js/etemplate/Et2Select/Tag/Et2EmailTag.ts +++ b/api/js/etemplate/Et2Select/Tag/Et2EmailTag.ts @@ -181,7 +181,7 @@ export class Et2EmailTag extends Et2Tag e.stopPropagation(); let extra = { - 'presets[email]': this.value + 'presets[email]': this.value ?? "" }; this.egw().open('', 'addressbook', 'add', extra); diff --git a/api/js/etemplate/Et2Select/test/Et2EmailTag.test.ts b/api/js/etemplate/Et2Select/test/Et2EmailTag.test.ts index 77a1af7c3e..e81966b651 100644 --- a/api/js/etemplate/Et2Select/test/Et2EmailTag.test.ts +++ b/api/js/etemplate/Et2Select/test/Et2EmailTag.test.ts @@ -76,7 +76,8 @@ describe('Et2EmailTag', () => assert.equal(extra['presets[email]'], 'test@example.com'); } }; - component.handleMouseDown(new MouseEvent('click')); + debugger; + component.shadowRoot.querySelector("et2-button-icon").dispatchEvent(new MouseEvent('click')); }); it('should open addressbook CRM on avatar click', async() =>