import {Et2Tag} from "../Tag/Et2Tag"; import {assert, elementUpdated, fixture, html, oneEvent} from '@open-wc/testing'; import * as sinon from 'sinon'; import {inputBasicTests} from "../../Et2InputWidget/test/InputBasicTests"; import {Et2Select} from "../Et2Select"; /** * Test file for Etemplate webComponent Select * * In here we test just the simple, basic widget stuff. */ // Stub global egw for cssImage to find // @ts-ignore window.egw = { image: () => "", lang: i => i + "*", tooltipUnbind: () => {}, webserverUrl: "" }; let element : Et2Select; async function before() { // Create an element to test with, and wait until it's ready // @ts-ignore element = await fixture(html` `); // Stub egw() sinon.stub(element, "egw").returns(window.egw); await elementUpdated(element); return element; } describe("Select widget basics", () => { // Setup run before each test beforeEach(before); // Make sure it works it('is defined', () => { assert.instanceOf(element, Et2Select); }); it('has a label', async() => { element.set_label("Label set"); await elementUpdated(element); assert.equal(element.querySelector("[slot='label']").textContent, "Label set"); }) it("starts empty", () => { assert.notExists(element.querySelector("option"), "Static option not found in DOM"); assert.deepEqual(element.select_options, [], "Unexpected option(s)"); }) it("closes when losing focus", async() => { // WIP const blurSpy = sinon.spy(); element.addEventListener('sl-hide', blurSpy); const showPromise = new Promise(resolve => { element.addEventListener("sl-after-show", resolve); }); const hidePromise = new Promise(resolve => { element.addEventListener("sl-hide", resolve); }); await elementUpdated(element); element.focus(); await showPromise; await elementUpdated(element); element.blur(); await elementUpdated(element); await hidePromise; sinon.assert.calledOnce(blurSpy); // Check that it actually closed dropdown assert.isFalse(element.select?.hasAttribute("open")); }) }); describe("Multiple", () => { beforeEach(async() => { // Create an element to test with, and wait until it's ready // @ts-ignore element = await fixture(html` `); element.loadFromXML(element); element.set_value("one,two"); // Stub egw() sinon.stub(element, "egw").returns(window.egw); return element; }); it("Can remove tags", async() => { assert.equal(element.select.querySelectorAll("sl-option").length, 2, "Did not find options"); assert.sameMembers(element.value, ["one", "two"]); let tags = element.select.combobox.querySelectorAll('.select__tags et2-tag'); // Await tags to render 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']"); assert.exists(removeButton, "Could not find tag remove button"); removeButton.dispatchEvent(new Event("click")); await listener; // Wait for widget to update await element.updateComplete; tag_updates = [] element.select.combobox.querySelectorAll('et2-tag').forEach((t : Et2Tag) => 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"); }); }); inputBasicTests(async() => { const element = await before(); element.noLang = true; element.select_options = [{value: "", label: ""}]; return element }, "", "sl-select");