@sbb-esta/lyne-elements 1.12.1 → 1.13.0

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.
Files changed (100) hide show
  1. package/autocomplete-grid/autocomplete-grid-button.js +1 -1
  2. package/button/common.js +1 -1
  3. package/button/mini-button.js +5 -5
  4. package/core/mixins/form-associated-checkbox-mixin.d.ts +1 -0
  5. package/core/mixins/form-associated-checkbox-mixin.d.ts.map +1 -1
  6. package/core/mixins/form-associated-mixin.d.ts +6 -6
  7. package/core/mixins/form-associated-mixin.d.ts.map +1 -1
  8. package/core/mixins/required-mixin.d.ts +1 -1
  9. package/core/mixins/required-mixin.d.ts.map +1 -1
  10. package/core/mixins.js +2 -6
  11. package/core/styles/mixins/buttons.scss +1 -1
  12. package/core/styles/node_modules_@sbb-esta_lyne-design-tokens_dist_scss_sbb-variables.scss +4 -0
  13. package/core/styles/node_modules_@sbb-esta_lyne-design-tokens_dist_scss_sbb-variables_css--mixin.scss +5 -1
  14. package/core.css +5 -1
  15. package/custom-elements.json +526 -267
  16. package/datepicker/datepicker-next-day.js +1 -1
  17. package/datepicker/datepicker-previous-day.js +1 -1
  18. package/development/autocomplete-grid/autocomplete-grid-button.js +1 -1
  19. package/development/button/common.js +1 -1
  20. package/development/button/mini-button.js +1 -1
  21. package/development/core/mixins/form-associated-checkbox-mixin.d.ts +1 -0
  22. package/development/core/mixins/form-associated-checkbox-mixin.d.ts.map +1 -1
  23. package/development/core/mixins/form-associated-mixin.d.ts +6 -6
  24. package/development/core/mixins/form-associated-mixin.d.ts.map +1 -1
  25. package/development/core/mixins/required-mixin.d.ts +1 -1
  26. package/development/core/mixins/required-mixin.d.ts.map +1 -1
  27. package/development/core/mixins.js +3 -6
  28. package/development/datepicker/datepicker-next-day.js +1 -1
  29. package/development/datepicker/datepicker-previous-day.js +1 -1
  30. package/development/file-selector/file-selector.d.ts +18 -8
  31. package/development/file-selector/file-selector.d.ts.map +1 -1
  32. package/development/file-selector.js +72 -36
  33. package/development/form-field/form-field-clear.js +1 -1
  34. package/development/header/common.js +52 -5
  35. package/development/image.js +1 -1
  36. package/development/loading-indicator.js +2 -6
  37. package/development/paginator/paginator.d.ts +1 -1
  38. package/development/paginator/paginator.d.ts.map +1 -1
  39. package/development/paginator.js +36 -14
  40. package/development/popover/popover-trigger.js +1 -1
  41. package/development/select/select.d.ts +12 -5
  42. package/development/select/select.d.ts.map +1 -1
  43. package/development/select.js +50 -17
  44. package/development/slider/slider.d.ts +2 -1
  45. package/development/slider/slider.d.ts.map +1 -1
  46. package/development/slider.js +4 -1
  47. package/file-selector/file-selector.d.ts +18 -8
  48. package/file-selector/file-selector.d.ts.map +1 -1
  49. package/file-selector.js +91 -67
  50. package/form-field/form-field-clear.js +1 -1
  51. package/header/common.js +11 -11
  52. package/loading-indicator.js +5 -5
  53. package/package.json +3 -3
  54. package/paginator/paginator.d.ts +1 -1
  55. package/paginator/paginator.d.ts.map +1 -1
  56. package/paginator.js +57 -55
  57. package/popover/popover-trigger.js +1 -1
  58. package/select/select.d.ts +12 -5
  59. package/select/select.d.ts.map +1 -1
  60. package/select.js +111 -83
  61. package/slider/slider.d.ts +2 -1
  62. package/slider/slider.d.ts.map +1 -1
  63. package/slider.js +3 -0
  64. package/standard-theme.css +5 -1
  65. package/autocomplete-grid/autocomplete-grid-optgroup/index.d.ts +0 -2
  66. package/autocomplete-grid/autocomplete-grid-optgroup/index.d.ts.map +0 -1
  67. package/button/mini-button-group/index.d.ts +0 -2
  68. package/button/mini-button-group/index.d.ts.map +0 -1
  69. package/development/autocomplete-grid/autocomplete-grid-optgroup/index.d.ts +0 -2
  70. package/development/autocomplete-grid/autocomplete-grid-optgroup/index.d.ts.map +0 -1
  71. package/development/button/mini-button-group/index.d.ts +0 -2
  72. package/development/button/mini-button-group/index.d.ts.map +0 -1
  73. package/development/flip-card/flip-card/index.d.ts +0 -2
  74. package/development/flip-card/flip-card/index.d.ts.map +0 -1
  75. package/development/flip-card/flip-card-details/index.d.ts +0 -2
  76. package/development/flip-card/flip-card-details/index.d.ts.map +0 -1
  77. package/development/flip-card/flip-card-summary/index.d.ts +0 -2
  78. package/development/flip-card/flip-card-summary/index.d.ts.map +0 -1
  79. package/development/link-list/link-list-anchor/index.d.ts +0 -2
  80. package/development/link-list/link-list-anchor/index.d.ts.map +0 -1
  81. package/development/paginator/index.d.ts +0 -2
  82. package/development/paginator/index.d.ts.map +0 -1
  83. package/development/radio-button/radio-button-panel/index.d.ts +0 -2
  84. package/development/radio-button/radio-button-panel/index.d.ts.map +0 -1
  85. package/development/tabs/tab/index.d.ts +0 -2
  86. package/development/tabs/tab/index.d.ts.map +0 -1
  87. package/flip-card/flip-card/index.d.ts +0 -2
  88. package/flip-card/flip-card/index.d.ts.map +0 -1
  89. package/flip-card/flip-card-details/index.d.ts +0 -2
  90. package/flip-card/flip-card-details/index.d.ts.map +0 -1
  91. package/flip-card/flip-card-summary/index.d.ts +0 -2
  92. package/flip-card/flip-card-summary/index.d.ts.map +0 -1
  93. package/link-list/link-list-anchor/index.d.ts +0 -2
  94. package/link-list/link-list-anchor/index.d.ts.map +0 -1
  95. package/paginator/index.d.ts +0 -2
  96. package/paginator/index.d.ts.map +0 -1
  97. package/radio-button/radio-button-panel/index.d.ts +0 -2
  98. package/radio-button/radio-button-panel/index.d.ts.map +0 -1
  99. package/tabs/tab/index.d.ts +0 -2
  100. package/tabs/tab/index.d.ts.map +0 -1
@@ -1,13 +1,13 @@
1
1
  import { css, LitElement, nothing } from "lit";
2
- import { property, state, customElement } from "lit/decorators.js";
2
+ import { property, customElement } from "lit/decorators.js";
3
3
  import { ref } from "lit/directives/ref.js";
4
4
  import { html, unsafeStatic } from "lit/static-html.js";
5
5
  import { sbbInputModalityDetector } from "./core/a11y.js";
6
6
  import { SbbLanguageController } from "./core/controllers.js";
7
7
  import { slotState } from "./core/decorators.js";
8
- import { EventEmitter } from "./core/eventing.js";
8
+ import { EventEmitter, forwardEventToHost } from "./core/eventing.js";
9
9
  import { i18nFileSelectorCurrentlySelected, i18nFileSelectorButtonLabel, i18nFileSelectorSubtitleLabel, i18nFileSelectorDeleteFile } from "./core/i18n.js";
10
- import { SbbDisabledMixin } from "./core/mixins.js";
10
+ import { SbbDisabledMixin, SbbFormAssociatedMixin } from "./core/mixins.js";
11
11
  import "./button/secondary-button.js";
12
12
  import "./button/secondary-button-static.js";
13
13
  import "./icon.js";
@@ -32,7 +32,7 @@ const style = css`*,
32
32
  }
33
33
 
34
34
  @media (forced-colors: active) {
35
- :host([disabled]) {
35
+ :host(:disabled) {
36
36
  --sbb-file-selector-color: GrayText;
37
37
  --sbb-file-selector-subtitle-color: GrayText;
38
38
  --sbb-file-selector-border-color: GrayText;
@@ -169,13 +169,14 @@ var __decorateClass = (decorators, target, key, kind) => {
169
169
  if (kind && result) __defProp(target, key, result);
170
170
  return result;
171
171
  };
172
- let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
172
+ let SbbFileSelectorElement = class extends SbbDisabledMixin(SbbFormAssociatedMixin(LitElement)) {
173
173
  constructor() {
174
174
  super(...arguments);
175
175
  this.variant = "default";
176
176
  this.size = "m";
177
177
  this.multiple = false;
178
178
  this.multipleMode = "default";
179
+ this._files = [];
179
180
  this._fileChangedEvent = new EventEmitter(
180
181
  this,
181
182
  SbbFileSelectorElement.events.fileChangedEvent
@@ -184,17 +185,45 @@ let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
184
185
  this._suffixes = ["B", "kB", "MB", "GB", "TB"];
185
186
  this._language = new SbbLanguageController(this);
186
187
  }
187
- /**
188
- * Gets the currently selected files.
189
- */
188
+ set value(value) {
189
+ this._hiddenInput.value = value ?? "";
190
+ if (!value) {
191
+ this.files = [];
192
+ }
193
+ }
194
+ get value() {
195
+ var _a;
196
+ return (_a = this._hiddenInput) == null ? void 0 : _a.value;
197
+ }
198
+ set files(value) {
199
+ this._files = value ?? [];
200
+ const dt = new DataTransfer();
201
+ this.files.forEach((e) => dt.items.add(e));
202
+ this._hiddenInput.files = dt.files;
203
+ this.updateFormValue();
204
+ }
190
205
  get files() {
191
- return this._files || [];
206
+ return this._files;
192
207
  }
193
208
  /**
194
- * @deprecated use 'files' property instead
209
+ * @deprecated use the 'files' property instead
195
210
  */
196
211
  getFiles() {
197
- return this._files || [];
212
+ return this.files;
213
+ }
214
+ formResetCallback() {
215
+ this.files = [];
216
+ }
217
+ formStateRestoreCallback(state, _reason) {
218
+ if (!state) {
219
+ return;
220
+ }
221
+ this.files = state.map(([_, value]) => value);
222
+ }
223
+ updateFormValue() {
224
+ const formValue = new FormData();
225
+ this.files.forEach((file) => formValue.append(this.name, file));
226
+ this.internals.setFormValue(formValue);
198
227
  }
199
228
  _blockEvent(event) {
200
229
  event.stopPropagation();
@@ -205,21 +234,21 @@ let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
205
234
  }
206
235
  _onDragEnter(event) {
207
236
  this._counter++;
208
- if (!this.disabled) {
237
+ if (!this.disabled && !this.formDisabled) {
209
238
  this._setDragState(event.target, true);
210
239
  this._blockEvent(event);
211
240
  }
212
241
  }
213
242
  _onDragLeave(event) {
214
243
  this._counter--;
215
- if (!this.disabled && event.target === this._dragTarget && this._counter === 0) {
244
+ if (!this.disabled && !this.formDisabled && event.target === this._dragTarget && this._counter === 0) {
216
245
  this._setDragState();
217
246
  this._blockEvent(event);
218
247
  }
219
248
  }
220
249
  _onFileDrop(event) {
221
250
  this._counter = 0;
222
- if (!this.disabled) {
251
+ if (!this.disabled && !this.formDisabled) {
223
252
  this._setDragState();
224
253
  this._blockEvent(event);
225
254
  this._createFileList(event.dataTransfer.files);
@@ -245,37 +274,41 @@ let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
245
274
  if (fileInput.files) {
246
275
  this._createFileList(fileInput.files);
247
276
  }
277
+ forwardEventToHost(event, this);
248
278
  }
249
279
  _createFileList(files) {
250
- if (!this.multiple || this.multipleMode !== "persistent" || !this._files || this._files.length === 0) {
251
- this._files = Array.from(files);
280
+ if (!this.multiple || this.multipleMode !== "persistent" || this.files.length === 0) {
281
+ this.files = Array.from(files);
252
282
  } else {
253
- this._files = Array.from(files).filter(
254
- (newFile) => this._files.findIndex((oldFile) => this._checkFileEquality(newFile, oldFile)) === -1
255
- ).concat(this._files);
283
+ this.files = Array.from(files).filter(
284
+ // Remove duplicates
285
+ (newFile) => this.files.findIndex((oldFile) => this._checkFileEquality(newFile, oldFile)) === -1
286
+ ).concat(this.files);
256
287
  }
257
- this._liveRegion.innerText = i18nFileSelectorCurrentlySelected(this._files.map((e) => e.name))[this._language.current];
258
- this._fileChangedEvent.emit(this._files);
288
+ this._updateA11yLiveRegion();
289
+ this._fileChangedEvent.emit(this.files);
259
290
  }
260
291
  _removeFile(file) {
261
- this._files = this._files.filter((f) => !this._checkFileEquality(file, f));
262
- const dt = new DataTransfer();
263
- this._files.forEach((e) => dt.items.add(e));
264
- this._hiddenInput.files = dt.files;
265
- this._liveRegion.innerText = i18nFileSelectorCurrentlySelected(this._files.map((e) => e.name))[this._language.current];
266
- this._fileChangedEvent.emit(this._files);
292
+ this.files = this.files.filter((f) => !this._checkFileEquality(file, f));
293
+ this._updateA11yLiveRegion();
294
+ this.dispatchEvent(new Event("input", { composed: true, bubbles: true }));
295
+ this.dispatchEvent(new Event("change", { bubbles: true }));
296
+ this._fileChangedEvent.emit(this.files);
267
297
  }
268
298
  /** Calculates the correct unit for the file's size. */
269
299
  _formatFileSize(size) {
270
300
  const i = Math.floor(Math.log(size) / Math.log(1024));
271
301
  return `${(size / Math.pow(1024, i)).toFixed(0)} ${this._suffixes[i]}`;
272
302
  }
303
+ _updateA11yLiveRegion() {
304
+ this._liveRegion.innerText = i18nFileSelectorCurrentlySelected(this.files.map((e) => e.name))[this._language.current];
305
+ }
273
306
  _renderDefaultMode() {
274
307
  return html`
275
308
  <sbb-secondary-button-static
276
309
  size=${this.size}
277
310
  icon-name="folder-open-small"
278
- ?disabled=${this.disabled}
311
+ ?disabled=${this.disabled || this.formDisabled}
279
312
  ${ref((el) => {
280
313
  this._loadButton = el;
281
314
  })}
@@ -299,7 +332,7 @@ let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
299
332
  <span class="sbb-file-selector__dropzone-area--button">
300
333
  <sbb-secondary-button-static
301
334
  size=${this.size}
302
- ?disabled=${this.disabled}
335
+ ?disabled=${this.disabled || this.formDisabled}
303
336
  ${ref((el) => {
304
337
  this._loadButton = el;
305
338
  })}
@@ -311,10 +344,10 @@ let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
311
344
  `;
312
345
  }
313
346
  _renderFileList() {
314
- const TAG_NAME = this._files.length > 1 ? { WRAPPER: "ul", ELEMENT: "li" } : { WRAPPER: "div", ELEMENT: "span" };
347
+ const TAG_NAME = this.files.length > 1 ? { WRAPPER: "ul", ELEMENT: "li" } : { WRAPPER: "div", ELEMENT: "span" };
315
348
  return html`
316
349
  <${unsafeStatic(TAG_NAME.WRAPPER)} class="sbb-file-selector__file-list">
317
- ${this._files.map(
350
+ ${this.files.map(
318
351
  (file) => html`
319
352
  <${unsafeStatic(TAG_NAME.ELEMENT)} class="sbb-file-selector__file">
320
353
  <span class="sbb-file-selector__file-details">
@@ -348,7 +381,7 @@ let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
348
381
  <input
349
382
  class="sbb-file-selector__visually-hidden"
350
383
  type="file"
351
- ?disabled=${this.disabled}
384
+ ?disabled=${this.disabled || this.formDisabled}
352
385
  ?multiple=${this.multiple}
353
386
  accept=${this.accept || nothing}
354
387
  aria-label=${ariaLabel || nothing}
@@ -366,7 +399,7 @@ let SbbFileSelectorElement = class extends SbbDisabledMixin(LitElement) {
366
399
  class="sbb-file-selector__visually-hidden"
367
400
  ${ref((p) => this._liveRegion = p)}
368
401
  ></p>
369
- ${this._files && this._files.length > 0 ? this._renderFileList() : nothing}
402
+ ${this.files.length > 0 ? this._renderFileList() : nothing}
370
403
  <div class="sbb-file-selector__error">
371
404
  <slot name="error"></slot>
372
405
  </div>
@@ -400,8 +433,11 @@ __decorateClass([
400
433
  property({ attribute: "accessibility-label" })
401
434
  ], SbbFileSelectorElement.prototype, "accessibilityLabel", 2);
402
435
  __decorateClass([
403
- state()
404
- ], SbbFileSelectorElement.prototype, "_files", 2);
436
+ property({ attribute: false })
437
+ ], SbbFileSelectorElement.prototype, "value", 1);
438
+ __decorateClass([
439
+ property({ attribute: false })
440
+ ], SbbFileSelectorElement.prototype, "files", 1);
405
441
  SbbFileSelectorElement = __decorateClass([
406
442
  customElement("sbb-file-selector"),
407
443
  slotState()
@@ -409,4 +445,4 @@ SbbFileSelectorElement = __decorateClass([
409
445
  export {
410
446
  SbbFileSelectorElement
411
447
  };
412
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"file-selector.js","sources":["../../../src/elements/file-selector/file-selector.ts"],"sourcesContent":["import type { CSSResultGroup, TemplateResult } from 'lit';\nimport { LitElement, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\nimport { html, unsafeStatic } from 'lit/static-html.js';\n\nimport type { SbbSecondaryButtonStaticElement } from '../button.js';\nimport { sbbInputModalityDetector } from '../core/a11y.js';\nimport { SbbLanguageController } from '../core/controllers.js';\nimport { slotState } from '../core/decorators.js';\nimport { EventEmitter } from '../core/eventing.js';\nimport {\n  i18nFileSelectorButtonLabel,\n  i18nFileSelectorCurrentlySelected,\n  i18nFileSelectorDeleteFile,\n  i18nFileSelectorSubtitleLabel,\n} from '../core/i18n.js';\nimport { SbbDisabledMixin } from '../core/mixins.js';\n\nimport style from './file-selector.scss?lit&inline';\n\nimport '../button/secondary-button.js';\nimport '../button/secondary-button-static.js';\nimport '../icon.js';\n\nexport type DOMEvent = globalThis.Event;\n\n/**\n * It allows to select one or more file from storage devices and display them.\n *\n * @slot error - Use this to provide a `sbb-form-error` to show an error message.\n * @event {CustomEvent<File[]>} fileChanged - An event which is emitted each time the file list changes.\n */\n@customElement('sbb-file-selector')\n@slotState()\nexport class SbbFileSelectorElement extends SbbDisabledMixin(LitElement) {\n  public static override styles: CSSResultGroup = style;\n  public static readonly events = {\n    fileChangedEvent: 'fileChanged',\n  } as const;\n\n  /** Whether the component has a dropzone area or not. */\n  @property() public variant: 'default' | 'dropzone' = 'default';\n\n  /** Size variant, either s or m. */\n  @property({ reflect: true }) public size: 's' | 'm' = 'm';\n\n  /** Whether more than one file can be selected. */\n  @property({ type: Boolean }) public multiple: boolean = false;\n\n  /** Whether the newly added files should override the previously added ones. */\n  @property({ attribute: 'multiple-mode' })\n  public multipleMode: 'default' | 'persistent' = 'default';\n\n  /** A comma-separated list of allowed unique file type specifiers. */\n  @property() public accept?: string;\n\n  /** The title displayed in `dropzone` variant. */\n  @property({ attribute: 'title-content' }) public titleContent?: string;\n\n  /** This will be forwarded as aria-label to the native input element. */\n  @property({ attribute: 'accessibility-label' }) public accessibilityLabel: string | undefined;\n\n  /** The list of selected files. */\n  @state() private _files?: File[];\n\n  /** An event which is emitted each time the file list changes. */\n  private _fileChangedEvent: EventEmitter<File[]> = new EventEmitter(\n    this,\n    SbbFileSelectorElement.events.fileChangedEvent,\n  );\n\n  /**\n   * Gets the currently selected files.\n   */\n  public get files(): File[] {\n    return this._files || [];\n  }\n\n  // Safari has a peculiar behavior when dragging files on the inner button in 'dropzone' variant;\n  // this will require a counter to correctly handle the dragEnter/dragLeave.\n  private _counter: number = 0;\n\n  private _loadButton!: SbbSecondaryButtonStaticElement;\n  private _dragTarget?: HTMLElement;\n  private _hiddenInput!: HTMLInputElement;\n  private _suffixes: string[] = ['B', 'kB', 'MB', 'GB', 'TB'];\n  private _liveRegion!: HTMLParagraphElement;\n\n  private _language = new SbbLanguageController(this);\n\n  /**\n   * @deprecated use 'files' property instead\n   */\n  public getFiles(): File[] {\n    return this._files || [];\n  }\n\n  private _blockEvent(event: DragEvent): void {\n    event.stopPropagation();\n    event.preventDefault();\n  }\n\n  private _checkFileEquality(file1: File, file2: File): boolean {\n    return (\n      file1.name === file2.name &&\n      file1.size === file2.size &&\n      file1.lastModified === file2.lastModified\n    );\n  }\n\n  private _onDragEnter(event: DragEvent): void {\n    this._counter++;\n    if (!this.disabled) {\n      this._setDragState(event.target as HTMLElement, true);\n      this._blockEvent(event);\n    }\n  }\n\n  private _onDragLeave(event: DragEvent): void {\n    this._counter--;\n    if (!this.disabled && event.target === this._dragTarget && this._counter === 0) {\n      this._setDragState();\n      this._blockEvent(event);\n    }\n  }\n\n  private _onFileDrop(event: DragEvent): void {\n    this._counter = 0;\n    if (!this.disabled) {\n      this._setDragState();\n      this._blockEvent(event);\n      this._createFileList(event.dataTransfer!.files);\n    }\n  }\n\n  private _onFocus(): void {\n    if (sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n      this._loadButton.toggleAttribute('data-focus-visible', true);\n    }\n  }\n\n  private _onBlur(): void {\n    if (sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n      this._loadButton.removeAttribute('data-focus-visible');\n    }\n  }\n\n  private _setDragState(\n    dragTarget: HTMLElement | undefined = undefined,\n    isDragEnter: boolean = false,\n  ): void {\n    this._dragTarget = dragTarget;\n    this.toggleAttribute('data-active', isDragEnter);\n    this._loadButton.toggleAttribute('data-active', isDragEnter);\n  }\n\n  private _readFiles(event: DOMEvent): void {\n    const fileInput = event.target as HTMLInputElement;\n    if (fileInput.files) {\n      this._createFileList(fileInput.files);\n    }\n  }\n\n  private _createFileList(files: FileList): void {\n    if (\n      !this.multiple ||\n      this.multipleMode !== 'persistent' ||\n      !this._files ||\n      this._files.length === 0\n    ) {\n      this._files = Array.from(files);\n    } else {\n      this._files = Array.from(files)\n        .filter(\n          (newFile: File): boolean =>\n            this._files!.findIndex((oldFile: File) => this._checkFileEquality(newFile, oldFile)) ===\n            -1,\n        )\n        .concat(this._files);\n    }\n    this._liveRegion.innerText = i18nFileSelectorCurrentlySelected(this._files.map((e) => e.name))[\n      this._language.current\n    ];\n    this._fileChangedEvent.emit(this._files);\n  }\n\n  private _removeFile(file: File): void {\n    this._files = this._files!.filter((f: File) => !this._checkFileEquality(file, f));\n    // The item must be removed from the hidden file input too; the FileList API is flawed, so the DataTransfer object is used.\n    const dt: DataTransfer = new DataTransfer();\n    this._files.forEach((e: File) => dt.items.add(e));\n    this._hiddenInput.files = dt.files;\n    this._liveRegion.innerText = i18nFileSelectorCurrentlySelected(this._files.map((e) => e.name))[\n      this._language.current\n    ];\n    this._fileChangedEvent.emit(this._files);\n  }\n\n  /** Calculates the correct unit for the file's size. */\n  private _formatFileSize(size: number): string {\n    const i: number = Math.floor(Math.log(size) / Math.log(1024));\n    return `${(size / Math.pow(1024, i)).toFixed(0)} ${this._suffixes[i]}`;\n  }\n\n  private _renderDefaultMode(): TemplateResult {\n    return html`\n      <sbb-secondary-button-static\n        size=${this.size}\n        icon-name=\"folder-open-small\"\n        ?disabled=${this.disabled}\n        ${ref((el?: Element): void => {\n          this._loadButton = el as SbbSecondaryButtonStaticElement;\n        })}\n      >\n        ${i18nFileSelectorButtonLabel[this._language.current]}\n      </sbb-secondary-button-static>\n    `;\n  }\n\n  private _renderDropzoneArea(): TemplateResult {\n    return html`\n      <span class=\"sbb-file-selector__dropzone-area\">\n        <span class=\"sbb-file-selector__dropzone-area--icon\">\n          <sbb-icon\n            name=${this.size === 'm' ? 'folder-open-medium' : 'folder-open-small'}\n          ></sbb-icon>\n        </span>\n        <span class=\"sbb-file-selector__dropzone-area--title\">${this.titleContent}</span>\n        <span class=\"sbb-file-selector__dropzone-area--subtitle\">\n          ${i18nFileSelectorSubtitleLabel[this._language.current]}\n        </span>\n        <span class=\"sbb-file-selector__dropzone-area--button\">\n          <sbb-secondary-button-static\n            size=${this.size}\n            ?disabled=${this.disabled}\n            ${ref((el?: Element): void => {\n              this._loadButton = el as SbbSecondaryButtonStaticElement;\n            })}\n          >\n            ${i18nFileSelectorButtonLabel[this._language.current]}\n          </sbb-secondary-button-static>\n        </span>\n      </span>\n    `;\n  }\n\n  private _renderFileList(): TemplateResult {\n    const TAG_NAME: Record<string, string> =\n      this._files!.length > 1\n        ? { WRAPPER: 'ul', ELEMENT: 'li' }\n        : { WRAPPER: 'div', ELEMENT: 'span' };\n\n    /* eslint-disable lit/binding-positions */\n    return html`\n      <${unsafeStatic(TAG_NAME.WRAPPER)} class=\"sbb-file-selector__file-list\">\n        ${this._files!.map(\n          (file: File) => html`\n          <${unsafeStatic(TAG_NAME.ELEMENT)} class=\"sbb-file-selector__file\">\n            <span class=\"sbb-file-selector__file-details\">\n              <span class=\"sbb-file-selector__file-name\">${file.name}</span>\n              <span class=\"sbb-file-selector__file-size\">${this._formatFileSize(file.size)}</span>\n            </span>\n            <sbb-secondary-button\n              size=${this.size}\n              icon-name=\"trash-small\"\n              @click=${() => this._removeFile(file)}\n              aria-label=${`${i18nFileSelectorDeleteFile[this._language.current]} - ${file.name}`}\n            ></sbb-secondary-button>\n          </${unsafeStatic(TAG_NAME.ELEMENT)}>`,\n        )}\n      </${unsafeStatic(TAG_NAME.WRAPPER)}>\n    `;\n  }\n\n  protected override render(): TemplateResult {\n    const ariaLabel = this.accessibilityLabel\n      ? `${i18nFileSelectorButtonLabel[this._language.current]} - ${this.accessibilityLabel}`\n      : undefined;\n    return html`\n      <div class=\"sbb-file-selector\">\n        <div\n          class=\"sbb-file-selector__input-container\"\n          @dragenter=${this._onDragEnter}\n          @dragover=${this._blockEvent}\n          @dragleave=${this._onDragLeave}\n          @drop=${this._onFileDrop}\n        >\n          <label>\n            ${this.variant === 'default' ? this._renderDefaultMode() : this._renderDropzoneArea()}\n            <input\n              class=\"sbb-file-selector__visually-hidden\"\n              type=\"file\"\n              ?disabled=${this.disabled}\n              ?multiple=${this.multiple}\n              accept=${this.accept || nothing}\n              aria-label=${ariaLabel || nothing}\n              @change=${this._readFiles}\n              @focus=${this._onFocus}\n              @blur=${this._onBlur}\n              ${ref((el?: Element): void => {\n                this._hiddenInput = el as HTMLInputElement;\n              })}\n            />\n          </label>\n        </div>\n        <p\n          role=\"status\"\n          class=\"sbb-file-selector__visually-hidden\"\n          ${ref((p?: Element) => (this._liveRegion = p as HTMLParagraphElement))}\n        ></p>\n        ${this._files && this._files.length > 0 ? this._renderFileList() : nothing}\n        <div class=\"sbb-file-selector__error\">\n          <slot name=\"error\"></slot>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-file-selector': SbbFileSelectorElement;\n  }\n}\n"],"names":[],"mappings":"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"}
448
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"file-selector.js","sources":["../../../src/elements/file-selector/file-selector.ts"],"sourcesContent":["import { type CSSResultGroup, LitElement, nothing, type TemplateResult } from 'lit';\nimport { customElement, property } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\nimport { html, unsafeStatic } from 'lit/static-html.js';\n\nimport type { SbbSecondaryButtonStaticElement } from '../button.js';\nimport { sbbInputModalityDetector } from '../core/a11y.js';\nimport { SbbLanguageController } from '../core/controllers.js';\nimport { slotState } from '../core/decorators.js';\nimport { EventEmitter, forwardEventToHost } from '../core/eventing.js';\nimport {\n  i18nFileSelectorButtonLabel,\n  i18nFileSelectorCurrentlySelected,\n  i18nFileSelectorDeleteFile,\n  i18nFileSelectorSubtitleLabel,\n} from '../core/i18n.js';\nimport {\n  type FormRestoreReason,\n  type FormRestoreState,\n  SbbDisabledMixin,\n  SbbFormAssociatedMixin,\n} from '../core/mixins.js';\n\nimport style from './file-selector.scss?lit&inline';\n\nimport '../button/secondary-button.js';\nimport '../button/secondary-button-static.js';\nimport '../icon.js';\n\nexport type DOMEvent = globalThis.Event;\n\n/**\n * It allows to select one or more file from storage devices and display them.\n *\n * @slot error - Use this to provide a `sbb-form-error` to show an error message.\n * @event {CustomEvent<File[]>} fileChanged - An event which is emitted each time the file list changes.\n * @event change - An event which is emitted each time the user modifies the value. Unlike the input event, the change event is not necessarily fired for each alteration to an element's value\n * @event input - An event which is emitted each time the value changes as a direct result of a user action.\n */\n@customElement('sbb-file-selector')\n@slotState()\nexport class SbbFileSelectorElement extends SbbDisabledMixin(SbbFormAssociatedMixin(LitElement)) {\n  public static override styles: CSSResultGroup = style;\n  public static readonly events = {\n    fileChangedEvent: 'fileChanged',\n  } as const;\n\n  /** Whether the component has a dropzone area or not. */\n  @property() public variant: 'default' | 'dropzone' = 'default';\n\n  /** Size variant, either s or m. */\n  @property({ reflect: true }) public size: 's' | 'm' = 'm';\n\n  /** Whether more than one file can be selected. */\n  @property({ type: Boolean }) public multiple: boolean = false;\n\n  /** Whether the newly added files should override the previously added ones. */\n  @property({ attribute: 'multiple-mode' })\n  public multipleMode: 'default' | 'persistent' = 'default';\n\n  /** A comma-separated list of allowed unique file type specifiers. */\n  @property() public accept?: string;\n\n  /** The title displayed in `dropzone` variant. */\n  @property({ attribute: 'title-content' }) public titleContent?: string;\n\n  /** This will be forwarded as aria-label to the native input element. */\n  @property({ attribute: 'accessibility-label' }) public accessibilityLabel: string | undefined;\n\n  /** The path of the first selected file. Empty string ('') if no file is selected */\n  @property({ attribute: false })\n  public override set value(value: string | null) {\n    this._hiddenInput.value = value ?? '';\n\n    if (!value) {\n      this.files = [];\n    }\n  }\n  public override get value(): string | null {\n    return this._hiddenInput?.value;\n  }\n\n  /**\n   * The list of selected files.\n   */\n  @property({ attribute: false })\n  public set files(value: File[]) {\n    this._files = value ?? [];\n\n    // update the inner input\n    const dt: DataTransfer = new DataTransfer();\n    this.files.forEach((e: File) => dt.items.add(e));\n    this._hiddenInput.files = dt.files;\n\n    this.updateFormValue();\n  }\n  public get files(): File[] {\n    return this._files;\n  }\n  private _files: File[] = [];\n\n  /** An event which is emitted each time the file list changes. */\n  private _fileChangedEvent: EventEmitter<File[]> = new EventEmitter(\n    this,\n    SbbFileSelectorElement.events.fileChangedEvent,\n  );\n\n  // Safari has a peculiar behavior when dragging files on the inner button in 'dropzone' variant;\n  // this will require a counter to correctly handle the dragEnter/dragLeave.\n  private _counter: number = 0;\n\n  private _loadButton!: SbbSecondaryButtonStaticElement;\n  private _dragTarget?: HTMLElement;\n  private _hiddenInput!: HTMLInputElement;\n  private _suffixes: string[] = ['B', 'kB', 'MB', 'GB', 'TB'];\n  private _liveRegion!: HTMLParagraphElement;\n\n  private _language = new SbbLanguageController(this);\n\n  /**\n   * @deprecated use the 'files' property instead\n   */\n  public getFiles(): File[] {\n    return this.files;\n  }\n\n  public override formResetCallback(): void {\n    this.files = [];\n  }\n\n  public override formStateRestoreCallback(\n    state: FormRestoreState | null,\n    _reason?: FormRestoreReason,\n  ): void {\n    if (!state) {\n      return;\n    }\n    this.files = (state as [string, FormDataEntryValue][]).map(([_, value]) => value as File);\n  }\n\n  protected override updateFormValue(): void {\n    const formValue = new FormData();\n    this.files.forEach((file) => formValue.append(this.name, file));\n    this.internals.setFormValue(formValue);\n  }\n\n  private _blockEvent(event: DragEvent): void {\n    event.stopPropagation();\n    event.preventDefault();\n  }\n\n  private _checkFileEquality(file1: File, file2: File): boolean {\n    return (\n      file1.name === file2.name &&\n      file1.size === file2.size &&\n      file1.lastModified === file2.lastModified\n    );\n  }\n\n  private _onDragEnter(event: DragEvent): void {\n    this._counter++;\n    if (!this.disabled && !this.formDisabled) {\n      this._setDragState(event.target as HTMLElement, true);\n      this._blockEvent(event);\n    }\n  }\n\n  private _onDragLeave(event: DragEvent): void {\n    this._counter--;\n    if (\n      !this.disabled &&\n      !this.formDisabled &&\n      event.target === this._dragTarget &&\n      this._counter === 0\n    ) {\n      this._setDragState();\n      this._blockEvent(event);\n    }\n  }\n\n  private _onFileDrop(event: DragEvent): void {\n    this._counter = 0;\n    if (!this.disabled && !this.formDisabled) {\n      this._setDragState();\n      this._blockEvent(event);\n      this._createFileList(event.dataTransfer!.files);\n    }\n  }\n\n  private _onFocus(): void {\n    if (sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n      this._loadButton.toggleAttribute('data-focus-visible', true);\n    }\n  }\n\n  private _onBlur(): void {\n    if (sbbInputModalityDetector.mostRecentModality === 'keyboard') {\n      this._loadButton.removeAttribute('data-focus-visible');\n    }\n  }\n\n  private _setDragState(\n    dragTarget: HTMLElement | undefined = undefined,\n    isDragEnter: boolean = false,\n  ): void {\n    this._dragTarget = dragTarget;\n    this.toggleAttribute('data-active', isDragEnter);\n    this._loadButton.toggleAttribute('data-active', isDragEnter);\n  }\n\n  private _readFiles(event: DOMEvent): void {\n    const fileInput = event.target as HTMLInputElement;\n    if (fileInput.files) {\n      this._createFileList(fileInput.files);\n    }\n    forwardEventToHost(event, this);\n  }\n\n  private _createFileList(files: FileList): void {\n    if (!this.multiple || this.multipleMode !== 'persistent' || this.files.length === 0) {\n      this.files = Array.from(files);\n    } else {\n      this.files = Array.from(files)\n        .filter(\n          // Remove duplicates\n          (newFile: File): boolean =>\n            this.files!.findIndex((oldFile: File) => this._checkFileEquality(newFile, oldFile)) ===\n            -1,\n        )\n        .concat(this.files);\n    }\n    this._updateA11yLiveRegion();\n    this._fileChangedEvent.emit(this.files);\n  }\n\n  private _removeFile(file: File): void {\n    this.files = this.files.filter((f: File) => !this._checkFileEquality(file, f));\n    this._updateA11yLiveRegion();\n\n    // Dispatch native events as if the reset is done via the file selection window.\n    this.dispatchEvent(new Event('input', { composed: true, bubbles: true }));\n    this.dispatchEvent(new Event('change', { bubbles: true }));\n    this._fileChangedEvent.emit(this.files);\n  }\n\n  /** Calculates the correct unit for the file's size. */\n  private _formatFileSize(size: number): string {\n    const i: number = Math.floor(Math.log(size) / Math.log(1024));\n    return `${(size / Math.pow(1024, i)).toFixed(0)} ${this._suffixes[i]}`;\n  }\n\n  private _updateA11yLiveRegion(): void {\n    this._liveRegion.innerText = i18nFileSelectorCurrentlySelected(this.files.map((e) => e.name))[\n      this._language.current\n    ];\n  }\n\n  private _renderDefaultMode(): TemplateResult {\n    return html`\n      <sbb-secondary-button-static\n        size=${this.size}\n        icon-name=\"folder-open-small\"\n        ?disabled=${this.disabled || this.formDisabled}\n        ${ref((el?: Element): void => {\n          this._loadButton = el as SbbSecondaryButtonStaticElement;\n        })}\n      >\n        ${i18nFileSelectorButtonLabel[this._language.current]}\n      </sbb-secondary-button-static>\n    `;\n  }\n\n  private _renderDropzoneArea(): TemplateResult {\n    return html`\n      <span class=\"sbb-file-selector__dropzone-area\">\n        <span class=\"sbb-file-selector__dropzone-area--icon\">\n          <sbb-icon\n            name=${this.size === 'm' ? 'folder-open-medium' : 'folder-open-small'}\n          ></sbb-icon>\n        </span>\n        <span class=\"sbb-file-selector__dropzone-area--title\">${this.titleContent}</span>\n        <span class=\"sbb-file-selector__dropzone-area--subtitle\">\n          ${i18nFileSelectorSubtitleLabel[this._language.current]}\n        </span>\n        <span class=\"sbb-file-selector__dropzone-area--button\">\n          <sbb-secondary-button-static\n            size=${this.size}\n            ?disabled=${this.disabled || this.formDisabled}\n            ${ref((el?: Element): void => {\n              this._loadButton = el as SbbSecondaryButtonStaticElement;\n            })}\n          >\n            ${i18nFileSelectorButtonLabel[this._language.current]}\n          </sbb-secondary-button-static>\n        </span>\n      </span>\n    `;\n  }\n\n  private _renderFileList(): TemplateResult {\n    const TAG_NAME: Record<string, string> =\n      this.files.length > 1\n        ? { WRAPPER: 'ul', ELEMENT: 'li' }\n        : { WRAPPER: 'div', ELEMENT: 'span' };\n\n    /* eslint-disable lit/binding-positions */\n    return html`\n      <${unsafeStatic(TAG_NAME.WRAPPER)} class=\"sbb-file-selector__file-list\">\n        ${this.files.map(\n          (file: File) => html`\n          <${unsafeStatic(TAG_NAME.ELEMENT)} class=\"sbb-file-selector__file\">\n            <span class=\"sbb-file-selector__file-details\">\n              <span class=\"sbb-file-selector__file-name\">${file.name}</span>\n              <span class=\"sbb-file-selector__file-size\">${this._formatFileSize(file.size)}</span>\n            </span>\n            <sbb-secondary-button\n              size=${this.size}\n              icon-name=\"trash-small\"\n              @click=${() => this._removeFile(file)}\n              aria-label=${`${i18nFileSelectorDeleteFile[this._language.current]} - ${file.name}`}\n            ></sbb-secondary-button>\n          </${unsafeStatic(TAG_NAME.ELEMENT)}>`,\n        )}\n      </${unsafeStatic(TAG_NAME.WRAPPER)}>\n    `;\n  }\n\n  protected override render(): TemplateResult {\n    const ariaLabel = this.accessibilityLabel\n      ? `${i18nFileSelectorButtonLabel[this._language.current]} - ${this.accessibilityLabel}`\n      : undefined;\n    return html`\n      <div class=\"sbb-file-selector\">\n        <div\n          class=\"sbb-file-selector__input-container\"\n          @dragenter=${this._onDragEnter}\n          @dragover=${this._blockEvent}\n          @dragleave=${this._onDragLeave}\n          @drop=${this._onFileDrop}\n        >\n          <label>\n            ${this.variant === 'default' ? this._renderDefaultMode() : this._renderDropzoneArea()}\n            <input\n              class=\"sbb-file-selector__visually-hidden\"\n              type=\"file\"\n              ?disabled=${this.disabled || this.formDisabled}\n              ?multiple=${this.multiple}\n              accept=${this.accept || nothing}\n              aria-label=${ariaLabel || nothing}\n              @change=${this._readFiles}\n              @focus=${this._onFocus}\n              @blur=${this._onBlur}\n              ${ref((el?: Element): void => {\n                this._hiddenInput = el as HTMLInputElement;\n              })}\n            />\n          </label>\n        </div>\n        <p\n          role=\"status\"\n          class=\"sbb-file-selector__visually-hidden\"\n          ${ref((p?: Element) => (this._liveRegion = p as HTMLParagraphElement))}\n        ></p>\n        ${this.files.length > 0 ? this._renderFileList() : nothing}\n        <div class=\"sbb-file-selector__error\">\n          <slot name=\"error\"></slot>\n        </div>\n      </div>\n    `;\n  }\n}\n\ndeclare global {\n  interface HTMLElementTagNameMap {\n    // eslint-disable-next-line @typescript-eslint/naming-convention\n    'sbb-file-selector': SbbFileSelectorElement;\n  }\n}\n"],"names":[],"mappings":"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"}
@@ -81,7 +81,7 @@ sbb-icon {
81
81
  --sbb-button-color-default-background: var(--sbb-color-black-alpha-0);
82
82
  --sbb-button-color-hover-background: var(--sbb-color-charcoal);
83
83
  --sbb-button-color-active-background: var(--sbb-color-iron);
84
- --sbb-button-color-disabled-background: var(--sbb-color-black-alpha-0);
84
+ --sbb-button-color-disabled-background: var(--sbb-color-charcoal);
85
85
  --sbb-button-color-disabled-border: var(--sbb-color-smoke);
86
86
  --sbb-button-color-disabled-text: var(--sbb-color-smoke);
87
87
  }
@@ -26,6 +26,10 @@ const style = css`*,
26
26
  --sbb-header-action-translate-y: 0;
27
27
  --sbb-header-action-gap: var(--sbb-spacing-fixed-2x);
28
28
  --sbb-header-action-icon-dimension: var(--sbb-size-icon-ui-small);
29
+ --sbb-header-action-active-border-width: var(--sbb-border-width-2x);
30
+ --sbb-header-action-active-border-color: var(--sbb-color-black);
31
+ --sbb-header-action-active-border-margin-inline: var(--sbb-header-action-padding-inline);
32
+ --sbb-header-action-active-border-scale: 0;
29
33
  --_sbb-header-first-item-padding-shift: calc(-1 * var(--sbb-header-action-padding-inline));
30
34
  --sbb-header-first-item-icon-shift: 0.125rem;
31
35
  --sbb-header-first-item-margin-inline-start: calc(
@@ -36,6 +40,7 @@ const style = css`*,
36
40
  :host {
37
41
  --sbb-header-action-border-color: CanvasText;
38
42
  --sbb-header-action-color: LinkText;
43
+ --sbb-header-action-active-border-width: 0;
39
44
  }
40
45
  }
41
46
 
@@ -60,13 +65,22 @@ const style = css`*,
60
65
  }
61
66
  }
62
67
 
68
+ :host(.sbb-active) {
69
+ --sbb-header-action-active-border-scale: 1;
70
+ }
71
+ @media (forced-colors: active) {
72
+ :host(.sbb-active) {
73
+ --sbb-header-action-border-color: Highlight;
74
+ }
75
+ }
76
+
63
77
  @media (forced-colors: active) {
64
78
  :host([role=button]) {
65
79
  --sbb-header-action-color: ButtonText;
66
80
  }
67
81
  }
68
82
 
69
- :is(.sbb-header-button, .sbb-header-link) {
83
+ .sbb-action-base {
70
84
  font-family: inherit;
71
85
  font-weight: inherit;
72
86
  line-height: inherit;
@@ -86,7 +100,7 @@ const style = css`*,
86
100
  user-select: none;
87
101
  outline: none;
88
102
  }
89
- :is(.sbb-header-button, .sbb-header-link)::before {
103
+ .sbb-action-base::before {
90
104
  position: absolute;
91
105
  content: "";
92
106
  inset: var(--sbb-header-action-background-inset);
@@ -98,14 +112,23 @@ const style = css`*,
98
112
  border: var(--sbb-border-width-1x) solid var(--sbb-header-action-border-color);
99
113
  }
100
114
  @media (forced-colors: active) {
101
- :is(.sbb-header-button, .sbb-header-link)::before {
115
+ .sbb-action-base::before {
102
116
  border-width: var(--sbb-border-width-2x);
103
117
  }
104
118
  }
105
- :host(:focus-visible:not([data-focus-origin=mouse], [data-focus-origin=touch])) :is(.sbb-header-button, .sbb-header-link)::before, :host(:not([data-focus-origin=mouse], [data-focus-origin=touch])) :is(.sbb-header-button, .sbb-header-link):focus-visible::before {
119
+ :host(:focus-visible:not([data-focus-origin=mouse], [data-focus-origin=touch])) .sbb-action-base::before, :host(:not([data-focus-origin=mouse], [data-focus-origin=touch])) .sbb-action-base:focus-visible::before {
106
120
  outline-offset: var(--sbb-focus-outline-offset);
107
121
  outline: var(--sbb-focus-outline-color) solid var(--sbb-focus-outline-width);
108
122
  }
123
+ .sbb-action-base::after {
124
+ content: "";
125
+ position: absolute;
126
+ border-bottom: var(--sbb-header-action-active-border-width) solid var(--sbb-header-action-active-border-color);
127
+ inset: auto 0 calc(-1 * var(--sbb-header-action-active-border-width));
128
+ margin-inline: var(--sbb-header-action-active-border-margin-inline);
129
+ scale: var(--sbb-header-action-active-border-scale);
130
+ transition: scale var(--sbb-header-action-transition-easing) var(--sbb-header-action-transition-duration);
131
+ }
109
132
 
110
133
  .sbb-header-action__wrapper {
111
134
  display: flex;
@@ -146,6 +169,9 @@ const style = css`*,
146
169
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
147
170
  );
148
171
  --sbb-header-action-padding-inline-zero: 0;
172
+ --sbb-header-action-active-border-margin-inline: calc(
173
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
174
+ );
149
175
  }
150
176
  :host([expand-from=zero]) .sbb-header-action__text {
151
177
  border: 0;
@@ -167,6 +193,9 @@ const style = css`*,
167
193
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
168
194
  );
169
195
  --sbb-header-action-padding-inline-zero: 0;
196
+ --sbb-header-action-active-border-margin-inline: calc(
197
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
198
+ );
170
199
  }
171
200
  :host([expand-from=micro]) .sbb-header-action__text {
172
201
  border: 0;
@@ -188,6 +217,9 @@ const style = css`*,
188
217
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
189
218
  );
190
219
  --sbb-header-action-padding-inline-zero: 0;
220
+ --sbb-header-action-active-border-margin-inline: calc(
221
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
222
+ );
191
223
  }
192
224
  :host([expand-from=small]) .sbb-header-action__text {
193
225
  border: 0;
@@ -209,6 +241,9 @@ const style = css`*,
209
241
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
210
242
  );
211
243
  --sbb-header-action-padding-inline-zero: 0;
244
+ --sbb-header-action-active-border-margin-inline: calc(
245
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
246
+ );
212
247
  }
213
248
  :host([expand-from=medium]) .sbb-header-action__text {
214
249
  border: 0;
@@ -230,6 +265,9 @@ const style = css`*,
230
265
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
231
266
  );
232
267
  --sbb-header-action-padding-inline-zero: 0;
268
+ --sbb-header-action-active-border-margin-inline: calc(
269
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
270
+ );
233
271
  }
234
272
  :host([expand-from=large]) .sbb-header-action__text {
235
273
  border: 0;
@@ -251,6 +289,9 @@ const style = css`*,
251
289
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
252
290
  );
253
291
  --sbb-header-action-padding-inline-zero: 0;
292
+ --sbb-header-action-active-border-margin-inline: calc(
293
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
294
+ );
254
295
  }
255
296
  :host([expand-from=wide]) .sbb-header-action__text {
256
297
  border: 0;
@@ -272,6 +313,9 @@ const style = css`*,
272
313
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
273
314
  );
274
315
  --sbb-header-action-padding-inline-zero: 0;
316
+ --sbb-header-action-active-border-margin-inline: calc(
317
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
318
+ );
275
319
  }
276
320
  :host([expand-from=ultra]) .sbb-header-action__text {
277
321
  border: 0;
@@ -293,6 +337,9 @@ const style = css`*,
293
337
  -0.5 * (var(--sbb-header-action-min-width) - var(--sbb-header-action-icon-dimension))
294
338
  );
295
339
  --sbb-header-action-padding-inline-zero: 0;
340
+ --sbb-header-action-active-border-margin-inline: calc(
341
+ 0.5 * (100% - var(--sbb-size-icon-ui-small))
342
+ );
296
343
  }
297
344
  :host([expand-from=ultra--max-content]) .sbb-header-action__text {
298
345
  border: 0;
@@ -343,4 +390,4 @@ const SbbHeaderActionCommonElementMixin = (superClass) => {
343
390
  export {
344
391
  SbbHeaderActionCommonElementMixin
345
392
  };
346
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbW9uLmpzIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvZWxlbWVudHMvaGVhZGVyL2NvbW1vbi9oZWFkZXItYWN0aW9uLWNvbW1vbi50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IENTU1Jlc3VsdEdyb3VwLCBUZW1wbGF0ZVJlc3VsdCB9IGZyb20gJ2xpdCc7XG5pbXBvcnQgeyBwcm9wZXJ0eSB9IGZyb20gJ2xpdC9kZWNvcmF0b3JzLmpzJztcbmltcG9ydCB7IGh0bWwgfSBmcm9tICdsaXQvc3RhdGljLWh0bWwuanMnO1xuXG5pbXBvcnQgdHlwZSB7IFNiYkFjdGlvbkJhc2VFbGVtZW50IH0gZnJvbSAnLi4vLi4vY29yZS9iYXNlLWVsZW1lbnRzLmpzJztcbmltcG9ydCB0eXBlIHsgU2JiSG9yaXpvbnRhbEZyb20gfSBmcm9tICcuLi8uLi9jb3JlL2ludGVyZmFjZXMuanMnO1xuaW1wb3J0IHR5cGUgeyBBYnN0cmFjdENvbnN0cnVjdG9yIH0gZnJvbSAnLi4vLi4vY29yZS9taXhpbnMuanMnO1xuaW1wb3J0IHsgU2JiSWNvbk5hbWVNaXhpbiwgdHlwZSBTYmJJY29uTmFtZU1peGluVHlwZSB9IGZyb20gJy4uLy4uL2ljb24uanMnO1xuXG5pbXBvcnQgc3R5bGUgZnJvbSAnLi9oZWFkZXItYWN0aW9uLnNjc3M/bGl0JmlubGluZSc7XG5cbmV4cG9ydCBkZWNsYXJlIGNsYXNzIFNiYkhlYWRlckFjdGlvbkNvbW1vbkVsZW1lbnRNaXhpblR5cGVcbiAgaW1wbGVtZW50cyBQYXJ0aWFsPFNiYkljb25OYW1lTWl4aW5UeXBlPlxue1xuICBwdWJsaWMgZXhwYW5kRnJvbTogU2JiSG9yaXpvbnRhbEZyb207XG4gIHB1YmxpYyBpY29uTmFtZT86IHN0cmluZztcbn1cblxuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uYW1pbmctY29udmVudGlvblxuZXhwb3J0IGNvbnN0IFNiYkhlYWRlckFjdGlvbkNvbW1vbkVsZW1lbnRNaXhpbiA9IDxcbiAgVCBleHRlbmRzIEFic3RyYWN0Q29uc3RydWN0b3I8U2JiQWN0aW9uQmFzZUVsZW1lbnQ+LFxuPihcbiAgc3VwZXJDbGFzczogVCxcbik6IEFic3RyYWN0Q29uc3RydWN0b3I8U2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudE1peGluVHlwZT4gJiBUID0+IHtcbiAgYWJzdHJhY3QgY2xhc3MgU2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudFxuICAgIGV4dGVuZHMgU2JiSWNvbk5hbWVNaXhpbihzdXBlckNsYXNzKVxuICAgIGltcGxlbWVudHMgUGFydGlhbDxTYmJIZWFkZXJBY3Rpb25Db21tb25FbGVtZW50TWl4aW5UeXBlPlxuICB7XG4gICAgcHVibGljIHN0YXRpYyBzdHlsZXM6IENTU1Jlc3VsdEdyb3VwID0gc3R5bGU7XG5cbiAgICAvKipcbiAgICAgKiBVc2VkIHRvIHNldCB0aGUgbWluaW11bSBicmVha3BvaW50IGZyb20gd2hpY2ggdGhlIHRleHQgaXMgZGlzcGxheWVkLlxuICAgICAqIEUuZy4gaWYgc2V0IHRvICdsYXJnZScsIHRoZSB0ZXh0IHdpbGwgYmUgdmlzaWJsZSBmb3IgYnJlYWtwb2ludHMgbGFyZ2UsIHdpZGUsIHVsdHJhLFxuICAgICAqIGFuZCBoaWRkZW4gZm9yIGFsbCB0aGUgb3RoZXJzLlxuICAgICAqL1xuICAgIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ2V4cGFuZC1mcm9tJywgcmVmbGVjdDogdHJ1ZSB9KVxuICAgIHB1YmxpYyBleHBhbmRGcm9tOiBTYmJIb3Jpem9udGFsRnJvbSA9ICdtZWRpdW0nO1xuXG4gICAgcHJvdGVjdGVkIG92ZXJyaWRlIHJlbmRlclRlbXBsYXRlKCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICAgIHJldHVybiBodG1sYFxuICAgICAgICA8c3BhbiBjbGFzcz1cInNiYi1oZWFkZXItYWN0aW9uX193cmFwcGVyXCI+XG4gICAgICAgICAgPHNwYW4gY2xhc3M9XCJzYmItaGVhZGVyLWFjdGlvbl9faWNvblwiPiAke3N1cGVyLnJlbmRlckljb25TbG90KCl9IDwvc3Bhbj5cbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInNiYi1oZWFkZXItYWN0aW9uX190ZXh0XCI+XG4gICAgICAgICAgICA8c2xvdD48L3Nsb3Q+XG4gICAgICAgICAgPC9zcGFuPlxuICAgICAgICA8L3NwYW4+XG4gICAgICBgO1xuICAgIH1cbiAgfVxuICByZXR1cm4gU2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudCBhcyB1bmtub3duIGFzIEFic3RyYWN0Q29uc3RydWN0b3I8U2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudE1peGluVHlwZT4gJlxuICAgIFQ7XG59O1xuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQW1CYSxNQUFBLG9DQUFvQyxDQUcvQyxlQUNtRTtBQUNuRSxRQUFlLGdDQUFmLE1BQWUsc0NBQ0wsaUJBQWlCLFVBQVUsRUFFckM7QUFBQSxJQUhBLGNBQUE7QUFBQSxZQUFBLEdBQUEsU0FBQTtBQVlFLFdBQU8sYUFBZ0M7QUFBQSxJQUFBO0FBQUEsSUFFcEIsaUJBQWlDO0FBQzNDLGFBQUE7QUFBQTtBQUFBLG1EQUVzQyxNQUFNLGdCQUFnQjtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSxJQU1yRTtBQUFBLEVBQ0Y7QUFwQkUsZ0NBQWMsU0FBeUI7QUFKekMsTUFBZSwrQkFBZjtBQVlTLGtCQUFBO0FBQUEsSUFETixTQUFTLEVBQUUsV0FBVyxlQUFlLFNBQVMsTUFBTTtBQUFBLEVBQUEsR0FYeEMsNkJBWU4sV0FBQSxZQUFBO0FBYUYsU0FBQTtBQUVUOyJ9
393
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbW9uLmpzIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvZWxlbWVudHMvaGVhZGVyL2NvbW1vbi9oZWFkZXItYWN0aW9uLWNvbW1vbi50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IENTU1Jlc3VsdEdyb3VwLCBUZW1wbGF0ZVJlc3VsdCB9IGZyb20gJ2xpdCc7XG5pbXBvcnQgeyBwcm9wZXJ0eSB9IGZyb20gJ2xpdC9kZWNvcmF0b3JzLmpzJztcbmltcG9ydCB7IGh0bWwgfSBmcm9tICdsaXQvc3RhdGljLWh0bWwuanMnO1xuXG5pbXBvcnQgdHlwZSB7IFNiYkFjdGlvbkJhc2VFbGVtZW50IH0gZnJvbSAnLi4vLi4vY29yZS9iYXNlLWVsZW1lbnRzLmpzJztcbmltcG9ydCB0eXBlIHsgU2JiSG9yaXpvbnRhbEZyb20gfSBmcm9tICcuLi8uLi9jb3JlL2ludGVyZmFjZXMuanMnO1xuaW1wb3J0IHR5cGUgeyBBYnN0cmFjdENvbnN0cnVjdG9yIH0gZnJvbSAnLi4vLi4vY29yZS9taXhpbnMuanMnO1xuaW1wb3J0IHsgU2JiSWNvbk5hbWVNaXhpbiwgdHlwZSBTYmJJY29uTmFtZU1peGluVHlwZSB9IGZyb20gJy4uLy4uL2ljb24uanMnO1xuXG5pbXBvcnQgc3R5bGUgZnJvbSAnLi9oZWFkZXItYWN0aW9uLnNjc3M/bGl0JmlubGluZSc7XG5cbmV4cG9ydCBkZWNsYXJlIGNsYXNzIFNiYkhlYWRlckFjdGlvbkNvbW1vbkVsZW1lbnRNaXhpblR5cGVcbiAgaW1wbGVtZW50cyBQYXJ0aWFsPFNiYkljb25OYW1lTWl4aW5UeXBlPlxue1xuICBwdWJsaWMgZXhwYW5kRnJvbTogU2JiSG9yaXpvbnRhbEZyb207XG4gIHB1YmxpYyBpY29uTmFtZT86IHN0cmluZztcbn1cblxuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uYW1pbmctY29udmVudGlvblxuZXhwb3J0IGNvbnN0IFNiYkhlYWRlckFjdGlvbkNvbW1vbkVsZW1lbnRNaXhpbiA9IDxcbiAgVCBleHRlbmRzIEFic3RyYWN0Q29uc3RydWN0b3I8U2JiQWN0aW9uQmFzZUVsZW1lbnQ+LFxuPihcbiAgc3VwZXJDbGFzczogVCxcbik6IEFic3RyYWN0Q29uc3RydWN0b3I8U2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudE1peGluVHlwZT4gJiBUID0+IHtcbiAgYWJzdHJhY3QgY2xhc3MgU2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudFxuICAgIGV4dGVuZHMgU2JiSWNvbk5hbWVNaXhpbihzdXBlckNsYXNzKVxuICAgIGltcGxlbWVudHMgUGFydGlhbDxTYmJIZWFkZXJBY3Rpb25Db21tb25FbGVtZW50TWl4aW5UeXBlPlxuICB7XG4gICAgcHVibGljIHN0YXRpYyBzdHlsZXM6IENTU1Jlc3VsdEdyb3VwID0gc3R5bGU7XG5cbiAgICAvKipcbiAgICAgKiBVc2VkIHRvIHNldCB0aGUgbWluaW11bSBicmVha3BvaW50IGZyb20gd2hpY2ggdGhlIHRleHQgaXMgZGlzcGxheWVkLlxuICAgICAqIEUuZy4gaWYgc2V0IHRvICdsYXJnZScsIHRoZSB0ZXh0IHdpbGwgYmUgdmlzaWJsZSBmb3IgYnJlYWtwb2ludHMgbGFyZ2UsIHdpZGUsIHVsdHJhLFxuICAgICAqIGFuZCBoaWRkZW4gZm9yIGFsbCB0aGUgb3RoZXJzLlxuICAgICAqL1xuICAgIEBwcm9wZXJ0eSh7IGF0dHJpYnV0ZTogJ2V4cGFuZC1mcm9tJywgcmVmbGVjdDogdHJ1ZSB9KVxuICAgIHB1YmxpYyBleHBhbmRGcm9tOiBTYmJIb3Jpem9udGFsRnJvbSA9ICdtZWRpdW0nO1xuXG4gICAgcHJvdGVjdGVkIG92ZXJyaWRlIHJlbmRlclRlbXBsYXRlKCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICAgIHJldHVybiBodG1sYFxuICAgICAgICA8c3BhbiBjbGFzcz1cInNiYi1oZWFkZXItYWN0aW9uX193cmFwcGVyXCI+XG4gICAgICAgICAgPHNwYW4gY2xhc3M9XCJzYmItaGVhZGVyLWFjdGlvbl9faWNvblwiPiAke3N1cGVyLnJlbmRlckljb25TbG90KCl9IDwvc3Bhbj5cbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInNiYi1oZWFkZXItYWN0aW9uX190ZXh0XCI+XG4gICAgICAgICAgICA8c2xvdD48L3Nsb3Q+XG4gICAgICAgICAgPC9zcGFuPlxuICAgICAgICA8L3NwYW4+XG4gICAgICBgO1xuICAgIH1cbiAgfVxuICByZXR1cm4gU2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudCBhcyB1bmtub3duIGFzIEFic3RyYWN0Q29uc3RydWN0b3I8U2JiSGVhZGVyQWN0aW9uQ29tbW9uRWxlbWVudE1peGluVHlwZT4gJlxuICAgIFQ7XG59O1xuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBbUJhLE1BQUEsb0NBQW9DLENBRy9DLGVBQ21FO0FBQ25FLFFBQWUsZ0NBQWYsTUFBZSxzQ0FDTCxpQkFBaUIsVUFBVSxFQUVyQztBQUFBLElBSEEsY0FBQTtBQUFBLFlBQUEsR0FBQSxTQUFBO0FBWUUsV0FBTyxhQUFnQztBQUFBLElBQUE7QUFBQSxJQUVwQixpQkFBaUM7QUFDM0MsYUFBQTtBQUFBO0FBQUEsbURBRXNDLE1BQU0sZ0JBQWdCO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLElBTXJFO0FBQUEsRUFDRjtBQXBCRSxnQ0FBYyxTQUF5QjtBQUp6QyxNQUFlLCtCQUFmO0FBWVMsa0JBQUE7QUFBQSxJQUROLFNBQVMsRUFBRSxXQUFXLGVBQWUsU0FBUyxNQUFNO0FBQUEsRUFBQSxHQVh4Qyw2QkFZTixXQUFBLFlBQUE7QUFhRixTQUFBO0FBRVQ7In0=