From 16464fb49f5f4e54b433d93561f9882d458ac6a9 Mon Sep 17 00:00:00 2001 From: Lukas Spirig Date: Tue, 10 Dec 2024 12:30:35 +0100 Subject: [PATCH 1/6] docs: fix link in the contributing guideline (#3283) --- docs/CONTRIBUTING.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/CONTRIBUTING.md b/docs/CONTRIBUTING.md index ef0514dd618..aa8d2798822 100644 --- a/docs/CONTRIBUTING.md +++ b/docs/CONTRIBUTING.md @@ -54,7 +54,7 @@ You can file new issues by providing the above information [here](https://github ### Submitting a Pull Request (PR) -- Search [GitHub](https://github.com/angular/components/pulls) for an open or closed PR +- Search [GitHub](https://github.com/sbb-design-systems/lyne-components/pulls) for an open or closed PR that relates to your submission. You don't want to duplicate effort. - Make your changes in a new git branch: From 2364424e4c6f05e2d64eea3061f0e017a5a99791 Mon Sep 17 00:00:00 2001 From: Lukas Spirig Date: Tue, 10 Dec 2024 12:31:09 +0100 Subject: [PATCH 2/6] docs: fix package names (#3282) --- README.md | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 78687f624e1..72de69a23e6 100644 --- a/README.md +++ b/README.md @@ -22,12 +22,12 @@ Lyne Design Tokens and Lyne Components are available for developers and designer ## đŸ”— Packages -| Package | Description | -| ---------------------------------- | --------------------------------------------------------------------------- | -| `@lyne-esta/elements` | Web components built on top of the Lyne Design System | -| `@lyne-esta/elements-experimental` | Web components that do not yet align with our architecture or testing goals | -| `@lyne-esta/react` | React wrappers for `@lyne-esta/elements` | -| `@lyne-esta/react-experimental` | React wrappers for `@lyne-esta/elements-experimental` | +| Package | Description | +| -------------------------------------- | --------------------------------------------------------------------------- | +| `@sbb-esta/lyne-elements` | Web components built on top of the Lyne Design System | +| `@sbb-esta/lyne-elements-experimental` | Web components that do not yet align with our architecture or testing goals | +| `@sbb-esta/lyne-react` | React wrappers for `@sbb-esta/lyne-elements` | +| `@sbb-esta/lyne-react-experimental` | React wrappers for `@sbb-esta/lyne-elements-experimental` | - [NPM Packages](https://www.npmjs.com/search?q=%40sbb-esta%2Flyne-) From f838e888f5178236192e9f172a83a6f3aeef599d Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Tue, 10 Dec 2024 15:12:11 +0000 Subject: [PATCH 3/6] chore(deps): update dependency lint-staged to v15.2.11 --- package.json | 2 +- yarn.lock | 36 ++++++++++++------------------------ 2 files changed, 13 insertions(+), 25 deletions(-) diff --git a/package.json b/package.json index 69575f1809f..f6af96995fe 100644 --- a/package.json +++ b/package.json @@ -122,7 +122,7 @@ "glob": "11.0.0", "globals": "15.13.0", "husky": "9.1.7", - "lint-staged": "15.2.10", + "lint-staged": "15.2.11", "lit-analyzer": "2.0.3", "madge": "8.0.0", "npm-run-all2": "7.0.1", diff --git a/yarn.lock b/yarn.lock index f5fbbcc8db6..f7397727db7 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3514,7 +3514,7 @@ debounce@1.2.1, debounce@^1.2.0: resolved "https://registry.yarnpkg.com/debounce/-/debounce-1.2.1.tgz#38881d8f4166a5c5848020c11827b834bcb3e0a5" integrity sha512-XRRe6Glud4rd/ZGQfiV1ruXSfbvfJedlV9Y6zOlP+2K04vBYiJEte6stfFkCP03aMnY5tsipamumUjL14fofug== -debug@4, debug@^4.0.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2, debug@^4.3.4, debug@^4.3.6, debug@^4.3.7: +debug@4, debug@^4.0.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2, debug@^4.3.4, debug@^4.3.6, debug@^4.3.7, debug@~4.4.0: version "4.4.0" resolved "https://registry.yarnpkg.com/debug/-/debug-4.4.0.tgz#2b3f2aea2ffeb776477460267377dc8710faba8a" integrity sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA== @@ -3535,13 +3535,6 @@ debug@^3.1.0, debug@^3.2.7: dependencies: ms "^2.1.1" -debug@~4.3.6: - version "4.3.7" - resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.7.tgz#87945b4151a011d76d95a198d7111c865c360a52" - integrity sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ== - dependencies: - ms "^2.1.3" - decode-named-character-reference@^1.0.0: version "1.0.2" resolved "https://registry.yarnpkg.com/decode-named-character-reference/-/decode-named-character-reference-1.0.2.tgz#daabac9690874c394c81e4162a0304b35d824f0e" @@ -5653,7 +5646,7 @@ lighthouse-logger@^1.0.0: debug "^2.6.9" marky "^1.2.2" -lilconfig@~3.1.2: +lilconfig@~3.1.3: version "3.1.3" resolved "https://registry.yarnpkg.com/lilconfig/-/lilconfig-3.1.3.tgz#a1bcfd6257f9585bf5ae14ceeebb7b559025e4c4" integrity sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw== @@ -5663,23 +5656,23 @@ lines-and-columns@^1.1.6: resolved "https://registry.yarnpkg.com/lines-and-columns/-/lines-and-columns-1.2.4.tgz#eca284f75d2965079309dc0ad9255abb2ebc1632" integrity sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg== -lint-staged@15.2.10: - version "15.2.10" - resolved "https://registry.yarnpkg.com/lint-staged/-/lint-staged-15.2.10.tgz#92ac222f802ba911897dcf23671da5bb80643cd2" - integrity sha512-5dY5t743e1byO19P9I4b3x8HJwalIznL5E1FWYnU6OWw33KxNBSLAc6Cy7F2PsFEO8FKnLwjwm5hx7aMF0jzZg== +lint-staged@15.2.11: + version "15.2.11" + resolved "https://registry.yarnpkg.com/lint-staged/-/lint-staged-15.2.11.tgz#e88440982f4a4c1d55a9a7a839259ec3806bd81b" + integrity sha512-Ev6ivCTYRTGs9ychvpVw35m/bcNDuBN+mnTeObCL5h+boS5WzBEC6LHI4I9F/++sZm1m+J2LEiy0gxL/R9TBqQ== dependencies: chalk "~5.3.0" commander "~12.1.0" - debug "~4.3.6" + debug "~4.4.0" execa "~8.0.1" - lilconfig "~3.1.2" - listr2 "~8.2.4" + lilconfig "~3.1.3" + listr2 "~8.2.5" micromatch "~4.0.8" pidtree "~0.6.0" string-argv "~0.3.2" - yaml "~2.5.0" + yaml "~2.6.1" -listr2@~8.2.4: +listr2@~8.2.5: version "8.2.5" resolved "https://registry.yarnpkg.com/listr2/-/listr2-8.2.5.tgz#5c9db996e1afeb05db0448196d3d5f64fec2593d" integrity sha512-iyAZCeyD+c1gPyE9qpFu8af0Y+MRtmKOncdGoA2S5EY8iFq99dmmvkNnHiWo+pj0s7yH7l3KPIgee77tKpXPWQ== @@ -8972,16 +8965,11 @@ yaml-eslint-parser@^1.2.1: lodash "^4.17.21" yaml "^2.0.0" -yaml@^2.0.0: +yaml@^2.0.0, yaml@~2.6.1: version "2.6.1" resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.6.1.tgz#42f2b1ba89203f374609572d5349fb8686500773" integrity sha512-7r0XPzioN/Q9kXBro/XPnA6kznR73DHq+GXh5ON7ZozRO6aMjbmiBuKste2wslTFkC5d1dw0GooOCepZXJ2SAg== -yaml@~2.5.0: - version "2.5.1" - resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.5.1.tgz#c9772aacf62cb7494a95b0c4f1fb065b563db130" - integrity sha512-bLQOjaX/ADgQ20isPJRvF0iRUHIxVhYvr53Of7wGcWlO2jvtUlH5m87DsmulFVxRpNLOnI4tB6p/oh8D7kpn9Q== - yargs-parser@^21.1.1: version "21.1.1" resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-21.1.1.tgz#9096bceebf990d21bb31fa9516e0ede294a77d35" From 4cb2cdd61983c70b4c42189641059487de996009 Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Tue, 10 Dec 2024 21:44:28 +0000 Subject: [PATCH 4/6] chore(deps): update dependency @custom-elements-manifest/analyzer to v0.10.4 --- package.json | 2 +- yarn.lock | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/package.json b/package.json index f6af96995fe..ac578e1cc81 100644 --- a/package.json +++ b/package.json @@ -73,7 +73,7 @@ "devDependencies": { "@commitlint/cli": "19.6.0", "@commitlint/config-conventional": "19.6.0", - "@custom-elements-manifest/analyzer": "0.10.3", + "@custom-elements-manifest/analyzer": "0.10.4", "@custom-elements-manifest/to-markdown": "0.1.0", "@eslint/eslintrc": "3.2.0", "@eslint/js": "9.16.0", diff --git a/yarn.lock b/yarn.lock index f7397727db7..fd795b5fd88 100644 --- a/yarn.lock +++ b/yarn.lock @@ -223,10 +223,10 @@ resolved "https://registry.yarnpkg.com/@csstools/selector-specificity/-/selector-specificity-5.0.0.tgz#037817b574262134cabd68fc4ec1a454f168407b" integrity sha512-PCqQV3c4CoVm3kdPhyeZ07VmBRdH2EpMFA/pd9OASpOEC3aXNGoqPDAZ80D0cLpMBxnmk0+yNhGsEx31hq7Gtw== -"@custom-elements-manifest/analyzer@0.10.3": - version "0.10.3" - resolved "https://registry.yarnpkg.com/@custom-elements-manifest/analyzer/-/analyzer-0.10.3.tgz#3b12957514475b24672ed08f48e007c7e5f018ea" - integrity sha512-e2Ax59vK9sNedmDlPqZS11L54iAlKSjOJuv5etpTy5SygLBW3GcUtocHZm8wO013L0griTPpgWB0tuV7/JXy5A== +"@custom-elements-manifest/analyzer@0.10.4": + version "0.10.4" + resolved "https://registry.yarnpkg.com/@custom-elements-manifest/analyzer/-/analyzer-0.10.4.tgz#7bd063cd1249a75b13d5bcc6ed4302dd72dd1f88" + integrity sha512-hse8o20Jd82BwWank29/J9OC4PmSTwUoEmll3LEjDF3WLY/Lc8g3TUYSib/3GARCS8Q5myT2RPqEWfRa+6bkIg== dependencies: "@custom-elements-manifest/find-dependencies" "^0.0.5" "@github/catalyst" "^1.6.0" From 6da37fc78c0864cf802950c519b2a274611acdb9 Mon Sep 17 00:00:00 2001 From: Jeri Peier Date: Wed, 11 Dec 2024 08:39:06 +0100 Subject: [PATCH 5/6] fix: improve handling of animation events for zero duration (#3284) This PR enables updating Playwright and therefore provides a solution for the chromium bug https://issues.chromium.org/issues/373506511 Only for autocomplete and select, some code improvements were needed to support zero animation duration. --- .../alert-group.snapshot.spec.snap.js | 4 +- .../alert/alert-group/alert-group.spec.ts | 9 +- .../__snapshots__/alert.snapshot.spec.snap.js | 4 +- src/elements/alert/alert/alert.scss | 4 +- src/elements/alert/alert/alert.spec.ts | 18 ++ src/elements/alert/alert/alert.ts | 20 ++- .../autocomplete-grid.spec.ts | 78 +++++++-- .../autocomplete-grid/autocomplete-grid.ts | 12 +- .../autocomplete/autocomplete-base-element.ts | 38 +++-- .../autocomplete/autocomplete.spec.ts | 50 +++++- src/elements/autocomplete/autocomplete.ts | 10 -- src/elements/button/common/button-common.scss | 2 +- src/elements/calendar/calendar.scss | 2 +- src/elements/clock/clock.scss | 2 +- .../container/sticky-bar/sticky-bar.scss | 6 +- .../container/sticky-bar/sticky-bar.spec.ts | 15 ++ .../container/sticky-bar/sticky-bar.ts | 9 +- src/elements/core/dom.ts | 1 + src/elements/core/dom/animation.ts | 10 ++ src/elements/core/mixins/panel-common.scss | 2 +- .../core/styles/mixins/animation.scss | 4 +- src/elements/core/styles/mixins/buttons.scss | 2 +- src/elements/core/styles/mixins/card.scss | 2 +- src/elements/core/styles/mixins/panel.scss | 2 +- .../dialog.snapshot.spec.snap.js | 2 +- src/elements/dialog/dialog/dialog.spec.ts | 23 +++ src/elements/dialog/dialog/dialog.ts | 92 +++++----- .../expansion-panel-header.scss | 2 +- .../expansion-panel/expansion-panel.scss | 2 +- .../expansion-panel/expansion-panel.spec.ts | 16 ++ .../expansion-panel/expansion-panel.ts | 39 +++-- .../common/file-selector-common.scss | 2 +- .../flip-card-details/flip-card-details.scss | 2 +- .../flip-card-summary/flip-card-summary.scss | 2 +- .../flip-card/flip-card/flip-card.scss | 16 +- .../form-field/form-field/form-field.scss | 2 +- .../form-field/form-field/form-field.spec.ts | 2 +- .../form-field/form-field/form-field.ts | 7 +- src/elements/header/common/header-action.scss | 2 +- src/elements/header/header/header.scss | 2 +- src/elements/image/image.scss | 2 +- .../loading-indicator-circle.scss | 2 +- .../loading-indicator/loading-indicator.scss | 2 +- src/elements/map-container/map-container.scss | 2 +- src/elements/menu/menu/menu.scss | 2 +- src/elements/menu/menu/menu.spec.ts | 19 +++ src/elements/menu/menu/menu.ts | 78 ++++++--- .../navigation/common/navigation-action.scss | 3 +- .../navigation-marker/navigation-marker.scss | 2 +- .../navigation-section.scss | 4 +- .../navigation-section.spec.ts | 15 ++ .../navigation-section/navigation-section.ts | 66 ++++++-- .../navigation/navigation/navigation.scss | 4 +- .../navigation/navigation/navigation.spec.ts | 25 +++ .../navigation/navigation/navigation.ts | 77 ++++++--- .../notification.snapshot.spec.snap.js | 10 +- src/elements/notification/notification.scss | 4 +- .../notification/notification.spec.ts | 112 ++++++++----- src/elements/notification/notification.ts | 33 +++- .../overlay.snapshot.spec.snap.js | 3 +- src/elements/overlay/overlay-base-element.ts | 8 + src/elements/overlay/overlay.scss | 2 +- src/elements/overlay/overlay.spec.ts | 16 ++ src/elements/overlay/overlay.ts | 74 ++++---- .../paginator/paginator/paginator.scss | 2 +- src/elements/popover/popover/popover.spec.ts | 18 ++ src/elements/popover/popover/popover.ts | 76 ++++++--- .../common/radio-button-common.scss | 2 +- src/elements/select/select.spec.ts | 158 ++++++++++++++++-- src/elements/select/select.ts | 52 +++--- .../selection-expansion-panel.scss | 2 +- .../selection-expansion-panel.spec.ts | 19 ++- .../selection-expansion-panel.ts | 40 +++-- src/elements/slider/slider.scss | 2 +- .../stepper/step-label/step-label.scss | 2 +- src/elements/stepper/step/step.scss | 2 +- src/elements/stepper/stepper/stepper.scss | 2 +- src/elements/tabs/tab-group/tab-group.scss | 2 +- src/elements/tabs/tab-label/tab-label.scss | 2 +- src/elements/tag/tag/tag.scss | 2 +- .../teaser-product/teaser-product.scss | 2 +- src/elements/teaser/teaser.scss | 2 +- src/elements/toast/toast.scss | 2 +- src/elements/toast/toast.spec.ts | 24 ++- src/elements/toast/toast.ts | 49 ++++-- src/elements/toggle-check/toggle-check.scss | 2 +- src/elements/toggle/toggle/toggle.scss | 2 +- 87 files changed, 1132 insertions(+), 418 deletions(-) create mode 100644 src/elements/core/dom/animation.ts diff --git a/src/elements/alert/alert-group/__snapshots__/alert-group.snapshot.spec.snap.js b/src/elements/alert/alert-group/__snapshots__/alert-group.snapshot.spec.snap.js index aa506dc2e55..1602b0d3a64 100644 --- a/src/elements/alert/alert-group/__snapshots__/alert-group.snapshot.spec.snap.js +++ b/src/elements/alert/alert-group/__snapshots__/alert-group.snapshot.spec.snap.js @@ -9,7 +9,7 @@ snapshots["sbb-alert-group renders DOM"] = > @@ -42,7 +42,7 @@ snapshots["sbb-alert-group renders with slotted DOM"] = diff --git a/src/elements/alert/alert-group/alert-group.spec.ts b/src/elements/alert/alert-group/alert-group.spec.ts index 59297a14309..8f800b5e831 100644 --- a/src/elements/alert/alert-group/alert-group.spec.ts +++ b/src/elements/alert/alert-group/alert-group.spec.ts @@ -17,6 +17,10 @@ describe(`sbb-alert-group`, () => { const accessibilityTitle = 'Disruptions'; const accessibilityTitleLevel = '3'; + const alertOpenedEventSpy = new EventSpy(SbbAlertElement.events.didOpen, null, { + capture: true, + }); + // Given sbb-alert-group with two alerts element = await fixture(html` { Second `); + const emptySpy = new EventSpy(SbbAlertGroupElement.events.empty); const alert1 = element.querySelector('sbb-alert#alert1')!; const alert2 = element.querySelector('sbb-alert#alert2')!; const alert1ClosedEventSpy = new EventSpy(SbbAlertElement.events.didClose, alert1); const alert2ClosedEventSpy = new EventSpy(SbbAlertElement.events.didClose, alert2); - await new EventSpy(SbbAlertElement.events.didOpen, alert1).calledOnce(); - await new EventSpy(SbbAlertElement.events.didOpen, alert2).calledOnce(); + // Wait until both alerts are opened + await alertOpenedEventSpy.calledTimes(2); // Then two alerts should be rendered and accessibility title should be displayed expect(element.querySelectorAll('sbb-alert').length).to.be.equal(2); diff --git a/src/elements/alert/alert/__snapshots__/alert.snapshot.spec.snap.js b/src/elements/alert/alert/__snapshots__/alert.snapshot.spec.snap.js index f894df87bd9..48b21e0daee 100644 --- a/src/elements/alert/alert/__snapshots__/alert.snapshot.spec.snap.js +++ b/src/elements/alert/alert/__snapshots__/alert.snapshot.spec.snap.js @@ -4,7 +4,7 @@ export const snapshots = {}; snapshots["sbb-alert should render default properties DOM"] = ` @@ -77,7 +77,7 @@ snapshots["sbb-alert should render default properties Shadow DOM"] = snapshots["sbb-alert should render customized properties DOM"] = ` { expect(didCloseSpy.count).to.be.equal(1); }); + it('should fire animation events with non-zero animation duration', async () => { + const didOpenSpy = new EventSpy(SbbAlertElement.events.didOpen, null, { capture: true }); + const didCloseSpy = new EventSpy(SbbAlertElement.events.didClose, null, { capture: true }); + + const alert: SbbAlertElement = await fixture( + html` + Interruption + `, + ); + + await didOpenSpy.calledOnce(); + + alert.close(); + + await didCloseSpy.calledOnce(); + expect(didCloseSpy.count).to.be.equal(1); + }); + it('should respect canceled willClose event', async () => { const didOpenSpy = new EventSpy(SbbAlertElement.events.didOpen, null, { capture: true }); const willCloseSpy = new EventSpy(SbbAlertElement.events.willClose, null, { capture: true }); diff --git a/src/elements/alert/alert/alert.ts b/src/elements/alert/alert/alert.ts index 2efd00cf377..5115d1bf9a6 100644 --- a/src/elements/alert/alert/alert.ts +++ b/src/elements/alert/alert/alert.ts @@ -4,7 +4,7 @@ import { customElement, property } from 'lit/decorators.js'; import { SbbOpenCloseBaseElement } from '../../core/base-elements.js'; import { SbbLanguageController } from '../../core/controllers.js'; import { forceType } from '../../core/decorators.js'; -import { isLean } from '../../core/dom.js'; +import { isLean, isZeroAnimationDuration } from '../../core/dom.js'; import { i18nCloseAlert } from '../../core/i18n.js'; import { SbbIconNameMixin } from '../../icon.js'; import type { SbbTitleLevel } from '../../title.js'; @@ -75,14 +75,26 @@ class SbbAlertElement extends SbbIconNameMixin(SbbOpenCloseBaseElement) { /** Open the alert. */ public open(): void { - this.willOpen.emit(); this.state = 'opening'; + this.willOpen.emit(); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `opened` state. + if (this._isZeroAnimationDuration()) { + this._handleOpening(); + } } /** Close the alert. */ public close(): void { if (this.state === 'opened' && this.willClose.emit()) { this.state = 'closing'; + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `closed` state. + if (this._isZeroAnimationDuration()) { + this._handleClosing(); + } } } @@ -92,6 +104,10 @@ class SbbAlertElement extends SbbIconNameMixin(SbbOpenCloseBaseElement) { this.open(); } + private _isZeroAnimationDuration(): boolean { + return isZeroAnimationDuration(this, '--sbb-alert-animation-duration'); + } + private _onAnimationEnd(event: AnimationEvent): void { if (this.state === 'opening' && event.animationName === 'open-opacity') { this._handleOpening(); diff --git a/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.spec.ts b/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.spec.ts index fef7f2791de..15984653626 100644 --- a/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.spec.ts +++ b/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.spec.ts @@ -1,4 +1,4 @@ -import { assert, expect } from '@open-wc/testing'; +import { assert, aTimeout, expect } from '@open-wc/testing'; import { sendKeys, sendMouse } from '@web/test-runner-commands'; import { html } from 'lit/static-html.js'; @@ -25,9 +25,9 @@ describe(`sbb-autocomplete-grid`, () => { - Option 1 + + Option 1 + { - Option 2 + + Option 2 + { expect(input).to.have.attribute('role', 'combobox'); expect(input).to.have.attribute('aria-autocomplete', 'list'); expect(input).to.have.attribute('aria-haspopup', 'grid'); - expect(input).to.have.attribute('aria-controls', 'myAutocomplete'); - expect(input).to.have.attribute('aria-owns', 'myAutocomplete'); + expect(input).to.have.attribute('aria-controls', element.id); + expect(input).to.have.attribute('aria-owns', element.id); expect(input).to.have.attribute('aria-expanded', 'false'); }); }); @@ -83,12 +83,14 @@ describe(`sbb-autocomplete-grid`, () => { expect(element).not.to.have.attribute('autocomplete-origin-borderless'); + const id = element.shadowRoot!.querySelector('.sbb-autocomplete__options')!.id; + expect(input).to.have.attribute('autocomplete', 'off'); expect(input).to.have.attribute('role', 'combobox'); expect(input).to.have.attribute('aria-autocomplete', 'list'); expect(input).to.have.attribute('aria-haspopup', 'grid'); - expect(input).to.have.attribute('aria-controls', 'sbb-autocomplete-grid-11'); - expect(input).to.have.attribute('aria-owns', 'sbb-autocomplete-grid-11'); + expect(input).to.have.attribute('aria-controls', id); + expect(input).to.have.attribute('aria-owns', id); expect(input).to.have.attribute('aria-expanded', 'false'); }); }); @@ -99,7 +101,8 @@ describe(`sbb-autocomplete-grid`, () => { const willCloseEventSpy = new EventSpy(SbbAutocompleteGridElement.events.willClose, element); const didCloseEventSpy = new EventSpy(SbbAutocompleteGridElement.events.didClose, element); - input.click(); + input.focus(); + await willOpenEventSpy.calledOnce(); expect(willOpenEventSpy.count).to.be.equal(1); @@ -145,15 +148,34 @@ describe(`sbb-autocomplete-grid`, () => { expect(input).to.have.attribute('aria-expanded', 'false'); }); + it('opens and closes with non-zero animation duration', async () => { + element.style.setProperty('--sbb-options-panel-animation-duration', '1ms'); + const didOpenEventSpy = new EventSpy(SbbAutocompleteGridElement.events.didOpen, element); + const didCloseEventSpy = new EventSpy(SbbAutocompleteGridElement.events.didClose, element); + + input.focus(); + + await didOpenEventSpy.calledOnce(); + expect(input).to.have.attribute('aria-expanded', 'true'); + + await sendKeys({ press: 'Escape' }); + await didCloseEventSpy.calledOnce(); + + expect(input).to.have.attribute('aria-expanded', 'false'); + }); + it('select by mouse', async () => { const didOpenEventSpy = new EventSpy(SbbAutocompleteGridElement.events.didOpen, element); const optionSelectedEventSpy = new EventSpy( SbbAutocompleteGridOptionElement.events.optionSelected, ); + const inputEventSpy = new EventSpy('input', input); + const changeEventSpy = new EventSpy('change', input); const optTwo = element.querySelector('#option-2')!; input.focus(); await didOpenEventSpy.calledOnce(); + const positionRect = optTwo.getBoundingClientRect(); await sendMouse({ @@ -165,8 +187,11 @@ describe(`sbb-autocomplete-grid`, () => { }); await waitForLitRender(element); + expect(inputEventSpy.count).to.be.equal(1); + expect(changeEventSpy.count).to.be.equal(1); expect(optionSelectedEventSpy.count).to.be.equal(1); expect(optionSelectedEventSpy.firstEvent!.target).to.have.property('id', 'option-2'); + expect(document.activeElement).to.be.equal(input); }); it('select button and get related option', async () => { @@ -187,7 +212,7 @@ describe(`sbb-autocomplete-grid`, () => { await clickSpy.calledOnce(); expect(clickSpy.count).to.be.equal(1); expect( - (clickSpy.firstEvent!.target as SbbAutocompleteGridButtonElement).option!.textContent, + (clickSpy.firstEvent!.target as SbbAutocompleteGridButtonElement).option!.textContent!.trim(), ).to.be.equal('Option 1'); expect( (clickSpy.firstEvent!.target as SbbAutocompleteGridButtonElement).option!.value, @@ -244,8 +269,10 @@ describe(`sbb-autocomplete-grid`, () => { const optionSelectedEventSpy = new EventSpy( SbbAutocompleteGridOptionElement.events.optionSelected, ); - const optOne = element.querySelector('#option-1'); - const optTwo = element.querySelector('#option-2'); + const inputEventSpy = new EventSpy('input', input); + const changeEventSpy = new EventSpy('change', input); + const optOne = element.querySelector('#option-1'); + const optTwo = element.querySelector('#option-2'); const keydownSpy = new EventSpy('keydown', input); input.focus(); @@ -269,11 +296,27 @@ describe(`sbb-autocomplete-grid`, () => { expect(optTwo).not.to.have.attribute('data-active'); expect(optTwo).to.have.attribute('selected'); + expect(inputEventSpy.count).to.be.equal(1); + expect(changeEventSpy.count).to.be.equal(1); expect(optionSelectedEventSpy.count).to.be.equal(1); expect(input).to.have.attribute('aria-expanded', 'false'); expect(input).not.to.have.attribute('aria-activedescendant'); }); + it('should not close on disabled option click', async () => { + const didOpenEventSpy = new EventSpy(SbbAutocompleteGridElement.events.didOpen, element); + const optOne = element.querySelector('#option-1')!; + optOne.disabled = true; + + input.focus(); + await didOpenEventSpy.calledOnce(); + + optOne.click(); + + await aTimeout(0); + expect(element).to.have.attribute('data-state', 'opened'); + }); + it('opens and select button with keyboard', async () => { const didOpenEventSpy = new EventSpy(SbbAutocompleteGridElement.events.didOpen, element); const clickSpy = new EventSpy('click'); @@ -307,10 +350,11 @@ describe(`sbb-autocomplete-grid`, () => { await sendKeys({ press: 'Enter' }); await clickSpy.calledTimes(2); expect(clickSpy.count).to.be.equal(2); + expect(element).to.have.attribute('data-state', 'opened'); }); it('should stay closed when disabled', async () => { - input.setAttribute('disabled', ''); + input.toggleAttribute('disabled', true); input.focus(); await waitForLitRender(element); @@ -326,7 +370,7 @@ describe(`sbb-autocomplete-grid`, () => { }); it('should stay closed when readonly', async () => { - input.setAttribute('readonly', ''); + input.toggleAttribute('readonly', true); input.focus(); await waitForLitRender(element); diff --git a/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.ts b/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.ts index 249bc26c8f6..2fdfff62989 100644 --- a/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.ts +++ b/src/elements/autocomplete-grid/autocomplete-grid/autocomplete-grid.ts @@ -6,7 +6,7 @@ import { hostAttributes } from '../../core/decorators.js'; import { isSafari } from '../../core/dom.js'; import { setAriaComboBoxAttributes } from '../../core/overlay.js'; import type { SbbDividerElement } from '../../divider.js'; -import type { SbbOptGroupElement, SbbOptionElement } from '../../option.js'; +import type { SbbOptGroupElement } from '../../option.js'; import type { SbbAutocompleteGridButtonElement } from '../autocomplete-grid-button.js'; import { SbbAutocompleteGridOptionElement } from '../autocomplete-grid-option.js'; import type { SbbAutocompleteGridRowElement } from '../autocomplete-grid-row.js'; @@ -54,16 +54,6 @@ class SbbAutocompleteGridElement extends SbbAutocompleteBaseElement { ); } - protected onOptionClick(event: MouseEvent): void { - if ( - (event.target as Element).localName !== 'sbb-autocomplete-grid-option' || - (event.target as SbbOptionElement).disabled - ) { - return; - } - this.close(); - } - public override connectedCallback(): void { super.connectedCallback(); const signal = this.abort.signal; diff --git a/src/elements/autocomplete/autocomplete-base-element.ts b/src/elements/autocomplete/autocomplete-base-element.ts index 613efb10a22..07f456bb370 100644 --- a/src/elements/autocomplete/autocomplete-base-element.ts +++ b/src/elements/autocomplete/autocomplete-base-element.ts @@ -12,7 +12,7 @@ import { ref } from 'lit/directives/ref.js'; import { SbbOpenCloseBaseElement } from '../core/base-elements.js'; import { SbbConnectedAbortController } from '../core/controllers.js'; import { forceType } from '../core/decorators.js'; -import { findReferencedElement, isSafari } from '../core/dom.js'; +import { findReferencedElement, isSafari, isZeroAnimationDuration } from '../core/dom.js'; import { SbbNegativeMixin, SbbHydrationMixin } from '../core/mixins.js'; import { isEventOnElement, @@ -85,7 +85,6 @@ export abstract class SbbAutocompleteBaseElement extends SbbNegativeMixin( protected abstract selectByKeyboard(event: KeyboardEvent): void; protected abstract setNextActiveOption(event: KeyboardEvent): void; protected abstract resetActiveElement(): void; - protected abstract onOptionClick(event: MouseEvent): void; /** Opens the autocomplete. */ public open(): void { @@ -103,6 +102,12 @@ export abstract class SbbAutocompleteBaseElement extends SbbNegativeMixin( this.state = 'opening'; this._setOverlayPosition(); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `opened` state. + if (this._isZeroAnimationDuration()) { + this._handleOpening(); + } } /** Closes the autocomplete. */ @@ -116,6 +121,16 @@ export abstract class SbbAutocompleteBaseElement extends SbbNegativeMixin( this.state = 'closing'; this._openPanelEventsController.abort(); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `closed` state. + if (this._isZeroAnimationDuration()) { + this._handleClosing(); + } + } + + private _isZeroAnimationDuration(): boolean { + return isZeroAnimationDuration(this, '--sbb-options-panel-animation-duration'); } public override connectedCallback(): void { @@ -123,7 +138,6 @@ export abstract class SbbAutocompleteBaseElement extends SbbNegativeMixin( if (ariaRoleOnHost) { this.id ||= this.overlayId; } - const signal = this.abort.signal; const formField = this.closest('sbb-form-field') ?? this.closest('[data-form-field]'); if (formField) { @@ -134,8 +148,6 @@ export abstract class SbbAutocompleteBaseElement extends SbbNegativeMixin( this._componentSetup(); } this.syncNegative(); - - this.addEventListener('click', (e: MouseEvent) => this.onOptionClick(e), { signal }); } protected override willUpdate(changedProperties: PropertyValues): void { @@ -187,6 +199,7 @@ export abstract class SbbAutocompleteBaseElement extends SbbNegativeMixin( // Manually trigger the change events this.triggerElement.dispatchEvent(new Event('change', { bubbles: true })); this.triggerElement.dispatchEvent(new InputEvent('input', { bubbles: true, composed: true })); + this.triggerElement.focus(); } this.close(); @@ -309,26 +322,27 @@ export abstract class SbbAutocompleteBaseElement extends SbbNegativeMixin( ); } - /** On open/close animation end. - * In rare cases it can be that the animationEnd event is triggered twice. - * To avoid entering a corrupt state, exit when state is not expected. + /** + * On open/close animation end. + * In rare cases it can be that the animationEnd event is triggered twice. + * To avoid entering a corrupt state, exit when state is not expected. */ private _onAnimationEnd(event: AnimationEvent): void { if (event.animationName === 'open' && this.state === 'opening') { - this._onOpenAnimationEnd(); + this._handleOpening(); } else if (event.animationName === 'close' && this.state === 'closing') { - this._onCloseAnimationEnd(); + this._handleClosing(); } } - private _onOpenAnimationEnd(): void { + private _handleOpening(): void { this.state = 'opened'; this._attachOpenPanelEvents(); this.triggerElement?.setAttribute('aria-expanded', 'true'); this.didOpen.emit(); } - private _onCloseAnimationEnd(): void { + private _handleClosing(): void { this.state = 'closed'; this.triggerElement?.setAttribute('aria-expanded', 'false'); this.resetActiveElement(); diff --git a/src/elements/autocomplete/autocomplete.spec.ts b/src/elements/autocomplete/autocomplete.spec.ts index 8aa4ae589ed..76257118f05 100644 --- a/src/elements/autocomplete/autocomplete.spec.ts +++ b/src/elements/autocomplete/autocomplete.spec.ts @@ -1,4 +1,4 @@ -import { assert, expect } from '@open-wc/testing'; +import { assert, aTimeout, expect } from '@open-wc/testing'; import { sendKeys, sendMouse } from '@web/test-runner-commands'; import { html } from 'lit/static-html.js'; @@ -40,8 +40,8 @@ describe(`sbb-autocomplete`, () => { expect(input).to.have.attribute('role', 'combobox'); expect(input).to.have.attribute('aria-autocomplete', 'list'); expect(input).to.have.attribute('aria-haspopup', 'listbox'); - expect(input).to.have.attribute('aria-controls', 'myAutocomplete'); - expect(input).to.have.attribute('aria-owns', 'myAutocomplete'); + expect(input).to.have.attribute('aria-controls', element.id); + expect(input).to.have.attribute('aria-owns', element.id); expect(input).to.have.attribute('aria-expanded', 'false'); }); }); @@ -53,12 +53,14 @@ describe(`sbb-autocomplete`, () => { expect(element).not.to.have.attribute('autocomplete-origin-borderless'); + const id = element.shadowRoot!.querySelector('.sbb-autocomplete__options')!.id; + expect(input).to.have.attribute('autocomplete', 'off'); expect(input).to.have.attribute('role', 'combobox'); expect(input).to.have.attribute('aria-autocomplete', 'list'); expect(input).to.have.attribute('aria-haspopup', 'listbox'); - expect(input).to.have.attribute('aria-controls', 'sbb-autocomplete-8'); - expect(input).to.have.attribute('aria-owns', 'sbb-autocomplete-8'); + expect(input).to.have.attribute('aria-controls', id); + expect(input).to.have.attribute('aria-owns', id); expect(input).to.have.attribute('aria-expanded', 'false'); }); }); @@ -69,7 +71,8 @@ describe(`sbb-autocomplete`, () => { const willCloseEventSpy = new EventSpy(SbbAutocompleteElement.events.willClose, element); const didCloseEventSpy = new EventSpy(SbbAutocompleteElement.events.didClose, element); - input.click(); + input.focus(); + await willOpenEventSpy.calledOnce(); expect(willOpenEventSpy.count).to.be.equal(1); @@ -115,6 +118,22 @@ describe(`sbb-autocomplete`, () => { expect(input).to.have.attribute('aria-expanded', 'false'); }); + it('opens and closes with non-zero animation duration', async () => { + element.style.setProperty('--sbb-options-panel-animation-duration', '1ms'); + const didOpenEventSpy = new EventSpy(SbbAutocompleteElement.events.didOpen, element); + const didCloseEventSpy = new EventSpy(SbbAutocompleteElement.events.didClose, element); + + input.focus(); + + await didOpenEventSpy.calledOnce(); + expect(input).to.have.attribute('aria-expanded', 'true'); + + await sendKeys({ press: 'Escape' }); + await didCloseEventSpy.calledOnce(); + + expect(input).to.have.attribute('aria-expanded', 'false'); + }); + it('select by mouse', async () => { const didOpenEventSpy = new EventSpy(SbbAutocompleteElement.events.didOpen, element); const optionSelectedEventSpy = new EventSpy(SbbOptionElement.events.optionSelected); @@ -140,6 +159,7 @@ describe(`sbb-autocomplete`, () => { expect(changeEventSpy.count).to.be.equal(1); expect(optionSelectedEventSpy.count).to.be.equal(1); expect(optionSelectedEventSpy.firstEvent!.target).to.have.property('id', 'option-2'); + expect(document.activeElement).to.be.equal(input); }); it('opens and select with keyboard', async () => { @@ -148,8 +168,8 @@ describe(`sbb-autocomplete`, () => { const optionSelectedEventSpy = new EventSpy(SbbOptionElement.events.optionSelected); const inputEventSpy = new EventSpy('input', input); const changeEventSpy = new EventSpy('change', input); - const optOne = element.querySelector('#option-1'); - const optTwo = element.querySelector('#option-2'); + const optOne = element.querySelector('#option-1'); + const optTwo = element.querySelector('#option-2'); const keydownSpy = new EventSpy('keydown', input); input.focus(); @@ -180,6 +200,20 @@ describe(`sbb-autocomplete`, () => { expect(input).not.to.have.attribute('aria-activedescendant'); }); + it('should not close on disabled option click', async () => { + const didOpenEventSpy = new EventSpy(SbbAutocompleteElement.events.didOpen, element); + const optOne = element.querySelector('#option-1')!; + optOne.disabled = true; + + input.focus(); + await didOpenEventSpy.calledOnce(); + + optOne.click(); + + await aTimeout(0); + expect(element).to.have.attribute('data-state', 'opened'); + }); + it('should stay closed when disabled', async () => { input.toggleAttribute('disabled', true); diff --git a/src/elements/autocomplete/autocomplete.ts b/src/elements/autocomplete/autocomplete.ts index 696790057ca..27a09dd50a1 100644 --- a/src/elements/autocomplete/autocomplete.ts +++ b/src/elements/autocomplete/autocomplete.ts @@ -42,16 +42,6 @@ class SbbAutocompleteElement extends SbbAutocompleteBaseElement { return Array.from(this.querySelectorAll?.('sbb-option') ?? []); } - protected onOptionClick(event: MouseEvent): void { - if ( - (event.target as Element).localName !== 'sbb-option' || - (event.target as SbbOptionElement).disabled - ) { - return; - } - this.close(); - } - public override connectedCallback(): void { super.connectedCallback(); const signal = this.abort.signal; diff --git a/src/elements/button/common/button-common.scss b/src/elements/button/common/button-common.scss index f07885d73aa..254bd06e44c 100644 --- a/src/elements/button/common/button-common.scss +++ b/src/elements/button/common/button-common.scss @@ -31,7 +31,7 @@ $active: ':active, [data-active]'; --sbb-button-border-radius: var(--sbb-border-radius-infinity); --sbb-button-min-height: var(--sbb-size-element-m); --sbb-button-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-button-transition-easing-function: var(--sbb-animation-easing); diff --git a/src/elements/calendar/calendar.scss b/src/elements/calendar/calendar.scss index 49fcd37e20d..68307924bf8 100644 --- a/src/elements/calendar/calendar.scss +++ b/src/elements/calendar/calendar.scss @@ -22,7 +22,7 @@ --sbb-calendar-cell-disabled-height: #{sbb.px-to-rem-build(1.5)}; --sbb-calendar-cell-disabled-width: #{sbb.px-to-rem-build(25.5)}; --sbb-calendar-cell-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-calendar-cell-transition-easing-function: var(--sbb-animation-easing); diff --git a/src/elements/clock/clock.scss b/src/elements/clock/clock.scss index 88fc7bae088..8c3ba96bf15 100644 --- a/src/elements/clock/clock.scss +++ b/src/elements/clock/clock.scss @@ -36,7 +36,7 @@ } .sbb-clock__hand-minutes { - transition: transform var(--sbb-disable-animation-zero-duration, 0.2s) + transition: transform var(--sbb-disable-animation-duration, 0.2s) cubic-bezier(0.4, 2.08, 0.55, 0.44); @supports not (aspect-ratio: 1 / 1) { diff --git a/src/elements/container/sticky-bar/sticky-bar.scss b/src/elements/container/sticky-bar/sticky-bar.scss index 9053717fe9c..be3b7b135b9 100644 --- a/src/elements/container/sticky-bar/sticky-bar.scss +++ b/src/elements/container/sticky-bar/sticky-bar.scss @@ -16,15 +16,15 @@ $intersector-overlapping: 1px; --sbb-sticky-bar-border-radius: var(--sbb-border-radius-8x); --sbb-sticky-bar-animation-easing: var(--sbb-animation-easing); --sbb-sticky-bar-fade-in-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-5x) ); --sbb-sticky-bar-fade-out-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-sticky-bar-slide-vertically-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); --sbb-sticky-bar-slide-vertically-animation-easing: ease-out; diff --git a/src/elements/container/sticky-bar/sticky-bar.spec.ts b/src/elements/container/sticky-bar/sticky-bar.spec.ts index a88d86fac06..b092b013e0e 100644 --- a/src/elements/container/sticky-bar/sticky-bar.spec.ts +++ b/src/elements/container/sticky-bar/sticky-bar.spec.ts @@ -139,6 +139,21 @@ describe(`sbb-sticky-bar`, () => { expect(willStickSpy.count).to.be.equal(1); expect(didStickSpy.count).to.be.equal(0); }); + + it('works with non-zero animation duration', async () => { + stickyBar.style.setProperty('--sbb-sticky-bar-slide-vertically-animation-duration', '1ms'); + + stickyBar.unstick(); + await didUnstickSpy.calledOnce(); + + stickyBar.stick(); + + await willStickSpy.calledOnce(); + await didStickSpy.calledOnce(); + + expect(willStickSpy.count).to.be.equal(1); + expect(didStickSpy.count).to.be.equal(1); + }); }); it('is settled when content is not long enough', async () => { diff --git a/src/elements/container/sticky-bar/sticky-bar.ts b/src/elements/container/sticky-bar/sticky-bar.ts index 9e26a9e2e39..b87d1743ed3 100644 --- a/src/elements/container/sticky-bar/sticky-bar.ts +++ b/src/elements/container/sticky-bar/sticky-bar.ts @@ -9,6 +9,7 @@ import { import { customElement, property } from 'lit/decorators.js'; import { hostAttributes } from '../../core/decorators.js'; +import { isZeroAnimationDuration } from '../../core/dom.js'; import { EventEmitter } from '../../core/eventing.js'; import { SbbUpdateSchedulerMixin } from '../../core/mixins.js'; @@ -114,6 +115,10 @@ class SbbStickyBarElement extends SbbUpdateSchedulerMixin(LitElement) { this._observer.observe(this); } + private _isZeroAnimationDuration(): boolean { + return isZeroAnimationDuration(this, '--sbb-sticky-bar-slide-vertically-animation-duration'); + } + private _detectStickyState(entry: IntersectionObserverEntry): void { this.toggleAttribute('data-initialized', true); @@ -152,7 +157,7 @@ class SbbStickyBarElement extends SbbUpdateSchedulerMixin(LitElement) { } this._state = 'sticking'; - if (!this.hasAttribute('data-sticking')) { + if (!this.hasAttribute('data-sticking') || this._isZeroAnimationDuration()) { this._stickyCallback(); } } @@ -165,7 +170,7 @@ class SbbStickyBarElement extends SbbUpdateSchedulerMixin(LitElement) { this._state = 'unsticking'; - if (!this.hasAttribute('data-sticking')) { + if (!this.hasAttribute('data-sticking') || this._isZeroAnimationDuration()) { this._unstickyCallback(); } } diff --git a/src/elements/core/dom.ts b/src/elements/core/dom.ts index 472d088fa28..b5b1cdc86a3 100644 --- a/src/elements/core/dom.ts +++ b/src/elements/core/dom.ts @@ -1,3 +1,4 @@ +export * from './dom/animation.js'; export * from './dom/breakpoint.js'; export * from './dom/find-referenced-element.js'; export * from './dom/host-context.js'; diff --git a/src/elements/core/dom/animation.ts b/src/elements/core/dom/animation.ts new file mode 100644 index 00000000000..dd97e383f64 --- /dev/null +++ b/src/elements/core/dom/animation.ts @@ -0,0 +1,10 @@ +import { isServer } from 'lit'; + +export function isZeroAnimationDuration(element: HTMLElement, cssVariableName: string): boolean { + if (isServer) { + return true; + } + const animationDuration = getComputedStyle(element).getPropertyValue(cssVariableName); + + return parseFloat(animationDuration) === 0; +} diff --git a/src/elements/core/mixins/panel-common.scss b/src/elements/core/mixins/panel-common.scss index fbd819aab24..b160b486cf2 100644 --- a/src/elements/core/mixins/panel-common.scss +++ b/src/elements/core/mixins/panel-common.scss @@ -20,7 +20,7 @@ --sbb-selection-panel-input-padding: var(--sbb-spacing-responsive-xs) var(--sbb-spacing-responsive-xxs); --sbb-selection-panel-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); --sbb-selection-panel-cursor: pointer; diff --git a/src/elements/core/styles/mixins/animation.scss b/src/elements/core/styles/mixins/animation.scss index 38bff8a2003..ba54aa0bc31 100644 --- a/src/elements/core/styles/mixins/animation.scss +++ b/src/elements/core/styles/mixins/animation.scss @@ -3,11 +3,9 @@ // ---------------------------------------------------------------------------------------------------- @mixin disable-animation { - --sbb-disable-animation-duration: 0.1ms; - --sbb-disable-animation-zero-duration: 0s; + --sbb-disable-animation-duration: 0s; } @mixin enable-animation { --sbb-disable-animation-duration: initial; - --sbb-disable-animation-zero-duration: initial; } diff --git a/src/elements/core/styles/mixins/buttons.scss b/src/elements/core/styles/mixins/buttons.scss index 9b7e8a426b5..e1af1525df8 100644 --- a/src/elements/core/styles/mixins/buttons.scss +++ b/src/elements/core/styles/mixins/buttons.scss @@ -67,7 +67,7 @@ $active: ':active, [data-active]'; --sbb-button-border-disabled-style: dashed; --sbb-button-border-radius: var(--sbb-border-radius-infinity); --sbb-button-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-button-transition-easing-function: var(--sbb-animation-easing); diff --git a/src/elements/core/styles/mixins/card.scss b/src/elements/core/styles/mixins/card.scss index 11671e4404b..805489d131e 100644 --- a/src/elements/core/styles/mixins/card.scss +++ b/src/elements/core/styles/mixins/card.scss @@ -13,7 +13,7 @@ --sbb-card-background-color: var(--sbb-color-white); --sbb-card-border-radius: var(--sbb-border-radius-4x); --sbb-card-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-1x) ); --sbb-card-animation-easing: var(--sbb-animation-easing); diff --git a/src/elements/core/styles/mixins/panel.scss b/src/elements/core/styles/mixins/panel.scss index b52efdf09dc..b43310896cd 100644 --- a/src/elements/core/styles/mixins/panel.scss +++ b/src/elements/core/styles/mixins/panel.scss @@ -19,7 +19,7 @@ --sbb-panel-padding-inline: var(--sbb-spacing-responsive-m); --sbb-panel-gap: var(--sbb-spacing-responsive-xs); --sbb-panel-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); --sbb-panel-animation-easing: var(--sbb-animation-easing); diff --git a/src/elements/dialog/dialog/__snapshots__/dialog.snapshot.spec.snap.js b/src/elements/dialog/dialog/__snapshots__/dialog.snapshot.spec.snap.js index ff3ca62d7fd..7c120a2cd24 100644 --- a/src/elements/dialog/dialog/__snapshots__/dialog.snapshot.spec.snap.js +++ b/src/elements/dialog/dialog/__snapshots__/dialog.snapshot.spec.snap.js @@ -2,7 +2,7 @@ export const snapshots = {}; snapshots["sbb-dialog renders an open dialog DOM"] = -` +` { expect(element).to.have.attribute('data-state', 'closed'); }); + it('opens and closes the overlay with non-zero animation duration', async () => { + element.style.setProperty('--sbb-dialog-animation-duration', '1ms'); + + const willClose = new EventSpy(SbbDialogElement.events.willClose, element); + const didClose = new EventSpy(SbbDialogElement.events.didClose, element); + + await openDialog(element); + + element.close(); + await waitForLitRender(element); + + await willClose.calledOnce(); + expect(willClose.count).to.be.equal(1); + await waitForLitRender(element); + + await didClose.calledOnce(); + expect(didClose.count).to.be.equal(1); + await waitForLitRender(element); + + expect(element).to.have.attribute('data-state', 'closed'); + }); + it('does not close the dialog on other overlay click', async () => { await setViewport({ width: 900, height: 600 }); element = await fixture(html` diff --git a/src/elements/dialog/dialog/dialog.ts b/src/elements/dialog/dialog/dialog.ts index 166a293d35f..a84dd0a41e3 100644 --- a/src/elements/dialog/dialog/dialog.ts +++ b/src/elements/dialog/dialog/dialog.ts @@ -4,7 +4,7 @@ import { customElement, property } from 'lit/decorators.js'; import { html } from 'lit/static-html.js'; import { getFirstFocusableElement, setModalityOnNextFocus } from '../../core/a11y.js'; -import { isBreakpoint } from '../../core/dom.js'; +import { isBreakpoint, isZeroAnimationDuration } from '../../core/dom.js'; import { overlayRefs, SbbOverlayBaseElement } from '../../overlay.js'; import type { SbbDialogActionsElement } from '../dialog-actions.js'; import type { SbbDialogTitleElement } from '../dialog-title.js'; @@ -82,6 +82,55 @@ class SbbDialogElement extends SbbOverlayBaseElement { // Disable scrolling for content below the dialog this.scrollHandler.disableScroll(); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `opened` state. + if (this.isZeroAnimationDuration()) { + this._handleOpening(); + } + } + + protected isZeroAnimationDuration(): boolean { + return isZeroAnimationDuration(this, '--sbb-dialog-animation-duration'); + } + + protected handleClosing(): void { + this._setHideHeaderDataAttribute(false); + this._dialogContentElement?.scrollTo(0, 0); + this.state = 'closed'; + this.inertController.deactivate(); + setModalityOnNextFocus(this.lastFocusedElement); + // Manually focus last focused element + this.lastFocusedElement?.focus(); + this.openOverlayController?.abort(); + this.focusHandler.disconnect(); + if (this._dialogContentElement) { + this._dialogContentResizeObserver.unobserve(this._dialogContentElement); + } + this.removeInstanceFromGlobalCollection(); + // Enable scrolling for content below the dialog if no dialog is open + if (!overlayRefs.length) { + this.scrollHandler.enableScroll(); + } + this.didClose.emit({ + returnValue: this.returnValue, + closeTarget: this.overlayCloseElement, + }); + } + + private _handleOpening(): void { + this.state = 'opened'; + this.didOpen.emit(); + this.inertController.activate(); + this.attachOpenOverlayEvents(); + this.setOverlayFocus(); + // Use timeout to read label after focused element + setTimeout(() => + this.setAriaLiveRefContent( + this.accessibilityLabel || this._dialogTitleElement?.innerText.trim(), + ), + ); + this.focusHandler.trap(this); } public override connectedCallback(): void { @@ -129,40 +178,9 @@ class SbbDialogElement extends SbbOverlayBaseElement { // To avoid entering a corrupt state, exit when state is not expected. protected onOverlayAnimationEnd(event: AnimationEvent): void { if (event.animationName === 'open' && this.state === 'opening') { - this.state = 'opened'; - this.didOpen.emit(); - this.inertController.activate(); - this.attachOpenOverlayEvents(); - this.setOverlayFocus(); - // Use timeout to read label after focused element - setTimeout(() => - this.setAriaLiveRefContent( - this.accessibilityLabel || this._dialogTitleElement?.innerText.trim(), - ), - ); - this.focusHandler.trap(this); + this._handleOpening(); } else if (event.animationName === 'close' && this.state === 'closing') { - this._setHideHeaderDataAttribute(false); - this._dialogContentElement?.scrollTo(0, 0); - this.state = 'closed'; - this.inertController.deactivate(); - setModalityOnNextFocus(this.lastFocusedElement); - // Manually focus last focused element - this.lastFocusedElement?.focus(); - this.openOverlayController?.abort(); - this.focusHandler.disconnect(); - if (this._dialogContentElement) { - this._dialogContentResizeObserver.unobserve(this._dialogContentElement); - } - this.removeInstanceFromGlobalCollection(); - // Enable scrolling for content below the dialog if no dialog is open - if (!overlayRefs.length) { - this.scrollHandler.enableScroll(); - } - this.didClose.emit({ - returnValue: this.returnValue, - closeTarget: this.overlayCloseElement, - }); + this.handleClosing(); } } @@ -285,11 +303,7 @@ class SbbDialogElement extends SbbOverlayBaseElement { protected override render(): TemplateResult { return html`
-
this.onOverlayAnimationEnd(event)} - class="sbb-dialog" - id=${this._dialogId} - > +
this.closeOnSbbOverlayCloseClick(event)} class="sbb-dialog__wrapper" diff --git a/src/elements/expansion-panel/expansion-panel-header/expansion-panel-header.scss b/src/elements/expansion-panel/expansion-panel-header/expansion-panel-header.scss index e610cf55217..99bcac17e75 100644 --- a/src/elements/expansion-panel/expansion-panel-header/expansion-panel-header.scss +++ b/src/elements/expansion-panel/expansion-panel-header/expansion-panel-header.scss @@ -82,7 +82,7 @@ } .sbb-expansion-panel-header__toggle-icon { - transition: transform var(--sbb-disable-animation-zero-duration, var(--sbb-animation-duration-4x)); + transition: transform var(--sbb-disable-animation-duration, var(--sbb-animation-duration-4x)); :host([aria-expanded]:not([aria-expanded='false'])) & { transform: rotate(-180deg); diff --git a/src/elements/expansion-panel/expansion-panel/expansion-panel.scss b/src/elements/expansion-panel/expansion-panel/expansion-panel.scss index ac9c6a633bb..66d12f374ce 100644 --- a/src/elements/expansion-panel/expansion-panel/expansion-panel.scss +++ b/src/elements/expansion-panel/expansion-panel/expansion-panel.scss @@ -11,7 +11,7 @@ $open-anim-opacity-to: 1; :host { --sbb-expansion-panel-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); --sbb-expansion-panel-background-color: var(--sbb-color-white); diff --git a/src/elements/expansion-panel/expansion-panel/expansion-panel.spec.ts b/src/elements/expansion-panel/expansion-panel/expansion-panel.spec.ts index d8bff065b54..bf9feef6e23 100644 --- a/src/elements/expansion-panel/expansion-panel/expansion-panel.spec.ts +++ b/src/elements/expansion-panel/expansion-panel/expansion-panel.spec.ts @@ -129,4 +129,20 @@ describe(`sbb-expansion-panel`, () => { expect(header).to.have.attribute('data-size', 'l'); expect(content).to.have.attribute('data-size', 'l'); }); + + it('should fire animation events with non-zero animation duration', async () => { + element.style.setProperty('--sbb-expansion-panel-animation-duration', '1ms'); + + const didOpenSpy = new EventSpy(SbbExpansionPanelElement.events.didOpen, element); + const didCloseSpy = new EventSpy(SbbExpansionPanelElement.events.didClose, element); + + element.expanded = true; + + await didOpenSpy.calledOnce(); + + element.expanded = false; + + await didCloseSpy.calledOnce(); + expect(didCloseSpy.count).to.be.equal(1); + }); }); diff --git a/src/elements/expansion-panel/expansion-panel/expansion-panel.ts b/src/elements/expansion-panel/expansion-panel/expansion-panel.ts index b1dcc2d5cd3..ee0d18f70e0 100644 --- a/src/elements/expansion-panel/expansion-panel/expansion-panel.ts +++ b/src/elements/expansion-panel/expansion-panel/expansion-panel.ts @@ -5,7 +5,7 @@ import { html, unsafeStatic } from 'lit/static-html.js'; import { SbbConnectedAbortController } from '../../core/controllers.js'; import { forceType } from '../../core/decorators.js'; -import { isLean } from '../../core/dom.js'; +import { isLean, isZeroAnimationDuration } from '../../core/dom.js'; import { EventEmitter } from '../../core/eventing.js'; import type { SbbOpenedClosedState } from '../../core/interfaces.js'; import { SbbHydrationMixin } from '../../core/mixins.js'; @@ -156,25 +156,46 @@ class SbbExpansionPanelElement extends SbbHydrationMixin(LitElement) { this._contentRef?.setAttribute('aria-hidden', String(!this.expanded)); if (this.expanded) { - this._open(!this._initialized); + this._open(); } else if (this._state === 'opened') { this._close(); } } - private _open(skipAnimation = false): void { + private _open(): void { this._state = 'opening'; this._willOpen.emit(); - if (skipAnimation) { - this._state = 'opened'; - this._didOpen.emit(); + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `opened` state. + if (!this._initialized || this._isZeroAnimationDuration()) { + this._handleOpening(); } } private _close(): void { this._state = 'closing'; this._willClose.emit(); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `closed` state. + if (this._isZeroAnimationDuration()) { + this._handleClosing(); + } + } + + private _isZeroAnimationDuration(): boolean { + return isZeroAnimationDuration(this, '--sbb-expansion-panel-animation-duration'); + } + + private _handleOpening(): void { + this._state = 'opened'; + this._didOpen.emit(); + } + + private _handleClosing(): void { + this._state = 'closed'; + this._didClose.emit(); } private _updateDisabledOnHeader(newDisabledValue: boolean): void { @@ -220,11 +241,9 @@ class SbbExpansionPanelElement extends SbbHydrationMixin(LitElement) { private _onAnimationEnd(event: AnimationEvent): void { if (event.animationName === 'open-opacity' && this._state === 'opening') { - this._state = 'opened'; - this._didOpen.emit(); + this._handleOpening(); } else if (event.animationName === 'close' && this._state === 'closing') { - this._state = 'closed'; - this._didClose.emit(); + this._handleClosing(); } } diff --git a/src/elements/file-selector/common/file-selector-common.scss b/src/elements/file-selector/common/file-selector-common.scss index 1cb1651112d..89fe9ba8c08 100644 --- a/src/elements/file-selector/common/file-selector-common.scss +++ b/src/elements/file-selector/common/file-selector-common.scss @@ -9,7 +9,7 @@ --sbb-file-selector-background-color: var(--sbb-color-white); --sbb-file-selector-border-color: var(--sbb-color-cloud); --sbb-file-selector-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-file-selector-transition-easing-function: var(--sbb-animation-easing); diff --git a/src/elements/flip-card/flip-card-details/flip-card-details.scss b/src/elements/flip-card/flip-card-details/flip-card-details.scss index 2d8343758eb..7ee3d180bbb 100644 --- a/src/elements/flip-card/flip-card-details/flip-card-details.scss +++ b/src/elements/flip-card/flip-card-details/flip-card-details.scss @@ -7,7 +7,7 @@ --sbb-flip-card-details-opacity: 0; --sbb-flip-card-details-translate-y: var(--sbb-spacing-fixed-2x); --sbb-flip-card-details-transition-delay: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-5x) ); --sbb-flip-card-details-padding: var(--sbb-spacing-responsive-s); diff --git a/src/elements/flip-card/flip-card-summary/flip-card-summary.scss b/src/elements/flip-card/flip-card-summary/flip-card-summary.scss index 64699826c49..9d4e48486d3 100644 --- a/src/elements/flip-card/flip-card-summary/flip-card-summary.scss +++ b/src/elements/flip-card/flip-card-summary/flip-card-summary.scss @@ -56,7 +56,7 @@ ::slotted(*:not([slot='image'])) { transform: translateY(var(--sbb-flip-card-translate-title-y-hover, #{sbb.px-to-rem-build(0)})); - transition: transform var(--sbb-disable-animation-zero-duration, var(--sbb-animation-duration-2x)) + transition: transform var(--sbb-disable-animation-duration, var(--sbb-animation-duration-2x)) var(--sbb-animation-easing); } diff --git a/src/elements/flip-card/flip-card/flip-card.scss b/src/elements/flip-card/flip-card/flip-card.scss index 745afa79fc6..878f430a027 100644 --- a/src/elements/flip-card/flip-card/flip-card.scss +++ b/src/elements/flip-card/flip-card/flip-card.scss @@ -9,15 +9,15 @@ --sbb-flip-card-border-radius: var(--sbb-border-radius-4x); --sbb-flip-card-min-height: #{sbb.px-to-rem-build(280)}; --sbb-flip-card-summary-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-5x) ); --sbb-flip-card-summary-transition-delay: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-flip-card-details-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); @@ -39,7 +39,7 @@ :host([data-flipped]) { --sbb-flip-card-toggle-icon-transform: rotate(45deg); --sbb-flip-card-details-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-flip-card-summary-transition-delay: 0s; @@ -116,8 +116,7 @@ // Use this large border radius to improve the appearance of the expanding dark background. border-radius: #{sbb.px-to-rem-build(256)}; - transition: var(--sbb-disable-animation-zero-duration, var(--sbb-animation-duration-3x)) - ease-out; + transition: var(--sbb-disable-animation-duration, var(--sbb-animation-duration-3x)) ease-out; :host([data-flipped]) & { opacity: 1; @@ -126,10 +125,7 @@ width: 100%; height: 100%; border-radius: var(--sbb-flip-card-border-radius); - transition-duration: var( - --sbb-disable-animation-zero-duration, - var(--sbb-animation-duration-5x) - ); + transition-duration: var(--sbb-disable-animation-duration, var(--sbb-animation-duration-5x)); } } } diff --git a/src/elements/form-field/form-field/form-field.scss b/src/elements/form-field/form-field/form-field.scss index 628ec29c40a..c58469fdd1f 100644 --- a/src/elements/form-field/form-field/form-field.scss +++ b/src/elements/form-field/form-field/form-field.scss @@ -361,7 +361,7 @@ will-change: transform, font-size; transition: { - duration: var(--sbb-disable-animation-zero-duration, var(--sbb-animation-duration-2x)); + duration: var(--sbb-disable-animation-duration, var(--sbb-animation-duration-2x)); timing-function: var(--sbb-animation-easing); property: transform, font-size; } diff --git a/src/elements/form-field/form-field/form-field.spec.ts b/src/elements/form-field/form-field/form-field.spec.ts index ae11694c55c..eca84d83c97 100644 --- a/src/elements/form-field/form-field/form-field.spec.ts +++ b/src/elements/form-field/form-field/form-field.spec.ts @@ -296,7 +296,7 @@ describe(`sbb-form-field`, () => { label.click(); await waitForLitRender(element); - expect(select).to.have.attribute('data-state', 'opening'); + expect(select).to.have.attribute('data-state', 'opened'); }); it('should focus select on form field click readonly', async () => { diff --git a/src/elements/form-field/form-field/form-field.ts b/src/elements/form-field/form-field/form-field.ts index 9f7e2a759a9..c1d3762236b 100644 --- a/src/elements/form-field/form-field/form-field.ts +++ b/src/elements/form-field/form-field/form-field.ts @@ -50,7 +50,7 @@ class SbbFormFieldElement extends SbbNegativeMixin(SbbHydrationMixin(LitElement) 'sbb-slider', ]; // List of elements that should not focus input on click - private readonly _excludedFocusElements = ['button', 'sbb-popover']; + private readonly _excludedFocusElements = ['button', 'sbb-popover', 'sbb-option']; private readonly _floatingLabelSupportedInputElements = [ 'input', @@ -168,7 +168,10 @@ class SbbFormFieldElement extends SbbNegativeMixin(SbbHydrationMixin(LitElement) return; } - if (this._input?.localName === 'sbb-select') { + if ( + this._input?.localName === 'sbb-select' && + (event.target as HTMLElement).localName !== 'sbb-select' + ) { this._input.click(); this._input.focus(); } else if ((event.target as Element).localName !== 'label') { diff --git a/src/elements/header/common/header-action.scss b/src/elements/header/common/header-action.scss index 6aedcf7a26e..9f80ec33d9e 100644 --- a/src/elements/header/common/header-action.scss +++ b/src/elements/header/common/header-action.scss @@ -16,7 +16,7 @@ --sbb-header-action-min-height: var(--sbb-size-element-s); --sbb-header-action-min-width: var(--sbb-header-action-min-height); --sbb-header-action-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-2x) ); --sbb-header-action-transition-easing: var(--sbb-animation-easing); diff --git a/src/elements/header/header/header.scss b/src/elements/header/header/header.scss index 444e94c1bf6..ad0883ad603 100644 --- a/src/elements/header/header/header.scss +++ b/src/elements/header/header/header.scss @@ -14,7 +14,7 @@ --sbb-signet-height: #{sbb.px-to-rem-build(16)}; --sbb-header-position: fixed; --sbb-header-transition-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-6x) ); --sbb-header-inset-inline-end: 0; diff --git a/src/elements/image/image.scss b/src/elements/image/image.scss index 79d62eb6eca..30b5a40530d 100644 --- a/src/elements/image/image.scss +++ b/src/elements/image/image.scss @@ -7,7 +7,7 @@ --sbb-image-border-radius: var(--sbb-border-radius-4x); --sbb-image-aspect-ratio: auto; --sbb-image-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); --sbb-image-object-fit: cover; diff --git a/src/elements/loading-indicator-circle/loading-indicator-circle.scss b/src/elements/loading-indicator-circle/loading-indicator-circle.scss index 0f4f57929a4..d28e21c7b42 100644 --- a/src/elements/loading-indicator-circle/loading-indicator-circle.scss +++ b/src/elements/loading-indicator-circle/loading-indicator-circle.scss @@ -9,7 +9,7 @@ --sbb-loading-indicator-circle-color: var(--sbb-color-red); --sbb-loading-indicator-circle-padding: #{sbb.px-to-rem-build(2)}; - --sbb-loading-indicator-circle-duration: var(--sbb-disable-animation-zero-duration, 1.5s); + --sbb-loading-indicator-circle-duration: var(--sbb-disable-animation-duration, 1.5s); --sbb-loading-indicator-circle-background-color: var(--sbb-color-white); --sbb-loading-indicator-circle-background: conic-gradient( from 90deg, diff --git a/src/elements/loading-indicator/loading-indicator.scss b/src/elements/loading-indicator/loading-indicator.scss index 0675cb57b2b..42d38904ec2 100644 --- a/src/elements/loading-indicator/loading-indicator.scss +++ b/src/elements/loading-indicator/loading-indicator.scss @@ -7,7 +7,7 @@ --sbb-loading-indicator-color: var(--sbb-color-red); --sbb-loading-indicator-padding: 0; --sbb-loading-indicator-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-6x) ); --sbb-loading-indicator-window-element-rotation: 55.24deg; diff --git a/src/elements/map-container/map-container.scss b/src/elements/map-container/map-container.scss index 563e2fdd98b..42ffce7c346 100644 --- a/src/elements/map-container/map-container.scss +++ b/src/elements/map-container/map-container.scss @@ -12,7 +12,7 @@ --sbb-map-container-sidebar-background-color: var(--sbb-color-white); --sbb-map-container-border-radius: var(--sbb-border-radius-4x); --sbb-map-container-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); --sbb-map-container-map-height: calc( diff --git a/src/elements/menu/menu/menu.scss b/src/elements/menu/menu/menu.scss index 72c5cab88bf..2652019ba83 100644 --- a/src/elements/menu/menu/menu.scss +++ b/src/elements/menu/menu/menu.scss @@ -12,7 +12,7 @@ --sbb-menu-position-x: 0; --sbb-menu-position-y: 0; --sbb-menu-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-6x) ); --sbb-menu-animation-easing: ease; diff --git a/src/elements/menu/menu/menu.spec.ts b/src/elements/menu/menu/menu.spec.ts index d1a8d41b611..bb40ed12a71 100644 --- a/src/elements/menu/menu/menu.spec.ts +++ b/src/elements/menu/menu/menu.spec.ts @@ -165,6 +165,25 @@ describe(`sbb-menu`, () => { expect(element).to.have.attribute('data-state', 'closed'); }); + it('opens and closes with non-zero animation duration', async () => { + element.style.setProperty('--sbb-menu-animation-duration', '1ms'); + const didOpenEventSpy = new EventSpy(SbbMenuElement.events.didOpen, element); + const didCloseEventSpy = new EventSpy(SbbMenuElement.events.didClose, element); + const menuLink = element.querySelector(':scope > sbb-block-link') as HTMLElement; + + trigger.click(); + await waitForLitRender(element); + + await didOpenEventSpy.calledOnce(); + + menuLink.click(); + await waitForLitRender(element); + + await didCloseEventSpy.calledOnce(); + + expect(element).to.have.attribute('data-state', 'closed'); + }); + it('is correctly positioned on desktop', async () => { const willOpenEventSpy = new EventSpy(SbbMenuElement.events.willOpen, element); const didOpenEventSpy = new EventSpy(SbbMenuElement.events.didOpen, element); diff --git a/src/elements/menu/menu/menu.ts b/src/elements/menu/menu/menu.ts index decb424b345..724ac8e640f 100644 --- a/src/elements/menu/menu/menu.ts +++ b/src/elements/menu/menu/menu.ts @@ -18,7 +18,11 @@ import { SbbMediaQueryBreakpointSmallAndBelow, } from '../../core/controllers.js'; import { forceType } from '../../core/decorators.js'; -import { findReferencedElement, SbbScrollHandler } from '../../core/dom.js'; +import { + findReferencedElement, + isZeroAnimationDuration, + SbbScrollHandler, +} from '../../core/dom.js'; import { SbbNamedSlotListMixin } from '../../core/mixins.js'; import { getElementPosition, @@ -128,6 +132,12 @@ class SbbMenuElement extends SbbNamedSlotListMixin< if (this._mediaMatcher.matches(SbbMediaQueryBreakpointSmallAndBelow)) { this._scrollHandler.disableScroll(); } + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `opened` state. + if (this._isZeroAnimationDuration()) { + this._handleOpening(); + } } /** @@ -144,6 +154,45 @@ class SbbMenuElement extends SbbNamedSlotListMixin< this.state = 'closing'; this._triggerElement?.setAttribute('aria-expanded', 'false'); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `closed` state. + if (this._isZeroAnimationDuration()) { + this._handleClosing(); + } + } + + private _isZeroAnimationDuration(): boolean { + return isZeroAnimationDuration(this, '--sbb-menu-animation-duration'); + } + + private _handleOpening(): void { + this.state = 'opened'; + this.didOpen.emit(); + this._inertController.activate(); + this._setMenuFocus(); + this._focusHandler.trap(this); + this._attachWindowEvents(); + } + + private _handleClosing(): void { + this.state = 'closed'; + this._menu?.firstElementChild?.scrollTo(0, 0); + this._inertController.deactivate(); + setModalityOnNextFocus(this._triggerElement); + // Manually focus last focused element + this._triggerElement?.focus({ + // When inside the sbb-header, we prevent the scroll to avoid the snapping to the top of the page + preventScroll: + this._triggerElement.localName === 'sbb-header-button' || + this._triggerElement.localName === 'sbb-header-link', + }); + this.didClose.emit(); + this._windowEventsController?.abort(); + this._focusHandler.disconnect(); + + // Starting from breakpoint medium, enable scroll + this._scrollHandler.enableScroll(); } /** @@ -314,30 +363,9 @@ class SbbMenuElement extends SbbNamedSlotListMixin< // To avoid entering a corrupt state, exit when state is not expected. private _onMenuAnimationEnd(event: AnimationEvent): void { if (event.animationName === 'open' && this.state === 'opening') { - this.state = 'opened'; - this.didOpen.emit(); - this._inertController.activate(); - this._setMenuFocus(); - this._focusHandler.trap(this); - this._attachWindowEvents(); + this._handleOpening(); } else if (event.animationName === 'close' && this.state === 'closing') { - this.state = 'closed'; - this._menu?.firstElementChild?.scrollTo(0, 0); - this._inertController.deactivate(); - setModalityOnNextFocus(this._triggerElement); - // Manually focus last focused element - this._triggerElement?.focus({ - // When inside the sbb-header, we prevent the scroll to avoid the snapping to the top of the page - preventScroll: - this._triggerElement.localName === 'sbb-header-button' || - this._triggerElement.localName === 'sbb-header-link', - }); - this.didClose.emit(); - this._windowEventsController?.abort(); - this._focusHandler.disconnect(); - - // Starting from breakpoint medium, enable scroll - this._scrollHandler.enableScroll(); + this._handleClosing(); } } @@ -379,7 +407,7 @@ class SbbMenuElement extends SbbNamedSlotListMixin< return html`
this._onMenuAnimationEnd(event)} + @animationend=${this._onMenuAnimationEnd} class="sbb-menu" ${ref((el?: Element) => (this._menu = el as HTMLDivElement))} > diff --git a/src/elements/navigation/common/navigation-action.scss b/src/elements/navigation/common/navigation-action.scss index 70e6947ae96..a69c11eb7ce 100644 --- a/src/elements/navigation/common/navigation-action.scss +++ b/src/elements/navigation/common/navigation-action.scss @@ -55,8 +55,7 @@ sbb-icon { display: flex; user-select: none; -webkit-tap-highlight-color: transparent; - transition: color var(--sbb-disable-animation-zero-duration, var(--sbb-animation-duration-3x)) - ease; + transition: color var(--sbb-disable-animation-duration, var(--sbb-animation-duration-3x)) ease; hyphens: auto; text-align: left; color: var(--sbb-navigation-action-color); diff --git a/src/elements/navigation/navigation-marker/navigation-marker.scss b/src/elements/navigation/navigation-marker/navigation-marker.scss index e58a08ad0e0..aa15700961b 100644 --- a/src/elements/navigation/navigation-marker/navigation-marker.scss +++ b/src/elements/navigation/navigation-marker/navigation-marker.scss @@ -54,7 +54,7 @@ border-block-start: var(--sbb-navigation-marker-border) solid var(--sbb-color-storm); margin-block: var(--sbb-navigation-marker-margin-block); transition: { - duration: var(--sbb-disable-animation-zero-duration, var(--sbb-animation-duration-6x)); + duration: var(--sbb-disable-animation-duration, var(--sbb-animation-duration-6x)); timing-function: ease; property: opacity, inset-block-start; } diff --git a/src/elements/navigation/navigation-section/navigation-section.scss b/src/elements/navigation/navigation-section/navigation-section.scss index 81e1fabc5ff..9e673f5dc9b 100644 --- a/src/elements/navigation/navigation-section/navigation-section.scss +++ b/src/elements/navigation/navigation-section/navigation-section.scss @@ -9,7 +9,7 @@ --sbb-navigation-section-position: fixed; --sbb-navigation-section-pointer-events: none; --sbb-navigation-section-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-3x) ); --sbb-navigation-section-animation-easing: ease-out; @@ -34,7 +34,7 @@ @include sbb.mq($from: large) { --sbb-navigation-section-column: 5 / 9; --sbb-navigation-section-animation-duration: var( - --sbb-disable-animation-zero-duration, + --sbb-disable-animation-duration, var(--sbb-animation-duration-4x) ); --sbb-navigation-section-padding-block: var(--sbb-spacing-responsive-xl); diff --git a/src/elements/navigation/navigation-section/navigation-section.spec.ts b/src/elements/navigation/navigation-section/navigation-section.spec.ts index 30e274b5c3e..d42d5383723 100644 --- a/src/elements/navigation/navigation-section/navigation-section.spec.ts +++ b/src/elements/navigation/navigation-section/navigation-section.spec.ts @@ -54,4 +54,19 @@ describe(`sbb-navigation-section`, () => { await waitForCondition(() => element.getAttribute('data-state') === 'closed'); expect(element).to.have.attribute('data-state', 'closed'); }); + + it('opens and closes with non-zero animation duration', async () => { + element.style.setProperty('--sbb-navigation-section-animation-duration', '1ms'); + + element.open(); + await waitForLitRender(element); + await waitForCondition(() => element.getAttribute('data-state') === 'opened'); + expect(element).to.have.attribute('data-state', 'opened'); + + element.close(); + await waitForLitRender(element); + + await waitForCondition(() => element.getAttribute('data-state') === 'closed'); + expect(element).to.have.attribute('data-state', 'closed'); + }); }); diff --git a/src/elements/navigation/navigation-section/navigation-section.ts b/src/elements/navigation/navigation-section/navigation-section.ts index 675a39c969d..8de12dba6ef 100644 --- a/src/elements/navigation/navigation-section/navigation-section.ts +++ b/src/elements/navigation/navigation-section/navigation-section.ts @@ -9,7 +9,12 @@ import { } from '../../core/a11y.js'; import { SbbLanguageController } from '../../core/controllers.js'; import { forceType, hostAttributes, omitEmptyConverter, slotState } from '../../core/decorators.js'; -import { findReferencedElement, isBreakpoint, setOrRemoveAttribute } from '../../core/dom.js'; +import { + findReferencedElement, + isBreakpoint, + isZeroAnimationDuration, + setOrRemoveAttribute, +} from '../../core/dom.js'; import { i18nGoBack } from '../../core/i18n.js'; import type { SbbOpenedClosedState } from '../../core/interfaces.js'; import { SbbUpdateSchedulerMixin } from '../../core/mixins.js'; @@ -111,6 +116,39 @@ class SbbNavigationSectionElement extends SbbUpdateSchedulerMixin(LitElement) { this.startUpdate(); this.inert = true; this._triggerElement?.setAttribute('aria-expanded', 'true'); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `opened` state. + if (this._isZeroAnimationDuration()) { + this._handleOpening(); + } + } + + private _isZeroAnimationDuration(): boolean { + return isZeroAnimationDuration(this, '--sbb-navigation-section-animation-duration'); + } + + private _handleOpening(): void { + this._state = 'opened'; + this.inert = false; + this._attachWindowEvents(); + this._setNavigationInert(); + this._setNavigationSectionFocus(); + this._checkActiveAction(); + this.completeUpdate(); + } + + private _handleClosing(): void { + this._state = 'closed'; + this._navigationSectionContainerElement.scrollTo(0, 0); + this._windowEventsController?.abort(); + this._resetLists(); + this._setNavigationInert(); + if (this._isZeroToLargeBreakpoint() && this._triggerElement) { + setModalityOnNextFocus(this._triggerElement); + this._triggerElement.focus(); + } + this.completeUpdate(); } private _setActiveNavigationAction(): void { @@ -133,6 +171,12 @@ class SbbNavigationSectionElement extends SbbUpdateSchedulerMixin(LitElement) { this.startUpdate(); this.inert = true; this._triggerElement?.setAttribute('aria-expanded', 'false'); + + // If the animation duration is zero, the animationend event is not always fired reliably. + // In this case we directly set the `closed` state. + if (this._isZeroAnimationDuration()) { + this._handleClosing(); + } } // Removes trigger click listener on trigger change. @@ -188,24 +232,10 @@ class SbbNavigationSectionElement extends SbbUpdateSchedulerMixin(LitElement) { // To avoid entering a corrupt state, exit when state is not expected. private _onAnimationEnd(event: AnimationEvent): void { if (event.animationName === 'open' && this._state === 'opening') { - this._state = 'opened'; - this.inert = false; - this._attachWindowEvents(); - this._setNavigationInert(); - this._setNavigationSectionFocus(); - this._checkActiveAction(); + this._handleOpening(); } else if (event.animationName === 'close' && this._state === 'closing') { - this._state = 'closed'; - this._navigationSectionContainerElement.scrollTo(0, 0); - this._windowEventsController?.abort(); - this._resetLists(); - this._setNavigationInert(); - if (this._isZeroToLargeBreakpoint() && this._triggerElement) { - setModalityOnNextFocus(this._triggerElement); - this._triggerElement.focus(); - } + this._handleClosing(); } - this.completeUpdate(); } private _resetLists(): void { @@ -336,7 +366,7 @@ class SbbNavigationSectionElement extends SbbUpdateSchedulerMixin(LitElement) { ${ref((el?: Element) => (this._navigationSectionContainerElement = el as HTMLElement))} >