From b43887d288f8c8d04939f4692d373a21a4493a34 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Aliz=C3=A9=20Debray?= <33580481+alizedebray@users.noreply.github.com> Date: Thu, 18 Jul 2024 08:26:52 +0200 Subject: [PATCH] feat(components): add a post-collapsible-trigger (#3209) Co-authored-by: Philipp Gfeller <1659006+gfellerph@users.noreply.github.com> --- .changeset/wild-bees-laugh.md | 8 ++ .../src/app/routes/home/home.component.html | 5 +- .../components/cypress/e2e/accordion.cy.ts | 10 +- .../components/cypress/e2e/collapsible.cy.ts | 72 +++++++---- .../components/cypress/support/commands.ts | 10 +- .../components/cypress/support/index.d.ts | 1 + packages/components/src/components.d.ts | 25 ++++ .../post-collapsible-trigger.tsx | 119 ++++++++++++++++++ .../post-collapsible-trigger/readme.md | 30 +++++ .../post-collapsible/post-collapsible.scss | 3 - .../post-collapsible/post-collapsible.tsx | 29 +++-- packages/components/src/index.ts | 1 + packages/components/src/utils/debounce.ts | 7 ++ packages/components/src/utils/index.ts | 1 + .../src/utils/tests/debounce.spec.ts | 48 +++++++ .../collapsible/collapsible.docs.mdx | 12 +- .../collapsible.snapshot.stories.ts | 32 ++--- .../collapsible/collapsible.stories.ts | 51 +++----- 18 files changed, 355 insertions(+), 109 deletions(-) create mode 100644 .changeset/wild-bees-laugh.md create mode 100644 packages/components/src/components/post-collapsible-trigger/post-collapsible-trigger.tsx create mode 100644 packages/components/src/components/post-collapsible-trigger/readme.md create mode 100644 packages/components/src/utils/debounce.ts create mode 100644 packages/components/src/utils/tests/debounce.spec.ts diff --git a/.changeset/wild-bees-laugh.md b/.changeset/wild-bees-laugh.md new file mode 100644 index 0000000000..25c1d126a9 --- /dev/null +++ b/.changeset/wild-bees-laugh.md @@ -0,0 +1,8 @@ +--- +'@swisspost/design-system-documentation': minor +'@swisspost/design-system-components': minor +'@swisspost/design-system-components-angular': minor +'@swisspost/design-system-components-react': minor +--- + +Added a `post-collapsible-trigger` component to properly handle the role, ARIA attributes, and event listeners for elements that toggle a `post-collapsible`. diff --git a/packages/components-angular/projects/consumer-app/src/app/routes/home/home.component.html b/packages/components-angular/projects/consumer-app/src/app/routes/home/home.component.html index 58327c6fe3..57d7de325f 100644 --- a/packages/components-angular/projects/consumer-app/src/app/routes/home/home.component.html +++ b/packages/components-angular/projects/consumer-app/src/app/routes/home/home.component.html @@ -30,7 +30,10 @@

Post Card-Control

Post Collapsible

- + + + +

Contentus momentus vero siteos et accusam iretea et justo.

diff --git a/packages/components/cypress/e2e/accordion.cy.ts b/packages/components/cypress/e2e/accordion.cy.ts index 9d8bdd7cc7..372947bb72 100644 --- a/packages/components/cypress/e2e/accordion.cy.ts +++ b/packages/components/cypress/e2e/accordion.cy.ts @@ -16,17 +16,17 @@ describe('accordion', () => { }); it('should only show the first element as expanded', () => { - cy.get('@collapsibles').first().find('.collapse').should('be.visible'); + cy.get('@collapsibles').first().shadow().find('post-collapsible').should('be.visible'); }); it('should show the last element as expanded after clicking it', () => { cy.get('@collapsibles').last().click(); - cy.get('@collapsibles').last().find('.collapse').should('be.visible'); + cy.get('@collapsibles').last().shadow().find('post-collapsible').should('be.visible'); }); it('should not show the first element as expanded after clicking the last element', () => { cy.get('@collapsibles').last().click(); - cy.get('@collapsibles').first().find('.collapse').should('be.hidden'); + cy.get('@collapsibles').first().shadow().find('post-collapsible').should('be.hidden'); }); it('should propagate "postToggle" event from post-accordion-item on post-accordion', () => { @@ -73,12 +73,12 @@ describe('accordion', () => { it('should show the last element as expanded after clicking it', () => { cy.get('@collapsibles').last().click(); - cy.get('@collapsibles').last().find('.collapse').should('be.visible'); + cy.get('@collapsibles').last().shadow().find('post-collapsible').should('be.visible'); }); it('should still show the first element as expanded after clicking the last element', () => { cy.get('@collapsibles').last().click(); - cy.get('@collapsibles').first().find('.collapse').should('be.visible'); + cy.get('@collapsibles').first().shadow().find('post-collapsible').should('be.visible'); }); }); }); diff --git a/packages/components/cypress/e2e/collapsible.cy.ts b/packages/components/cypress/e2e/collapsible.cy.ts index f7d2d8a5b7..5ae83f7e9a 100644 --- a/packages/components/cypress/e2e/collapsible.cy.ts +++ b/packages/components/cypress/e2e/collapsible.cy.ts @@ -3,57 +3,75 @@ const COLLAPSIBLE_ID = '6a91848c-16ec-4a23-bc45-51c797b5b2c3'; describe('collapsible', () => { describe('default', () => { beforeEach(() => { - cy.getComponent('collapsible', COLLAPSIBLE_ID); - cy.get('@collapsible').find('.collapse').as('collapse'); - cy.get(`#button--${COLLAPSIBLE_ID}--default`).as('toggler'); + cy.getComponents(COLLAPSIBLE_ID, 'default', 'post-collapsible', 'post-collapsible-trigger'); + cy.get('@collapsible-trigger').find('.btn').as('trigger'); }); - it('should render', () => { + it('should have a collapsible', () => { cy.get('@collapsible').should('exist'); }); - it('should have a collapse', () => { - cy.get('@collapse').should('exist'); + it('should have a trigger', () => { + cy.get('@trigger').should('exist'); }); - it('should have a toggle button', () => { - cy.get('@toggler').should('exist'); + it('should show the collapsible', () => { + cy.get('@collapsible').should(`be.visible`); }); - it('should be expanded', () => { - cy.get('@collapse').should(`be.visible`); + it('should set the correct ARIA attribute on the trigger', () => { + cy.get('@collapsible') + .invoke('attr', 'id') + .then(collapsibleId => { + cy.get('@trigger').should('have.attr', 'aria-controls', collapsibleId); + }); + cy.get('@trigger').should('have.attr', 'aria-expanded', 'true'); }); - it('should be collapsed after clicking on the toggle button once', () => { - cy.get('@toggler').click(); - cy.get('@collapse').should(`be.hidden`); + it('should hide the collapsible after clicking on the trigger once', () => { + cy.get('@trigger').click(); + cy.get('@collapsible').should(`be.hidden`); }); - it('should be expanded after clicking on the toggle button twice', () => { - cy.get('@toggler').dblclick(); - cy.get('@collapse').should(`be.visible`); + it('should update the "aria-expanded" attribute after hiding the collapsible', () => { + cy.get('@trigger').click(); + cy.get('@trigger').should('have.attr', 'aria-expanded', 'false'); + }); + + it('should show the collapsible after clicking on the trigger twice', () => { + cy.get('@trigger').dblclick(); + cy.get('@collapsible').should(`be.visible`); + }); + + it('should update the "aria-expanded" attribute after showing the collapsible', () => { + cy.get('@trigger').dblclick(); + cy.get('@trigger').should('have.attr', 'aria-expanded', 'true'); }); }); describe('initially collapsed', () => { beforeEach(() => { - cy.getComponent('collapsible', COLLAPSIBLE_ID, 'initially-collapsed'); - cy.get('@collapsible').find('.collapse').as('collapse'); - cy.get(`#button--${COLLAPSIBLE_ID}--initially-collapsed`).as('toggler'); + cy.getComponents( + COLLAPSIBLE_ID, + 'initially-collapsed', + 'post-collapsible', + 'post-collapsible-trigger', + ); + cy.get('@collapsible-trigger').find('.btn').as('trigger'); }); - it('should be collapsed', () => { - cy.get('@collapse').should(`be.hidden`); + it('should hide the collapsible', () => { + cy.get('@collapsible').should(`be.hidden`); }); - it('should be expanded after clicking on the toggle button once', () => { - cy.get('@toggler').click(); - cy.get('@collapse').should(`be.visible`); + it('should show the collapsible after clicking on the trigger once', () => { + cy.get('@trigger').click(); + cy.get('@collapsible').should(`be.visible`); }); - it('should be collapsed after clicking on the toggle button twice', () => { - cy.get('@toggler').dblclick(); - cy.get('@collapse').should(`be.hidden`); + it('should hide the collapsible after clicking on the trigger twice', () => { + cy.get('@trigger').dblclick(); + cy.get('@collapsible').should(`be.hidden`); }); }); }); diff --git a/packages/components/cypress/support/commands.ts b/packages/components/cypress/support/commands.ts index 29e8f58b8c..20f89a741e 100644 --- a/packages/components/cypress/support/commands.ts +++ b/packages/components/cypress/support/commands.ts @@ -49,10 +49,16 @@ export const isInViewport = function (_chai: Chai.ChaiStatic) { chai.use(isInViewport); Cypress.Commands.add('getComponent', (component: string, id: string, story = 'default') => { + cy.getComponents(id, story, component); +}); + +Cypress.Commands.add('getComponents', (id: string, story: string, ...components: string[]) => { cy.visit(`/iframe.html?id=${id}--${story}`); - const alias = component.replace(/^post-/, ''); - cy.get(`post-${alias}`, { timeout: 30000 }).as(alias); + components.forEach(component => { + const alias = component.replace(/^post-/, ''); + cy.get(`post-${alias}.hydrated`, { timeout: 30000 }).as(alias); + }); cy.injectAxe(); }); diff --git a/packages/components/cypress/support/index.d.ts b/packages/components/cypress/support/index.d.ts index d7b6a2cd32..96fbc79c05 100644 --- a/packages/components/cypress/support/index.d.ts +++ b/packages/components/cypress/support/index.d.ts @@ -2,6 +2,7 @@ declare global { namespace Cypress { interface Chainable { getComponent(component: string, id: string, story?: string): Chainable; + getComponents(id: string, story: string, ...component: string[]): Chainable; getSnapshots(component: string): Chainable; checkAriaExpanded( controlledElementSelector: string, diff --git a/packages/components/src/components.d.ts b/packages/components/src/components.d.ts index ba024f8726..902617a88f 100644 --- a/packages/components/src/components.d.ts +++ b/packages/components/src/components.d.ts @@ -134,6 +134,16 @@ export namespace Components { */ "toggle": (open?: boolean) => Promise; } + interface PostCollapsibleTrigger { + /** + * Link the trigger to a post-collapsible with this id + */ + "for": string; + /** + * Update the "aria-controls" and "aria-expanded" attributes on the trigger button + */ + "update": () => Promise; + } /** * @class PostIcon - representing a stencil component */ @@ -397,6 +407,12 @@ declare global { prototype: HTMLPostCollapsibleElement; new (): HTMLPostCollapsibleElement; }; + interface HTMLPostCollapsibleTriggerElement extends Components.PostCollapsibleTrigger, HTMLStencilElement { + } + var HTMLPostCollapsibleTriggerElement: { + prototype: HTMLPostCollapsibleTriggerElement; + new (): HTMLPostCollapsibleTriggerElement; + }; /** * @class PostIcon - representing a stencil component */ @@ -494,6 +510,7 @@ declare global { "post-alert": HTMLPostAlertElement; "post-card-control": HTMLPostCardControlElement; "post-collapsible": HTMLPostCollapsibleElement; + "post-collapsible-trigger": HTMLPostCollapsibleTriggerElement; "post-icon": HTMLPostIconElement; "post-popover": HTMLPostPopoverElement; "post-popovercontainer": HTMLPostPopovercontainerElement; @@ -612,6 +629,12 @@ declare namespace LocalJSX { */ "onPostToggle"?: (event: PostCollapsibleCustomEvent) => void; } + interface PostCollapsibleTrigger { + /** + * Link the trigger to a post-collapsible with this id + */ + "for"?: string; + } /** * @class PostIcon - representing a stencil component */ @@ -755,6 +778,7 @@ declare namespace LocalJSX { "post-alert": PostAlert; "post-card-control": PostCardControl; "post-collapsible": PostCollapsible; + "post-collapsible-trigger": PostCollapsibleTrigger; "post-icon": PostIcon; "post-popover": PostPopover; "post-popovercontainer": PostPopovercontainer; @@ -778,6 +802,7 @@ declare module "@stencil/core" { */ "post-card-control": LocalJSX.PostCardControl & JSXBase.HTMLAttributes; "post-collapsible": LocalJSX.PostCollapsible & JSXBase.HTMLAttributes; + "post-collapsible-trigger": LocalJSX.PostCollapsibleTrigger & JSXBase.HTMLAttributes; /** * @class PostIcon - representing a stencil component */ diff --git a/packages/components/src/components/post-collapsible-trigger/post-collapsible-trigger.tsx b/packages/components/src/components/post-collapsible-trigger/post-collapsible-trigger.tsx new file mode 100644 index 0000000000..9c049ae06d --- /dev/null +++ b/packages/components/src/components/post-collapsible-trigger/post-collapsible-trigger.tsx @@ -0,0 +1,119 @@ +import { Component, Element, Listen, Method, Prop, Watch } from '@stencil/core'; +import { version } from 'typescript'; +import { checkNonEmpty, checkType, debounce } from '@/utils'; +import { PostCollapsibleCustomEvent } from '@/components'; + +@Component({ + tag: 'post-collapsible-trigger', +}) +export class PostCollapsibleTrigger { + private trigger?: HTMLButtonElement; + private observer = new MutationObserver(() => this.setTrigger()); + + @Element() host: HTMLPostCollapsibleTriggerElement; + + /** + * Link the trigger to a post-collapsible with this id + */ + @Prop() for: string; + + /** + * Set the "aria-controls" and "aria-expanded" attributes on the trigger to match the state of the controlled post-collapsible + */ + @Watch('for') + setAriaAttributes() { + checkNonEmpty(this.for, 'The post-collapsible-trigger "for" prop is required.'); + checkType(this.for, 'string', 'The post-collapsible-trigger "for" prop should be a id.'); + + void this.update(); + } + + /** + * Initiate a mutation observer that updates the trigger whenever necessary + */ + connectedCallback() { + this.observer.observe(this.host, { childList: true, subtree: true }); + } + + /** + * Add the "data-version" to the host element and set the trigger + */ + componentDidLoad() { + this.host.setAttribute('data-version', version); + this.setTrigger(); + + if (!this.trigger) console.warn('The post-collapsible-trigger must contain a button.'); + } + + /** + * Disconnect the mutation observer + */ + disconnectedCallback() { + this.observer.disconnect(); + } + + /** + * Update the "aria-expanded" attribute on the trigger anytime the controlled post-collapsible is toggled + */ + @Listen('postToggle', { target: 'document' }) + setAriaExpanded(e: PostCollapsibleCustomEvent) { + if (!this.trigger || !e.target.isEqualNode(this.collapsible)) return; + this.trigger.setAttribute('aria-expanded', `${e.detail}`); + } + + /** + * Update the "aria-controls" and "aria-expanded" attributes on the trigger button + */ + @Method() + async update() { + this.debouncedUpdate(); + } + + private debouncedUpdate = debounce(() => { + if (!this.trigger) return; + + // add the provided id to the aria-controls list + const ariaControls = this.trigger.getAttribute('aria-controls'); + if (!ariaControls?.includes(this.for)) { + const newAriaControls = ariaControls ? `${ariaControls} ${this.for}` : this.for; + this.trigger.setAttribute('aria-controls', newAriaControls); + } + + // set the aria-expanded to `false` if the controlled collapsible is collapsed or undefined, set it to `true` otherwise + const isCollapsed = this.collapsible?.collapsed; + const newAriaExpanded = isCollapsed !== undefined ? !isCollapsed : undefined; + this.trigger.setAttribute('aria-expanded', `${newAriaExpanded}`); + }); + + /** + * Toggle the post-collapsible controlled by the trigger + */ + private async toggleCollapsible() { + await this.collapsible?.toggle(); + } + + /** + * Retrieve the post-collapsible controlled by the trigger + */ + private get collapsible(): HTMLPostCollapsibleElement | null { + const ref = document.getElementById(this.for); + if (ref && ref.localName === 'post-collapsible') { + return ref as HTMLPostCollapsibleElement; + } + + return null; + } + + /** + * Find the button and add the proper event listener and ARIA attributes to it + */ + private setTrigger() { + const trigger = this.host.querySelector('button'); + if (!trigger || (this.trigger && trigger.isEqualNode(this.trigger))) return; + + this.trigger = trigger; + + this.trigger.addEventListener('click', () => this.toggleCollapsible()); + this.setAriaAttributes(); + } +} diff --git a/packages/components/src/components/post-collapsible-trigger/readme.md b/packages/components/src/components/post-collapsible-trigger/readme.md new file mode 100644 index 0000000000..3c5f16db95 --- /dev/null +++ b/packages/components/src/components/post-collapsible-trigger/readme.md @@ -0,0 +1,30 @@ +# post-collapsible-trigger + + + + + + +## Properties + +| Property | Attribute | Description | Type | Default | +| -------- | --------- | --------------------------------------------------- | -------- | ----------- | +| `for` | `for` | Link the trigger to a post-collapsible with this id | `string` | `undefined` | + + +## Methods + +### `update() => Promise` + +Update the "aria-controls" and "aria-expanded" attributes on the trigger button + +#### Returns + +Type: `Promise` + + + + +---------------------------------------------- + +*Built with [StencilJS](https://stenciljs.com/)* diff --git a/packages/components/src/components/post-collapsible/post-collapsible.scss b/packages/components/src/components/post-collapsible/post-collapsible.scss index 0aaf688255..20947d5cd1 100644 --- a/packages/components/src/components/post-collapsible/post-collapsible.scss +++ b/packages/components/src/components/post-collapsible/post-collapsible.scss @@ -1,7 +1,4 @@ :host { display: block; -} - -.collapse { overflow: hidden; } diff --git a/packages/components/src/components/post-collapsible/post-collapsible.tsx b/packages/components/src/components/post-collapsible/post-collapsible.tsx index 41a29b4323..7473f56975 100644 --- a/packages/components/src/components/post-collapsible/post-collapsible.tsx +++ b/packages/components/src/components/post-collapsible/post-collapsible.tsx @@ -7,7 +7,6 @@ import { Host, Method, Prop, - State, Watch, } from '@stencil/core'; import { version } from '@root/package.json'; @@ -26,12 +25,9 @@ import { checkEmptyOrType, isMotionReduced } from '@/utils'; export class PostCollapsible { private isLoaded = false; private isOpen = true; - private collapsible: HTMLElement; @Element() host: HTMLPostCollapsibleElement; - @State() id: string; - /** * If `true`, the element is initially collapsed otherwise it is displayed. */ @@ -57,13 +53,11 @@ export class PostCollapsible { this.validateCollapsed(); } - componentWillRender() { - this.id = this.host.id || `c${crypto.randomUUID()}`; - } - componentDidLoad() { if (this.collapsed) void this.toggle(false); this.isLoaded = true; + + this.updateTriggers(); } /** @@ -78,7 +72,7 @@ export class PostCollapsible { this.isOpen = !this.isOpen; if (this.isLoaded) this.postToggle.emit(this.isOpen); - const animation = open ? expand(this.collapsible) : collapse(this.collapsible); + const animation = open ? expand(this.host) : collapse(this.host); if (!this.isLoaded || isMotionReduced()) animation.finish(); @@ -89,12 +83,21 @@ export class PostCollapsible { return this.isOpen; } + /** + * Update all post-collapsible-trigger elements referring to the collapsible + */ + private updateTriggers() { + const triggers: NodeListOf = document.querySelectorAll( + `post-collapsible-trigger[for=${this.host.id}]`, + ); + + triggers.forEach(trigger => trigger.update()); + } + render() { return ( - -
(this.collapsible = el)}> - -
+ + ); } diff --git a/packages/components/src/index.ts b/packages/components/src/index.ts index 6ad5b611b6..0130213e14 100644 --- a/packages/components/src/index.ts +++ b/packages/components/src/index.ts @@ -6,6 +6,7 @@ export { PostAccordionItem } from './components/post-accordion-item/post-accordi export { PostAlert } from './components/post-alert/post-alert'; export { PostCardControl } from './components/post-card-control/post-card-control'; export { PostCollapsible } from './components/post-collapsible/post-collapsible'; +export { PostCollapsibleTrigger } from './components/post-collapsible-trigger/post-collapsible-trigger'; export { PostIcon } from './components/post-icon/post-icon'; export { PostPopover } from './components/post-popover/post-popover'; export { PostPopovercontainer } from './components/post-popovercontainer/post-popovercontainer'; diff --git a/packages/components/src/utils/debounce.ts b/packages/components/src/utils/debounce.ts new file mode 100644 index 0000000000..fe68e14988 --- /dev/null +++ b/packages/components/src/utils/debounce.ts @@ -0,0 +1,7 @@ +export function debounce(callback: (...args: T) => void, timeout = 200) { + let id: ReturnType; + return (...args: T) => { + if (id) clearTimeout(id); + id = setTimeout(callback, timeout, ...args); + }; +} diff --git a/packages/components/src/utils/index.ts b/packages/components/src/utils/index.ts index 52d8b3df1e..f7421da8a9 100644 --- a/packages/components/src/utils/index.ts +++ b/packages/components/src/utils/index.ts @@ -1,4 +1,5 @@ export * from './property-checkers'; +export * from './debounce'; export * from './is-motion-reduced'; export * from './sass-export'; export * from './timeout'; diff --git a/packages/components/src/utils/tests/debounce.spec.ts b/packages/components/src/utils/tests/debounce.spec.ts new file mode 100644 index 0000000000..d48eb5b757 --- /dev/null +++ b/packages/components/src/utils/tests/debounce.spec.ts @@ -0,0 +1,48 @@ +import { debounce } from '../debounce'; + +const timeout = 50; + +describe('debounce', () => { + let callback: jest.MockedFn<(...args: unknown[]) => void>; + let debouncedCallback: (...args: unknown[]) => void; + + beforeEach(() => { + callback = jest.fn(); + debouncedCallback = debounce(callback, timeout); + }); + + it('should wait until the provided timeout elapses before executing the callback function', done => { + debouncedCallback(); + + setTimeout(() => { + expect(callback).not.toHaveBeenCalled(); + }, timeout / 2); + + setTimeout(() => { + expect(callback).toHaveBeenCalled(); + done(); + }, timeout * 2); + }); + + it('should only execute the callback function once when the timeout elapses', done => { + debouncedCallback(); + debouncedCallback(); + debouncedCallback(); + + setTimeout(() => { + expect(callback).toHaveBeenCalledTimes(1); + done(); + }, timeout * 2); + }); + + it('should pass all provided arguments to the callback function', done => { + const args = [25, 'myArg', false]; + + debouncedCallback(...args); + + setTimeout(() => { + expect(callback).toHaveBeenCalledWith(...args); + done(); + }, timeout * 2); + }); +}); diff --git a/packages/documentation/src/stories/components/collapsible/collapsible.docs.mdx b/packages/documentation/src/stories/components/collapsible/collapsible.docs.mdx index ab60064f56..60f4145bda 100644 --- a/packages/documentation/src/stories/components/collapsible/collapsible.docs.mdx +++ b/packages/documentation/src/stories/components/collapsible/collapsible.docs.mdx @@ -24,19 +24,13 @@ To make the collapsible content hidden by default, just use the `collapsible="tr -### Custom Trigger +### Programmatic Toggle -The `` component offers a `.toggle()` method that allows to trigger the collapse programmatically. +The `` component offers a `.toggle()` method that allows you to trigger the collapse programmatically. This method is asynchronous and returns a promise that resolves with the current open state. -It optionally takes a boolean parameter that forces open when `true` or close when `false`. +It optionally takes a boolean parameter: `true` forces it open, and `false` forces it closed. - -To ensure good accessibility, identify the collapsible with an `id`, -then add an `aria-controls` attribute to your control element referencing this `id`. -Also make sure to add an `aria-expanded` attribute to the control element: -if the collapsible element is closed, the attribute on the control element must have a value of `aria-expanded="false"` -and `aria-expanded="true"` otherwise. diff --git a/packages/documentation/src/stories/components/collapsible/collapsible.snapshot.stories.ts b/packages/documentation/src/stories/components/collapsible/collapsible.snapshot.stories.ts index 38e0585172..552139bb72 100644 --- a/packages/documentation/src/stories/components/collapsible/collapsible.snapshot.stories.ts +++ b/packages/documentation/src/stories/components/collapsible/collapsible.snapshot.stories.ts @@ -2,13 +2,12 @@ import { html } from 'lit'; import type { Args, StoryContext, StoryObj } from '@storybook/web-components'; import { bombArgs } from '@/utils'; -import meta, { Default } from './collapsible.stories'; +import meta from './collapsible.stories'; const { id, ...metaWithoutId } = meta; export default { ...metaWithoutId, - decorators: [], title: 'Snapshots', }; @@ -16,21 +15,26 @@ type Story = StoryObj; export const Collapsible: Story = { render: (_args: Args, context: StoryContext) => { - const templateVariants = bombArgs({ - collapsed: [false, true], - }).map((args: Args) => { - return html` -
-

collapsed: ${args.collapsed}

- ${meta.render?.({ ...context.args, ...Default.args, ...args }, context)} -
- `; - }); - return html`
${['white', 'dark'].map( - bg => html`
${templateVariants}
`, + bg => html` +
+ ${bombArgs({ + collapsed: [false, true], + }).map( + (args: Args, i: number) => html` +
+

collapsed: ${args.collapsed}

+ ${meta.render?.( + { ...context.args, ...args }, + { ...context, id: `${context.id}-${bg}-${i}` }, + )} +
+ `, + )} +
+ `, )}
`; diff --git a/packages/documentation/src/stories/components/collapsible/collapsible.stories.ts b/packages/documentation/src/stories/components/collapsible/collapsible.stories.ts index ad58063b28..ea034d8e98 100644 --- a/packages/documentation/src/stories/components/collapsible/collapsible.stories.ts +++ b/packages/documentation/src/stories/components/collapsible/collapsible.stories.ts @@ -2,6 +2,7 @@ import { StoryContext, StoryFn, StoryObj } from '@storybook/web-components'; import { html } from 'lit'; import { spreadArgs } from '@/utils'; import { MetaComponent } from '@root/types'; +import { unsafeHTML } from 'lit/directives/unsafe-html.js'; const meta: MetaComponent = { id: '6a91848c-16ec-4a23-bc45-51c797b5b2c3', @@ -9,7 +10,7 @@ const meta: MetaComponent = { tags: ['package:WebComponents'], component: 'post-collapsible', render: renderCollapsible, - decorators: [externalControls], + decorators: [gap], parameters: { badges: [], controls: { @@ -25,46 +26,26 @@ const meta: MetaComponent = { export default meta; // DECORATORS -function externalControls(story: StoryFn, context: StoryContext) { - const { args, canvasElement } = context; - const togglerId = `button--${context.id}`; - - let collapsible!: HTMLPostCollapsibleElement; - let toggler!: HTMLButtonElement; - setTimeout(async () => { - collapsible = canvasElement.querySelector('post-collapsible') as HTMLPostCollapsibleElement; - toggler = canvasElement.querySelector(`#${togglerId}`) as HTMLButtonElement; - - await collapsible.componentOnReady(); - - toggler.setAttribute('aria-controls', collapsible.id); - }); - - const toggle = async () => { - const isOpen = await collapsible.toggle(); - toggler.setAttribute('aria-expanded', String(isOpen)); - }; +function gap(story: StoryFn, context: StoryContext) { + return html`
${story(context.args, context)}
`; +} +//RENDERER +function renderCollapsible( + { innerHTML, ...args }: Partial, + context: StoryContext, +) { return html` - + + + - ${story(args, context)} + + ${unsafeHTML(innerHTML)} + `; } -//RENDERER -function renderCollapsible(args: Partial) { - return html` `; -} - // STORIES type Story = StoryObj;