@spectrum-web-components/tabs 0.48.1 → 0.49.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["TabsOverflow.ts"],
4
- "sourcesContent": ["/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\nimport {\n CSSResultArray,\n html,\n PropertyValueMap,\n PropertyValues,\n SizedMixin,\n SpectrumElement,\n TemplateResult,\n} from '@spectrum-web-components/base';\nimport {\n property,\n query,\n queryAssignedElements,\n state,\n} from '@spectrum-web-components/base/src/decorators.js';\nimport { classMap } from '@spectrum-web-components/base/src/directives.js';\nimport { ResizeController } from '@lit-labs/observers/resize-controller.js';\nimport { Tabs } from './Tabs.js';\nimport '@spectrum-web-components/action-button/sp-action-button.js';\nimport '@spectrum-web-components/icons-ui/icons/sp-icon-chevron100.js';\nimport chevronIconStyles from '@spectrum-web-components/icon/src/spectrum-icon-chevron.css.js';\nimport tabSizes from './tabs-sizes.css.js';\nimport styles from './tabs-overflow.css.js';\n\ninterface TabsOverflowState {\n canScrollLeft: boolean;\n canScrollRight: boolean;\n}\n/**\n * @element sp-tabs-overflow\n */\nexport class TabsOverflow extends SizedMixin(SpectrumElement) {\n public static override get styles(): CSSResultArray {\n return [chevronIconStyles, styles, tabSizes];\n }\n\n @property({ type: Boolean, reflect: true })\n public compact = false;\n\n @property({ type: String, attribute: 'label-previous' })\n public labelPrevious = 'Scroll to previous tabs';\n\n @property({ type: String, attribute: 'label-next' })\n public labelNext = 'Scroll to next tabs';\n\n @property({ reflect: true })\n public override dir!: 'ltr' | 'rtl';\n\n @state()\n private overflowState: TabsOverflowState = {\n canScrollLeft: false,\n canScrollRight: false,\n };\n\n @queryAssignedElements({ selector: 'sp-tabs', flatten: true })\n private scrollContent!: Tabs[];\n\n @query('.tabs-overflow-container')\n private overflowContainer!: HTMLDivElement;\n\n resizeController!: ResizeController;\n\n public constructor() {\n super();\n this.resizeController = new ResizeController(this, {\n target: this,\n callback: (): void => {\n this._updateScrollState();\n },\n });\n }\n\n protected override firstUpdated(changes: PropertyValues): void {\n super.firstUpdated(changes);\n // enable scroll event\n const [tabs] = this.scrollContent;\n if (tabs) {\n tabs.enableTabsScroll = true;\n }\n this.resizeController.observe(this.overflowContainer);\n }\n\n private async _handleSlotChange(): Promise<void> {\n const [tabsElement] = this.scrollContent;\n await tabsElement?.updateComplete;\n this._updateScrollState();\n }\n\n private _updateScrollState(): void {\n const { scrollContent, overflowState } = this;\n\n if (scrollContent) {\n const [tabsElement] = this.scrollContent;\n const { canScrollLeft, canScrollRight } =\n tabsElement?.scrollState || {\n canScrollLeft: false,\n canScrollRight: false,\n };\n\n this.overflowState = {\n ...overflowState,\n canScrollLeft,\n canScrollRight,\n };\n }\n }\n\n private _handleScrollClick(event: MouseEvent): void {\n const currentTarget = event.currentTarget as HTMLElement;\n const [tabsElement] = this.scrollContent;\n\n const dist = tabsElement.clientWidth * 0.5;\n const left = currentTarget.classList.contains('left-scroll')\n ? -dist\n : dist;\n tabsElement.scrollTabs(left, 'smooth');\n }\n\n protected override updated(\n changedProperties: PropertyValueMap<this>\n ): void {\n super.updated(changedProperties);\n if (changedProperties.has('dir')) {\n this._updateScrollState();\n }\n }\n\n protected override render(): TemplateResult {\n const { canScrollRight, canScrollLeft } = this.overflowState;\n const ariaLabelPrevious = this.labelPrevious;\n const ariaLabelNext = this.labelNext;\n return html`\n <div\n class=${classMap({\n 'tabs-overflow-container': true,\n 'left-shadow': canScrollLeft,\n 'right-shadow': canScrollRight,\n })}\n >\n <sp-action-button\n class=${classMap({\n 'left-scroll': true,\n show: canScrollLeft,\n })}\n aria-label=${ariaLabelPrevious}\n quiet\n dir=\"rtl\"\n tabindex=\"-1\"\n @click=${this._handleScrollClick}\n >\n <sp-icon-chevron100\n slot=\"icon\"\n class=\"spectrum-UIIcon-ChevronLeft300\"\n ></sp-icon-chevron100>\n </sp-action-button>\n <sp-action-button\n class=${classMap({\n 'right-scroll': true,\n show: canScrollRight,\n })}\n aria-label=${ariaLabelNext}\n quiet\n tabindex=\"-1\"\n @click=${this._handleScrollClick}\n >\n <sp-icon-chevron100\n slot=\"icon\"\n class=\"spectrum-UIIcon-ChevronRight300\"\n ></sp-icon-chevron100>\n </sp-action-button>\n <slot\n @slotchange=${this._handleSlotChange}\n @sp-tabs-scroll=${this._updateScrollState}\n ></slot>\n </div>\n `;\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\nimport {\n CSSResultArray,\n html,\n PropertyValueMap,\n PropertyValues,\n SizedMixin,\n SpectrumElement,\n TemplateResult,\n} from '@spectrum-web-components/base';\nimport {\n property,\n query,\n queryAssignedElements,\n state,\n} from '@spectrum-web-components/base/src/decorators.js';\nimport { classMap } from '@spectrum-web-components/base/src/directives.js';\nimport { ResizeController } from '@lit-labs/observers/resize-controller.js';\nimport { Tabs } from './Tabs.js';\nimport '@spectrum-web-components/action-button/sp-action-button.js';\nimport '@spectrum-web-components/icons-ui/icons/sp-icon-chevron100.js';\nimport chevronIconStyles from '@spectrum-web-components/icon/src/spectrum-icon-chevron.css.js';\nimport tabSizes from './tabs-sizes.css.js';\nimport styles from './tabs-overflow.css.js';\n\ninterface TabsOverflowState {\n canScrollLeft: boolean;\n canScrollRight: boolean;\n}\n/**\n * @element sp-tabs-overflow\n */\nexport class TabsOverflow extends SizedMixin(SpectrumElement) {\n public static override get styles(): CSSResultArray {\n return [chevronIconStyles, styles, tabSizes];\n }\n\n @property({ type: Boolean, reflect: true })\n public compact = false;\n\n @property({ type: String, attribute: 'label-previous' })\n public labelPrevious = 'Scroll to previous tabs';\n\n @property({ type: String, attribute: 'label-next' })\n public labelNext = 'Scroll to next tabs';\n\n @property({ reflect: true })\n public override dir!: 'ltr' | 'rtl';\n\n @state()\n private overflowState: TabsOverflowState = {\n canScrollLeft: false,\n canScrollRight: false,\n };\n\n @queryAssignedElements({ selector: 'sp-tabs', flatten: true })\n private scrollContent!: Tabs[];\n\n @query('.tabs-overflow-container')\n private overflowContainer!: HTMLDivElement;\n\n resizeController!: ResizeController;\n\n public constructor() {\n super();\n this.resizeController = new ResizeController(this, {\n target: this,\n callback: (): void => {\n this._updateScrollState();\n },\n });\n }\n\n protected override firstUpdated(changes: PropertyValues): void {\n super.firstUpdated(changes);\n // enable scroll event\n const [tabs] = this.scrollContent;\n if (tabs) {\n tabs.enableTabsScroll = true;\n }\n this.resizeController.observe(this.overflowContainer);\n }\n\n private async _handleSlotChange(): Promise<void> {\n const [tabsElement] = this.scrollContent;\n await tabsElement?.updateComplete;\n this._updateScrollState();\n }\n\n private _updateScrollState(): void {\n const { scrollContent, overflowState } = this;\n\n if (scrollContent) {\n const [tabsElement] = this.scrollContent;\n const { canScrollLeft, canScrollRight } =\n tabsElement?.scrollState || {\n canScrollLeft: false,\n canScrollRight: false,\n };\n\n this.overflowState = {\n ...overflowState,\n canScrollLeft,\n canScrollRight,\n };\n }\n }\n\n private scrollFactor = 0.5;\n private _handleScrollClick(event: MouseEvent): void {\n const currentTarget = event.currentTarget as HTMLElement;\n const [tabsElement] = this.scrollContent;\n\n const dist = tabsElement.clientWidth * this.scrollFactor;\n const left = currentTarget.classList.contains('left-scroll')\n ? -dist\n : dist;\n tabsElement.scrollTabs(left, 'smooth');\n }\n\n protected override updated(\n changedProperties: PropertyValueMap<this>\n ): void {\n super.updated(changedProperties);\n if (changedProperties.has('dir')) {\n this._updateScrollState();\n }\n }\n\n protected override render(): TemplateResult {\n const { canScrollRight, canScrollLeft } = this.overflowState;\n const ariaLabelPrevious = this.labelPrevious;\n const ariaLabelNext = this.labelNext;\n return html`\n <div\n class=${classMap({\n 'tabs-overflow-container': true,\n 'left-shadow': canScrollLeft,\n 'right-shadow': canScrollRight,\n })}\n >\n <sp-action-button\n class=${classMap({\n 'left-scroll': true,\n show: canScrollLeft,\n })}\n aria-label=${ariaLabelPrevious}\n quiet\n dir=\"rtl\"\n tabindex=\"-1\"\n @click=${this._handleScrollClick}\n >\n <sp-icon-chevron100\n slot=\"icon\"\n class=\"spectrum-UIIcon-ChevronLeft300\"\n ></sp-icon-chevron100>\n </sp-action-button>\n <sp-action-button\n class=${classMap({\n 'right-scroll': true,\n show: canScrollRight,\n })}\n aria-label=${ariaLabelNext}\n quiet\n tabindex=\"-1\"\n @click=${this._handleScrollClick}\n >\n <sp-icon-chevron100\n slot=\"icon\"\n class=\"spectrum-UIIcon-ChevronRight300\"\n ></sp-icon-chevron100>\n </sp-action-button>\n <slot\n @slotchange=${this._handleSlotChange}\n @sp-tabs-scroll=${this._updateScrollState}\n ></slot>\n </div>\n `;\n }\n}\n"],
5
+ "mappings": "qNAWA,OAEI,QAAAA,EAGA,cAAAC,EACA,mBAAAC,MAEG,gCACP,OACI,YAAAC,EACA,SAAAC,EACA,yBAAAC,EACA,SAAAC,MACG,kDACP,OAAS,YAAAC,MAAgB,kDACzB,OAAS,oBAAAC,MAAwB,2CAEjC,MAAO,6DACP,MAAO,gEACP,OAAOC,MAAuB,iEAC9B,OAAOC,MAAc,sBACrB,OAAOC,MAAY,yBASZ,aAAM,qBAAqBV,EAAWC,CAAe,CAAE,CA+BnD,aAAc,CACjB,MAAM,EA1BV,KAAO,QAAU,GAGjB,KAAO,cAAgB,0BAGvB,KAAO,UAAY,sBAMnB,KAAQ,cAAmC,CACvC,cAAe,GACf,eAAgB,EACpB,EAuDA,KAAQ,aAAe,GA3CnB,KAAK,iBAAmB,IAAIM,EAAiB,KAAM,CAC/C,OAAQ,KACR,SAAU,IAAY,CAClB,KAAK,mBAAmB,CAC5B,CACJ,CAAC,CACL,CAtCA,WAA2B,QAAyB,CAChD,MAAO,CAACC,EAAmBE,EAAQD,CAAQ,CAC/C,CAsCmB,aAAaE,EAA+B,CAC3D,MAAM,aAAaA,CAAO,EAE1B,KAAM,CAACC,CAAI,EAAI,KAAK,cAChBA,IACAA,EAAK,iBAAmB,IAE5B,KAAK,iBAAiB,QAAQ,KAAK,iBAAiB,CACxD,CAEA,MAAc,mBAAmC,CAC7C,KAAM,CAACC,CAAW,EAAI,KAAK,cAC3B,MAAMA,GAAA,YAAAA,EAAa,gBACnB,KAAK,mBAAmB,CAC5B,CAEQ,oBAA2B,CAC/B,KAAM,CAAE,cAAAC,EAAe,cAAAC,CAAc,EAAI,KAEzC,GAAID,EAAe,CACf,KAAM,CAACD,CAAW,EAAI,KAAK,cACrB,CAAE,cAAAG,EAAe,eAAAC,CAAe,GAClCJ,GAAA,YAAAA,EAAa,cAAe,CACxB,cAAe,GACf,eAAgB,EACpB,EAEJ,KAAK,cAAgB,CACjB,GAAGE,EACH,cAAAC,EACA,eAAAC,CACJ,CACJ,CACJ,CAGQ,mBAAmBC,EAAyB,CAChD,MAAMC,EAAgBD,EAAM,cACtB,CAACL,CAAW,EAAI,KAAK,cAErBO,EAAOP,EAAY,YAAc,KAAK,aACtCQ,EAAOF,EAAc,UAAU,SAAS,aAAa,EACrD,CAACC,EACDA,EACNP,EAAY,WAAWQ,EAAM,QAAQ,CACzC,CAEmB,QACfC,EACI,CACJ,MAAM,QAAQA,CAAiB,EAC3BA,EAAkB,IAAI,KAAK,GAC3B,KAAK,mBAAmB,CAEhC,CAEmB,QAAyB,CACxC,KAAM,CAAE,eAAAL,EAAgB,cAAAD,CAAc,EAAI,KAAK,cACzCO,EAAoB,KAAK,cACzBC,EAAgB,KAAK,UAC3B,OAAOzB;AAAA;AAAA,wBAESO,EAAS,CACb,0BAA2B,GAC3B,cAAeU,EACf,eAAgBC,CACpB,CAAC,CAAC;AAAA;AAAA;AAAA,4BAGUX,EAAS,CACb,cAAe,GACf,KAAMU,CACV,CAAC,CAAC;AAAA,iCACWO,CAAiB;AAAA;AAAA;AAAA;AAAA,6BAIrB,KAAK,kBAAkB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,4BAQxBjB,EAAS,CACb,eAAgB,GAChB,KAAMW,CACV,CAAC,CAAC;AAAA,iCACWO,CAAa;AAAA;AAAA;AAAA,6BAGjB,KAAK,kBAAkB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,kCAQlB,KAAK,iBAAiB;AAAA,sCAClB,KAAK,kBAAkB;AAAA;AAAA;AAAA,SAIzD,CACJ,CA7IWC,EAAA,CADNvB,EAAS,CAAE,KAAM,QAAS,QAAS,EAAK,CAAC,GALjC,aAMF,uBAGAuB,EAAA,CADNvB,EAAS,CAAE,KAAM,OAAQ,UAAW,gBAAiB,CAAC,GAR9C,aASF,6BAGAuB,EAAA,CADNvB,EAAS,CAAE,KAAM,OAAQ,UAAW,YAAa,CAAC,GAX1C,aAYF,yBAGSuB,EAAA,CADfvB,EAAS,CAAE,QAAS,EAAK,CAAC,GAdlB,aAeO,mBAGRuB,EAAA,CADPpB,EAAM,GAjBE,aAkBD,6BAMAoB,EAAA,CADPrB,EAAsB,CAAE,SAAU,UAAW,QAAS,EAAK,CAAC,GAvBpD,aAwBD,6BAGAqB,EAAA,CADPtB,EAAM,0BAA0B,GA1BxB,aA2BD",
6
6
  "names": ["html", "SizedMixin", "SpectrumElement", "property", "query", "queryAssignedElements", "state", "classMap", "ResizeController", "chevronIconStyles", "tabSizes", "styles", "changes", "tabs", "tabsElement", "scrollContent", "overflowState", "canScrollLeft", "canScrollRight", "event", "currentTarget", "dist", "left", "changedProperties", "ariaLabelPrevious", "ariaLabelNext", "__decorateClass"]
7
7
  }
@@ -1,58 +1,66 @@
1
1
  "use strict";
2
- import "@spectrum-web-components/theme/sp-theme.js";
3
- import "@spectrum-web-components/theme/scale-medium.js";
4
- import "@spectrum-web-components/theme/theme-light.js";
5
- import "@spectrum-web-components/tabs/sp-tab.js";
6
- import "@spectrum-web-components/tabs/sp-tabs.js";
7
- import "@spectrum-web-components/tabs/sp-tab-panel.js";
8
- import "@spectrum-web-components/tabs/sp-tabs-overflow.js";
2
+ import { isFirefox } from "@spectrum-web-components/shared/src/platform.js";
9
3
  import {
10
4
  calculateScrollTargetForLeftSide,
11
5
  calculateScrollTargetForRightSide
12
6
  } from "@spectrum-web-components/tabs";
13
- import { elementUpdated, expect, fixture } from "@open-wc/testing";
7
+ import "@spectrum-web-components/tabs/sp-tab-panel.js";
8
+ import "@spectrum-web-components/tabs/sp-tab.js";
9
+ import "@spectrum-web-components/tabs/sp-tabs-overflow.js";
10
+ import "@spectrum-web-components/tabs/sp-tabs.js";
11
+ import "@spectrum-web-components/theme/scale-medium.js";
12
+ import "@spectrum-web-components/theme/sp-theme.js";
13
+ import "@spectrum-web-components/theme/theme-light.js";
14
+ import { elementUpdated, expect, fixture, waitUntil } from "@open-wc/testing";
14
15
  import {
15
16
  ElementSizes,
16
17
  html,
17
18
  nothing
18
19
  } from "@spectrum-web-components/base";
20
+ import { sendKeys, setViewport } from "@web/test-runner-commands";
19
21
  import { repeat } from "lit/directives/repeat.js";
22
+ const RIGHT_BUTTON_SELECTOR = ".right-scroll";
23
+ const LEFT_BUTTON_SELECTOR = ".left-scroll";
20
24
  const renderTabsOverflow = async ({
21
25
  count,
22
26
  size,
23
27
  includeTabPanel,
24
- selected = 1
28
+ selected = 1,
29
+ dir = "ltr"
25
30
  }) => {
26
- const tabsContainer = await fixture(html`
27
- <div class="container" style="width: 200px; height: 150px;">
28
- <sp-tabs-overflow>
29
- <sp-tabs size=${size} selected=${selected}>
30
- ${repeat(
31
+ const theme = await fixture(html`
32
+ <sp-theme dir=${dir} system="spectrum" scale="medium" color="light">
33
+ <div class="container" style="width: 200px; height: 150px;">
34
+ <sp-tabs-overflow>
35
+ <sp-tabs size=${size} selected=${selected}>
36
+ ${repeat(
31
37
  new Array(count),
32
38
  (item) => item,
33
39
  (_item, index) => html`
34
- <sp-tab
35
- label=${`Tab Item ${index + 1}`}
36
- value=${index + 1}
37
- ></sp-tab>
38
- `
40
+ <sp-tab
41
+ label=${`Tab Item ${index + 1}`}
42
+ value=${index + 1}
43
+ ></sp-tab>
44
+ `
39
45
  )}
40
- ${includeTabPanel ? html`
41
- ${repeat(
46
+ ${includeTabPanel ? html`
47
+ ${repeat(
42
48
  new Array(count),
43
49
  (item) => item,
44
50
  (_item, index) => html`
45
- <sp-tab-panel value=${index + 1}>
46
- Content for Tab Item ${index + 1}
47
- </sp-tab-panel>
48
- `
51
+ <sp-tab-panel value=${index + 1}>
52
+ Content for Tab Item ${index + 1}
53
+ </sp-tab-panel>
54
+ `
49
55
  )}
50
- ` : nothing}
51
- </sp-tabs>
52
- </sp-tabs-overflow>
53
- </div>
56
+ ` : nothing}
57
+ </sp-tabs>
58
+ </sp-tabs-overflow>
59
+ </div>
60
+ </sp-theme>
54
61
  `);
55
- await elementUpdated(tabsContainer);
62
+ await elementUpdated(theme);
63
+ const tabsContainer = theme.querySelector(".container");
56
64
  return tabsContainer;
57
65
  };
58
66
  describe("TabsOverflow", () => {
@@ -127,6 +135,52 @@ describe("TabsOverflow", () => {
127
135
  const finalLeft = tabsEl.getBoundingClientRect().left;
128
136
  expect(finalLeft).to.be.lessThanOrEqual(initialLeft);
129
137
  });
138
+ it("should scroll up to the last item and back in LTR", async () => {
139
+ if (isFirefox()) return;
140
+ const el = await renderTabsOverflow({
141
+ count: 8,
142
+ size: ElementSizes.L,
143
+ includeTabPanel: true,
144
+ dir: "ltr"
145
+ });
146
+ await elementUpdated(el);
147
+ await setViewport({ width: 360, height: 640 });
148
+ await nextFrame();
149
+ const tabsOverflow = el.querySelector(
150
+ "sp-tabs-overflow"
151
+ );
152
+ expect(tabsOverflow["overflowState"].canScrollLeft).to.be.false;
153
+ expect(tabsOverflow["overflowState"].canScrollRight).to.be.true;
154
+ await scrollToEnd(el, RIGHT_BUTTON_SELECTOR, "ltr");
155
+ expect(tabsOverflow["overflowState"].canScrollLeft).to.be.true;
156
+ expect(tabsOverflow["overflowState"].canScrollRight).to.be.false;
157
+ await scrollToEnd(el, LEFT_BUTTON_SELECTOR, "ltr");
158
+ expect(tabsOverflow["overflowState"].canScrollLeft).to.be.false;
159
+ expect(tabsOverflow["overflowState"].canScrollRight).to.be.true;
160
+ });
161
+ it("should scroll up to the last item and back in RTL", async () => {
162
+ if (isFirefox()) return;
163
+ const el = await renderTabsOverflow({
164
+ count: 8,
165
+ size: ElementSizes.L,
166
+ includeTabPanel: true,
167
+ dir: "rtl"
168
+ });
169
+ await elementUpdated(el);
170
+ await setViewport({ width: 360, height: 640 });
171
+ await nextFrame();
172
+ const tabsOverflow = el.querySelector(
173
+ "sp-tabs-overflow"
174
+ );
175
+ expect(tabsOverflow["overflowState"].canScrollLeft).to.be.true;
176
+ expect(tabsOverflow["overflowState"].canScrollRight).to.be.false;
177
+ await scrollToEnd(el, LEFT_BUTTON_SELECTOR, "rtl");
178
+ expect(tabsOverflow["overflowState"].canScrollLeft).to.be.false;
179
+ expect(tabsOverflow["overflowState"].canScrollRight).to.be.true;
180
+ await scrollToEnd(el, RIGHT_BUTTON_SELECTOR, "rtl");
181
+ expect(tabsOverflow["overflowState"].canScrollLeft).to.be.true;
182
+ expect(tabsOverflow["overflowState"].canScrollRight).to.be.false;
183
+ });
130
184
  it("should fail properly if slot is not sp-tabs", async () => {
131
185
  const el = await fixture(html`
132
186
  <sp-tabs-overflow>
@@ -285,4 +339,51 @@ describe("calculateScrollTargetForLeftSide", () => {
285
339
  ).to.equal(0);
286
340
  });
287
341
  });
342
+ async function repeatScroll(options, iteration = 1) {
343
+ const {
344
+ times,
345
+ elementToUpdate,
346
+ elementToScroll,
347
+ distanceToReachInIteration
348
+ } = options;
349
+ if (iteration > times) return;
350
+ const distanceToReach = distanceToReachInIteration(iteration);
351
+ await sendKeys({ press: "Enter" });
352
+ await elementUpdated(elementToUpdate);
353
+ await waitUntil(
354
+ () => Math.ceil(Math.abs(elementToScroll.scrollLeft)) - Math.abs(distanceToReach) === 0,
355
+ `scroll to ${distanceToReach}`
356
+ );
357
+ return await repeatScroll(options, iteration + 1);
358
+ }
359
+ async function scrollToEnd(tabsContainer, buttonSelector, direction = "ltr") {
360
+ const tabs = tabsContainer.querySelector("sp-tabs");
361
+ const tabsList = tabs.shadowRoot.querySelector("#list");
362
+ const tabsOverflow = tabsContainer.querySelector(
363
+ "sp-tabs-overflow"
364
+ );
365
+ const button = tabsOverflow.shadowRoot.querySelector(
366
+ buttonSelector
367
+ );
368
+ const { scrollWidth, clientWidth } = tabsList;
369
+ const distPerScroll = clientWidth * tabsOverflow["scrollFactor"];
370
+ const totalScrollDist = scrollWidth - clientWidth;
371
+ const scrollsToEnd = Math.ceil(totalScrollDist / distPerScroll);
372
+ let distanceToReachInIteration;
373
+ if (direction === "ltr") {
374
+ distanceToReachInIteration = buttonSelector === LEFT_BUTTON_SELECTOR ? (iteration) => Math.max(totalScrollDist - iteration * distPerScroll, 0) : (iteration) => Math.min(iteration * distPerScroll, totalScrollDist);
375
+ } else {
376
+ distanceToReachInIteration = buttonSelector === LEFT_BUTTON_SELECTOR ? (iteration) => Math.max(-1 * iteration * distPerScroll, -totalScrollDist) : (iteration) => -Math.max(totalScrollDist - iteration * distPerScroll, 0);
377
+ }
378
+ button.focus();
379
+ return await repeatScroll({
380
+ times: scrollsToEnd,
381
+ elementToUpdate: tabsOverflow,
382
+ elementToScroll: tabsList,
383
+ distanceToReachInIteration
384
+ });
385
+ }
386
+ function nextFrame() {
387
+ return new Promise((res) => requestAnimationFrame(() => res()));
388
+ }
288
389
  //# sourceMappingURL=tabs-overflow.test.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["tabs-overflow.test.ts"],
4
- "sourcesContent": ["/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\nimport '@spectrum-web-components/theme/sp-theme.js';\nimport '@spectrum-web-components/theme/scale-medium.js';\nimport '@spectrum-web-components/theme/theme-light.js';\nimport '@spectrum-web-components/tabs/sp-tab.js';\nimport '@spectrum-web-components/tabs/sp-tabs.js';\nimport '@spectrum-web-components/tabs/sp-tab-panel.js';\nimport '@spectrum-web-components/tabs/sp-tabs-overflow.js';\nimport {\n calculateScrollTargetForLeftSide,\n calculateScrollTargetForRightSide,\n Tab,\n Tabs,\n TabsOverflow,\n} from '@spectrum-web-components/tabs';\nimport { ActionButton } from '@spectrum-web-components/action-button';\n\nimport { elementUpdated, expect, fixture } from '@open-wc/testing';\nimport {\n ElementSize,\n ElementSizes,\n html,\n nothing,\n} from '@spectrum-web-components/base';\nimport { repeat } from 'lit/directives/repeat.js';\n\ntype OverflowProperties = {\n count: number;\n size: ElementSize;\n includeTabPanel: boolean;\n selected?: number;\n labelPrev?: string;\n labelNext?: string;\n};\n\nconst renderTabsOverflow = async ({\n count,\n size,\n includeTabPanel,\n selected = 1,\n}: OverflowProperties): Promise<HTMLDivElement> => {\n const tabsContainer = await fixture<HTMLDivElement>(html`\n <div class=\"container\" style=\"width: 200px; height: 150px;\">\n <sp-tabs-overflow>\n <sp-tabs size=${size} selected=${selected}>\n ${repeat(\n new Array(count),\n (item) => item,\n (_item, index) => html`\n <sp-tab\n label=${`Tab Item ${index + 1}`}\n value=${index + 1}\n ></sp-tab>\n `\n )}\n ${includeTabPanel\n ? html`\n ${repeat(\n new Array(count),\n (item) => item,\n (_item, index) => html`\n <sp-tab-panel value=${index + 1}>\n Content for Tab Item ${index + 1}\n </sp-tab-panel>\n `\n )}\n `\n : nothing}\n </sp-tabs>\n </sp-tabs-overflow>\n </div>\n `);\n await elementUpdated(tabsContainer);\n return tabsContainer;\n};\n\ndescribe('TabsOverflow', () => {\n it('loads default tabs-overflow accessibly', async () => {\n const el = await fixture<TabsOverflow>(html`\n <sp-tabs-overflow>\n <sp-tabs size=\"m\" selected=\"1\">\n <sp-tab label=\"Tab Item 1\" value=\"1\"></sp-tab>\n <sp-tab label=\"Tab Item 2\" value=\"2\"></sp-tab>\n <sp-tab-panel value=\"1\">Tab Content 1</sp-tab-panel>\n <sp-tab-panel value=\"2\">Tab Content 2</sp-tab-panel>\n </sp-tabs>\n </sp-tabs-overflow>\n `);\n\n await elementUpdated(el);\n\n await expect(el).to.be.accessible();\n });\n\n it('show render left and right buttons in shadowDom', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.L,\n includeTabPanel: true,\n });\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n expect(rightButton).to.exist;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n expect(leftButton).to.exist;\n });\n\n it('reflect proper sp-tab size', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.M,\n includeTabPanel: true,\n });\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n\n expect(spTabsOverflows.getAttribute('size')).to.equal('m');\n });\n\n it('should scroll when the button is clicked', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.L,\n includeTabPanel: true,\n });\n await elementUpdated(el);\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n\n leftButton.dispatchEvent(new Event('click', {}));\n\n const tabsEl = spTabsOverflows.querySelector('sp-tab') as Tab;\n const initialLeft = tabsEl.getBoundingClientRect().left;\n rightButton.dispatchEvent(new Event('click', {}));\n await elementUpdated(el);\n rightButton.dispatchEvent(new Event('click', {}));\n await elementUpdated(el);\n rightButton.dispatchEvent(new Event('click', {}));\n await elementUpdated(el);\n const finalLeft = tabsEl.getBoundingClientRect().left;\n expect(finalLeft).to.be.lessThanOrEqual(initialLeft);\n });\n\n it('should fail properly if slot is not sp-tabs', async () => {\n const el = await fixture<TabsOverflow>(html`\n <sp-tabs-overflow>\n <div>Some div</div>\n </sp-tabs-overflow>\n `);\n\n await elementUpdated(el);\n const slot = el.shadowRoot.querySelector('slot');\n const slotContent = slot?.assignedElements() || '';\n expect(slotContent[0].toString()).to.not.contains('Tabs');\n });\n\n it('should automatically bring the selected tab into view', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.L,\n includeTabPanel: false,\n selected: 10,\n });\n await elementUpdated(el);\n\n // Grab the list of tabs.\n const tabsEl = el.querySelector('sp-tabs') as Tabs;\n\n // Grab the coordonates of the selected tab.\n let selectedTab = tabsEl.querySelector(\n `[role=\"tab\"][value=\"10\"]`\n ) as Tab;\n expect(selectedTab).to.exist;\n let selectedTabPosition = selectedTab.getBoundingClientRect();\n\n // Selected tab is in the viewport, offset left is greater than 0 and less than the width of the tabs.\n expect(selectedTabPosition.left).to.be.greaterThan(0);\n expect(selectedTabPosition.left).to.be.lessThan(tabsEl.offsetWidth);\n\n // First tab is not in the viewport anymore, its offset left is less than 0.\n const firstTab = tabsEl.querySelector(`[role=\"tab\"][value=\"1\"]`) as Tab;\n const firstTabPosition = firstTab.getBoundingClientRect();\n expect(firstTabPosition.left).to.be.lessThan(0);\n\n // Make the component automatically scroll left by selecting the first tab.\n tabsEl.selected = '1';\n await elementUpdated(tabsEl);\n\n selectedTab = tabsEl.querySelector(`[role=\"tab\"][value=\"1\"]`) as Tab;\n expect(selectedTab).to.exist;\n selectedTabPosition = selectedTab.getBoundingClientRect();\n\n // First tab is in the viewport, offset left is greater than 0 and less than the width of the tabs.\n expect(selectedTabPosition.left).to.be.greaterThan(0);\n expect(selectedTabPosition.left).to.be.lessThan(tabsEl.offsetWidth);\n\n // Tab nr. 10 is not in the viewport anymore.\n const previousSelection = tabsEl.querySelector(\n `[role=\"tab\"][value=\"10\"]`\n ) as Tab;\n const previousSelectionPosition =\n previousSelection.getBoundingClientRect();\n expect(previousSelectionPosition.left).to.be.greaterThan(\n tabsEl.offsetWidth\n );\n });\n\n it('prev and next buttons have default labels', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.M,\n includeTabPanel: true,\n });\n await elementUpdated(el);\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n\n expect(leftButton?.getAttribute('aria-label')).to.equal(\n 'Scroll to previous tabs'\n );\n expect(rightButton?.getAttribute('aria-label')).to.equal(\n 'Scroll to next tabs'\n );\n });\n\n it('prev and next buttons labels overwritten via attributes', async () => {\n const tabsContainer = await fixture<HTMLDivElement>(html`\n <div class=\"container\" style=\"width: 200px; height: 150px;\">\n <sp-tabs-overflow\n label-previous=\"custom label prev\"\n label-next=\"custom label next\"\n >\n <sp-tabs size=${ElementSizes.M} selected=${1}>\n ${repeat(\n new Array(20),\n (item) => item,\n (_item, index) => html`\n <sp-tab\n label=${`Tab Item ${index + 1}`}\n value=${index + 1}\n ></sp-tab>\n `\n )}\n ${repeat(\n new Array(20),\n (item) => item,\n (_item, index) => html`\n <sp-tab-panel value=${index + 1}>\n Content for Tab Item ${index + 1}\n </sp-tab-panel>\n `\n )}\n </sp-tabs>\n </sp-tabs-overflow>\n </div>\n `);\n await elementUpdated(tabsContainer);\n const el = tabsContainer;\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n\n expect(leftButton?.getAttribute('aria-label')).to.equal(\n 'custom label prev'\n );\n expect(rightButton?.getAttribute('aria-label')).to.equal(\n 'custom label next'\n );\n });\n});\n\ndescribe('calculateScrollTargetForRightSide', () => {\n const container = { offsetWidth: 100, scrollLeft: 0 } as HTMLDivElement;\n const tabs = [\n { offsetLeft: 0, offsetWidth: 100 }, // currently selected tab\n { offsetLeft: 100, offsetWidth: 100 },\n { offsetLeft: 200, offsetWidth: 100 },\n ] as Tab[];\n\n it('correctly aligns tab on the right side of the viewport', () => {\n // Where do I need to scroll on the x axis to get the tab at index 2 to be visible?\n expect(\n calculateScrollTargetForRightSide(2, 'ltr', tabs, container)\n ).to.equal(100); // You need to scroll 100px more\n\n // Repeat for RTL\n expect(\n calculateScrollTargetForRightSide(2, 'rtl', tabs, container)\n ).to.equal(0); // You need to scroll at the begining of the scrollable area\n });\n});\n\ndescribe('calculateScrollTargetForLeftSide', () => {\n const container = { offsetWidth: 100, scrollLeft: 200 } as HTMLDivElement;\n const tabs = [\n { offsetLeft: -200, offsetWidth: 100 },\n { offsetLeft: -100, offsetWidth: 100 },\n { offsetLeft: 0, offsetWidth: 100 }, // currently selected tab\n ] as Tab[];\n\n it('correctly aligns tab on the left side of the viewport', () => {\n // Where do I need to scroll on the x axis to get the tab at index 1 to be visible?\n expect(\n calculateScrollTargetForLeftSide(1, 'ltr', tabs, container)\n ).to.equal(-100); // you need to scroll back -100px\n\n // Where do I need to scroll on the x axis to get the first tab to be visible?\n expect(\n calculateScrollTargetForLeftSide(0, 'ltr', tabs, container)\n ).to.equal(0); // you need to scroll to the begining of the scrollable area\n\n // Repeat for RTL\n expect(\n calculateScrollTargetForLeftSide(1, 'rtl', tabs, container)\n ).to.equal(100);\n\n expect(\n calculateScrollTargetForLeftSide(0, 'rtl', tabs, container)\n ).to.equal(0);\n });\n});\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\nimport { ActionButton } from '@spectrum-web-components/action-button';\nimport { isFirefox } from '@spectrum-web-components/shared/src/platform.js';\nimport {\n calculateScrollTargetForLeftSide,\n calculateScrollTargetForRightSide,\n Tab,\n Tabs,\n TabsOverflow,\n} from '@spectrum-web-components/tabs';\nimport '@spectrum-web-components/tabs/sp-tab-panel.js';\nimport '@spectrum-web-components/tabs/sp-tab.js';\nimport '@spectrum-web-components/tabs/sp-tabs-overflow.js';\nimport '@spectrum-web-components/tabs/sp-tabs.js';\nimport '@spectrum-web-components/theme/scale-medium.js';\nimport '@spectrum-web-components/theme/sp-theme.js';\nimport '@spectrum-web-components/theme/theme-light.js';\n\nimport { elementUpdated, expect, fixture, waitUntil } from '@open-wc/testing';\nimport {\n ElementSize,\n ElementSizes,\n html,\n nothing,\n} from '@spectrum-web-components/base';\nimport { sendKeys, setViewport } from '@web/test-runner-commands';\nimport { repeat } from 'lit/directives/repeat.js';\n\nconst RIGHT_BUTTON_SELECTOR = '.right-scroll';\nconst LEFT_BUTTON_SELECTOR = '.left-scroll';\n\ntype OverflowProperties = {\n count: number;\n size: ElementSize;\n includeTabPanel: boolean;\n selected?: number;\n labelPrev?: string;\n labelNext?: string;\n dir?: 'ltr' | 'rtl';\n};\n\nconst renderTabsOverflow = async ({\n count,\n size,\n includeTabPanel,\n selected = 1,\n dir = 'ltr',\n}: OverflowProperties): Promise<HTMLDivElement> => {\n const theme = await fixture<HTMLDivElement>(html`\n <sp-theme dir=${dir} system=\"spectrum\" scale=\"medium\" color=\"light\">\n <div class=\"container\" style=\"width: 200px; height: 150px;\">\n <sp-tabs-overflow>\n <sp-tabs size=${size} selected=${selected}>\n ${repeat(\n new Array(count),\n (item) => item,\n (_item, index) => html`\n <sp-tab\n label=${`Tab Item ${index + 1}`}\n value=${index + 1}\n ></sp-tab>\n `\n )}\n ${includeTabPanel\n ? html`\n ${repeat(\n new Array(count),\n (item) => item,\n (_item, index) => html`\n <sp-tab-panel value=${index + 1}>\n Content for Tab Item ${index + 1}\n </sp-tab-panel>\n `\n )}\n `\n : nothing}\n </sp-tabs>\n </sp-tabs-overflow>\n </div>\n </sp-theme>\n `);\n await elementUpdated(theme);\n const tabsContainer = theme.querySelector('.container') as HTMLDivElement;\n\n return tabsContainer;\n};\n\ndescribe('TabsOverflow', () => {\n it('loads default tabs-overflow accessibly', async () => {\n const el = await fixture<TabsOverflow>(html`\n <sp-tabs-overflow>\n <sp-tabs size=\"m\" selected=\"1\">\n <sp-tab label=\"Tab Item 1\" value=\"1\"></sp-tab>\n <sp-tab label=\"Tab Item 2\" value=\"2\"></sp-tab>\n <sp-tab-panel value=\"1\">Tab Content 1</sp-tab-panel>\n <sp-tab-panel value=\"2\">Tab Content 2</sp-tab-panel>\n </sp-tabs>\n </sp-tabs-overflow>\n `);\n\n await elementUpdated(el);\n\n await expect(el).to.be.accessible();\n });\n\n it('show render left and right buttons in shadowDom', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.L,\n includeTabPanel: true,\n });\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n expect(rightButton).to.exist;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n expect(leftButton).to.exist;\n });\n\n it('reflect proper sp-tab size', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.M,\n includeTabPanel: true,\n });\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n\n expect(spTabsOverflows.getAttribute('size')).to.equal('m');\n });\n\n it('should scroll when the button is clicked', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.L,\n includeTabPanel: true,\n });\n await elementUpdated(el);\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n\n leftButton.dispatchEvent(new Event('click', {}));\n\n const tabsEl = spTabsOverflows.querySelector('sp-tab') as Tab;\n const initialLeft = tabsEl.getBoundingClientRect().left;\n rightButton.dispatchEvent(new Event('click', {}));\n await elementUpdated(el);\n rightButton.dispatchEvent(new Event('click', {}));\n await elementUpdated(el);\n rightButton.dispatchEvent(new Event('click', {}));\n await elementUpdated(el);\n const finalLeft = tabsEl.getBoundingClientRect().left;\n expect(finalLeft).to.be.lessThanOrEqual(initialLeft);\n });\n\n it('should scroll up to the last item and back in LTR', async () => {\n // TODO: run on iPhone as per https://github.com/adobe/spectrum-web-components/pull/4722\n // await setUserAgent(\n // 'Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148'\n // );\n\n // Skipping on Firefox due to timeouts on CI\n if (isFirefox()) return;\n\n const el = await renderTabsOverflow({\n count: 8,\n size: ElementSizes.L,\n includeTabPanel: true,\n dir: 'ltr',\n });\n await elementUpdated(el);\n await setViewport({ width: 360, height: 640 });\n await nextFrame();\n\n const tabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n\n expect(tabsOverflow['overflowState'].canScrollLeft).to.be.false;\n expect(tabsOverflow['overflowState'].canScrollRight).to.be.true;\n\n await scrollToEnd(el, RIGHT_BUTTON_SELECTOR, 'ltr');\n\n expect(tabsOverflow['overflowState'].canScrollLeft).to.be.true;\n expect(tabsOverflow['overflowState'].canScrollRight).to.be.false;\n\n await scrollToEnd(el, LEFT_BUTTON_SELECTOR, 'ltr');\n\n expect(tabsOverflow['overflowState'].canScrollLeft).to.be.false;\n expect(tabsOverflow['overflowState'].canScrollRight).to.be.true;\n });\n\n it('should scroll up to the last item and back in RTL', async () => {\n // TODO: run on iPhone as per https://github.com/adobe/spectrum-web-components/pull/4722\n // await setUserAgent(\n // 'Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148'\n // );\n\n // Skipping on Firefox due to timeouts on CI\n if (isFirefox()) return;\n\n const el = await renderTabsOverflow({\n count: 8,\n size: ElementSizes.L,\n includeTabPanel: true,\n dir: 'rtl',\n });\n await elementUpdated(el);\n await setViewport({ width: 360, height: 640 });\n await nextFrame();\n\n const tabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n\n expect(tabsOverflow['overflowState'].canScrollLeft).to.be.true;\n expect(tabsOverflow['overflowState'].canScrollRight).to.be.false;\n\n await scrollToEnd(el, LEFT_BUTTON_SELECTOR, 'rtl');\n\n expect(tabsOverflow['overflowState'].canScrollLeft).to.be.false;\n expect(tabsOverflow['overflowState'].canScrollRight).to.be.true;\n\n await scrollToEnd(el, RIGHT_BUTTON_SELECTOR, 'rtl');\n\n expect(tabsOverflow['overflowState'].canScrollLeft).to.be.true;\n expect(tabsOverflow['overflowState'].canScrollRight).to.be.false;\n });\n\n it('should fail properly if slot is not sp-tabs', async () => {\n const el = await fixture<TabsOverflow>(html`\n <sp-tabs-overflow>\n <div>Some div</div>\n </sp-tabs-overflow>\n `);\n\n await elementUpdated(el);\n const slot = el.shadowRoot.querySelector('slot');\n const slotContent = slot?.assignedElements() || '';\n expect(slotContent[0].toString()).to.not.contains('Tabs');\n });\n\n it('should automatically bring the selected tab into view', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.L,\n includeTabPanel: false,\n selected: 10,\n });\n await elementUpdated(el);\n\n // Grab the list of tabs.\n const tabsEl = el.querySelector('sp-tabs') as Tabs;\n\n // Grab the coordonates of the selected tab.\n let selectedTab = tabsEl.querySelector(\n `[role=\"tab\"][value=\"10\"]`\n ) as Tab;\n expect(selectedTab).to.exist;\n let selectedTabPosition = selectedTab.getBoundingClientRect();\n\n // Selected tab is in the viewport, offset left is greater than 0 and less than the width of the tabs.\n expect(selectedTabPosition.left).to.be.greaterThan(0);\n expect(selectedTabPosition.left).to.be.lessThan(tabsEl.offsetWidth);\n\n // First tab is not in the viewport anymore, its offset left is less than 0.\n const firstTab = tabsEl.querySelector(`[role=\"tab\"][value=\"1\"]`) as Tab;\n const firstTabPosition = firstTab.getBoundingClientRect();\n expect(firstTabPosition.left).to.be.lessThan(0);\n\n // Make the component automatically scroll left by selecting the first tab.\n tabsEl.selected = '1';\n await elementUpdated(tabsEl);\n\n selectedTab = tabsEl.querySelector(`[role=\"tab\"][value=\"1\"]`) as Tab;\n expect(selectedTab).to.exist;\n selectedTabPosition = selectedTab.getBoundingClientRect();\n\n // First tab is in the viewport, offset left is greater than 0 and less than the width of the tabs.\n expect(selectedTabPosition.left).to.be.greaterThan(0);\n expect(selectedTabPosition.left).to.be.lessThan(tabsEl.offsetWidth);\n\n // Tab nr. 10 is not in the viewport anymore.\n const previousSelection = tabsEl.querySelector(\n `[role=\"tab\"][value=\"10\"]`\n ) as Tab;\n const previousSelectionPosition =\n previousSelection.getBoundingClientRect();\n expect(previousSelectionPosition.left).to.be.greaterThan(\n tabsEl.offsetWidth\n );\n });\n\n it('prev and next buttons have default labels', async () => {\n const el = await renderTabsOverflow({\n count: 20,\n size: ElementSizes.M,\n includeTabPanel: true,\n });\n await elementUpdated(el);\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n\n expect(leftButton?.getAttribute('aria-label')).to.equal(\n 'Scroll to previous tabs'\n );\n expect(rightButton?.getAttribute('aria-label')).to.equal(\n 'Scroll to next tabs'\n );\n });\n\n it('prev and next buttons labels overwritten via attributes', async () => {\n const tabsContainer = await fixture<HTMLDivElement>(html`\n <div class=\"container\" style=\"width: 200px; height: 150px;\">\n <sp-tabs-overflow\n label-previous=\"custom label prev\"\n label-next=\"custom label next\"\n >\n <sp-tabs size=${ElementSizes.M} selected=${1}>\n ${repeat(\n new Array(20),\n (item) => item,\n (_item, index) => html`\n <sp-tab\n label=${`Tab Item ${index + 1}`}\n value=${index + 1}\n ></sp-tab>\n `\n )}\n ${repeat(\n new Array(20),\n (item) => item,\n (_item, index) => html`\n <sp-tab-panel value=${index + 1}>\n Content for Tab Item ${index + 1}\n </sp-tab-panel>\n `\n )}\n </sp-tabs>\n </sp-tabs-overflow>\n </div>\n `);\n await elementUpdated(tabsContainer);\n const el = tabsContainer;\n\n const spTabsOverflows: TabsOverflow = el.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const leftButton = spTabsOverflows.shadowRoot.querySelector(\n '.left-scroll'\n ) as ActionButton;\n const rightButton = spTabsOverflows.shadowRoot.querySelector(\n '.right-scroll'\n ) as ActionButton;\n\n expect(leftButton?.getAttribute('aria-label')).to.equal(\n 'custom label prev'\n );\n expect(rightButton?.getAttribute('aria-label')).to.equal(\n 'custom label next'\n );\n });\n});\n\ndescribe('calculateScrollTargetForRightSide', () => {\n const container = { offsetWidth: 100, scrollLeft: 0 } as HTMLDivElement;\n const tabs = [\n { offsetLeft: 0, offsetWidth: 100 }, // currently selected tab\n { offsetLeft: 100, offsetWidth: 100 },\n { offsetLeft: 200, offsetWidth: 100 },\n ] as Tab[];\n\n it('correctly aligns tab on the right side of the viewport', () => {\n // Where do I need to scroll on the x axis to get the tab at index 2 to be visible?\n expect(\n calculateScrollTargetForRightSide(2, 'ltr', tabs, container)\n ).to.equal(100); // You need to scroll 100px more\n\n // Repeat for RTL\n expect(\n calculateScrollTargetForRightSide(2, 'rtl', tabs, container)\n ).to.equal(0); // You need to scroll at the begining of the scrollable area\n });\n});\n\ndescribe('calculateScrollTargetForLeftSide', () => {\n const container = { offsetWidth: 100, scrollLeft: 200 } as HTMLDivElement;\n const tabs = [\n { offsetLeft: -200, offsetWidth: 100 },\n { offsetLeft: -100, offsetWidth: 100 },\n { offsetLeft: 0, offsetWidth: 100 }, // currently selected tab\n ] as Tab[];\n\n it('correctly aligns tab on the left side of the viewport', () => {\n // Where do I need to scroll on the x axis to get the tab at index 1 to be visible?\n expect(\n calculateScrollTargetForLeftSide(1, 'ltr', tabs, container)\n ).to.equal(-100); // you need to scroll back -100px\n\n // Where do I need to scroll on the x axis to get the first tab to be visible?\n expect(\n calculateScrollTargetForLeftSide(0, 'ltr', tabs, container)\n ).to.equal(0); // you need to scroll to the begining of the scrollable area\n\n // Repeat for RTL\n expect(\n calculateScrollTargetForLeftSide(1, 'rtl', tabs, container)\n ).to.equal(100);\n\n expect(\n calculateScrollTargetForLeftSide(0, 'rtl', tabs, container)\n ).to.equal(0);\n });\n});\n\nasync function repeatScroll(\n options: {\n times: number;\n elementToUpdate: TabsOverflow;\n elementToScroll: HTMLElement;\n distanceToReachInIteration: (iteration: number) => number;\n },\n iteration = 1\n): Promise<void> {\n const {\n times,\n elementToUpdate,\n elementToScroll,\n distanceToReachInIteration,\n } = options;\n if (iteration > times) return;\n\n const distanceToReach = distanceToReachInIteration(iteration);\n\n await sendKeys({ press: 'Enter' });\n await elementUpdated(elementToUpdate);\n await waitUntil(\n () =>\n Math.ceil(Math.abs(elementToScroll.scrollLeft)) -\n Math.abs(distanceToReach) ===\n 0,\n `scroll to ${distanceToReach}`\n );\n return await repeatScroll(options, iteration + 1);\n}\n\nasync function scrollToEnd(\n tabsContainer: HTMLDivElement,\n buttonSelector: string,\n direction: 'ltr' | 'rtl' = 'ltr'\n): Promise<void> {\n const tabs = tabsContainer.querySelector('sp-tabs') as Tabs;\n const tabsList = tabs.shadowRoot!.querySelector('#list') as HTMLElement;\n const tabsOverflow = tabsContainer.querySelector(\n 'sp-tabs-overflow'\n ) as TabsOverflow;\n const button = tabsOverflow.shadowRoot.querySelector(\n buttonSelector\n ) as ActionButton;\n\n const { scrollWidth, clientWidth } = tabsList;\n const distPerScroll = clientWidth * tabsOverflow['scrollFactor'];\n const totalScrollDist = scrollWidth - clientWidth;\n const scrollsToEnd = Math.ceil(totalScrollDist / distPerScroll);\n let distanceToReachInIteration: (iteration: number) => number;\n\n if (direction === 'ltr') {\n distanceToReachInIteration =\n buttonSelector === LEFT_BUTTON_SELECTOR\n ? (iteration: number) =>\n Math.max(totalScrollDist - iteration * distPerScroll, 0)\n : (iteration: number) =>\n Math.min(iteration * distPerScroll, totalScrollDist);\n } else {\n distanceToReachInIteration =\n buttonSelector === LEFT_BUTTON_SELECTOR\n ? (iteration: number) =>\n Math.max(-1 * iteration * distPerScroll, -totalScrollDist)\n : (iteration: number) =>\n -Math.max(totalScrollDist - iteration * distPerScroll, 0);\n }\n\n button.focus();\n return await repeatScroll({\n times: scrollsToEnd,\n elementToUpdate: tabsOverflow,\n elementToScroll: tabsList,\n distanceToReachInIteration,\n });\n}\n\nfunction nextFrame(): Promise<void> {\n return new Promise((res) => requestAnimationFrame(() => res()));\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }