@sbb-esta/lyne-elements 1.11.3 → 1.12.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 (47) hide show
  1. package/_index.scss +1 -1
  2. package/core/i18n/i18n.d.ts +5 -0
  3. package/core/i18n/i18n.d.ts.map +1 -1
  4. package/core/i18n.js +85 -55
  5. package/core/styles/lists.scss +4 -0
  6. package/core/styles/mixins/lists.scss +128 -76
  7. package/custom-elements.json +774 -76
  8. package/development/core/i18n/i18n.d.ts +5 -0
  9. package/development/core/i18n/i18n.d.ts.map +1 -1
  10. package/development/core/i18n.js +36 -1
  11. package/development/paginator/index.d.ts +2 -0
  12. package/development/paginator/index.d.ts.map +1 -0
  13. package/development/paginator/paginator.d.ts +91 -0
  14. package/development/paginator/paginator.d.ts.map +1 -0
  15. package/development/paginator.d.ts +2 -0
  16. package/development/paginator.d.ts.map +1 -0
  17. package/development/paginator.js +526 -0
  18. package/development/radio-button/radio-button-group/radio-button-group.d.ts +1 -1
  19. package/development/radio-button/radio-button-group/radio-button-group.d.ts.map +1 -1
  20. package/development/radio-button/radio-button-group.js +16 -12
  21. package/development/select/select.d.ts +6 -0
  22. package/development/select/select.d.ts.map +1 -1
  23. package/development/select.js +22 -2
  24. package/development/slider/slider.d.ts +35 -16
  25. package/development/slider/slider.d.ts.map +1 -1
  26. package/development/slider.js +102 -81
  27. package/index.d.ts +2 -0
  28. package/index.js +2 -0
  29. package/lists.css +102 -36
  30. package/package.json +6 -1
  31. package/paginator/index.d.ts +2 -0
  32. package/paginator/index.d.ts.map +1 -0
  33. package/paginator/paginator.d.ts +91 -0
  34. package/paginator/paginator.d.ts.map +1 -0
  35. package/paginator.d.ts +2 -0
  36. package/paginator.d.ts.map +1 -0
  37. package/paginator.js +263 -0
  38. package/radio-button/radio-button-group/radio-button-group.d.ts +1 -1
  39. package/radio-button/radio-button-group/radio-button-group.d.ts.map +1 -1
  40. package/radio-button/radio-button-group.js +11 -7
  41. package/select/select.d.ts +6 -0
  42. package/select/select.d.ts.map +1 -1
  43. package/select.js +40 -24
  44. package/slider/slider.d.ts +35 -16
  45. package/slider/slider.d.ts.map +1 -1
  46. package/slider.js +111 -78
  47. package/standard-theme.css +102 -36
@@ -4,9 +4,8 @@ import { ref } from "lit/directives/ref.js";
4
4
  import { styleMap } from "lit/directives/style-map.js";
5
5
  import { SbbConnectedAbortController } from "./core/controllers.js";
6
6
  import { hostAttributes } from "./core/decorators.js";
7
- import { setOrRemoveAttribute } from "./core/dom.js";
8
7
  import { EventEmitter, forwardEventToHost } from "./core/eventing.js";
9
- import { SbbDisabledMixin } from "./core/mixins.js";
8
+ import { SbbDisabledMixin, SbbFormAssociatedMixin } from "./core/mixins.js";
10
9
  import "./icon.js";
11
10
  const style = css`@charset "UTF-8";
12
11
  /** This mixin can be used to avoid spacing problems by inserting an invisible space as pseudo element. */
@@ -48,7 +47,7 @@ const style = css`@charset "UTF-8";
48
47
  }
49
48
  }
50
49
 
51
- :host([disabled]) {
50
+ :host(:disabled) {
52
51
  --sbb-slider-icon-color: var(--sbb-color-graphite);
53
52
  --sbb-slider-knob-border-color: var(--sbb-color-smoke);
54
53
  --sbb-slider-knob-border-size: var(--sbb-border-width-2x);
@@ -60,13 +59,13 @@ const style = css`@charset "UTF-8";
60
59
  --sbb-slider-knob-border-color: var(--sbb-slider-selected-line-disabled-color);
61
60
  }
62
61
 
63
- :host([disabled]),
62
+ :host(:disabled),
64
63
  :host([readonly]) {
65
64
  --sbb-slider-line-color: var(--sbb-slider-line-disabled-color);
66
65
  --sbb-slider-selected-line-color: var(--sbb-slider-selected-line-disabled-color);
67
66
  }
68
67
  @media (forced-colors: active) {
69
- :host([disabled]),
68
+ :host(:disabled),
70
69
  :host([readonly]) {
71
70
  --sbb-slider-icon-color: GrayText;
72
71
  --sbb-slider-selected-line-disabled-color: GrayText;
@@ -146,7 +145,7 @@ const style = css`@charset "UTF-8";
146
145
  outline: var(--sbb-focus-outline-color) solid var(--sbb-focus-outline-width);
147
146
  --sbb-slider-knob-size: var(--sbb-slider-knob-size-active);
148
147
  }
149
- :host(:not(:is([disabled], [readonly]))) .sbb-slider__range-input:active ~ .sbb-slider__knob {
148
+ :host(:not(:is(:disabled, [readonly]))) .sbb-slider__range-input:active ~ .sbb-slider__knob {
150
149
  --sbb-slider-knob-size: var(--sbb-slider-knob-size-active);
151
150
  }`;
152
151
  var __defProp = Object.defineProperty;
@@ -159,13 +158,11 @@ var __decorateClass = (decorators, target, key, kind) => {
159
158
  if (kind && result) __defProp(target, key, result);
160
159
  return result;
161
160
  };
162
- let SbbSliderElement = class extends SbbDisabledMixin(LitElement) {
161
+ let SbbSliderElement = class extends SbbDisabledMixin(SbbFormAssociatedMixin(LitElement)) {
163
162
  constructor() {
164
- super(...arguments);
165
- this.value = "";
166
- this.name = "";
167
- this.min = "0";
168
- this.max = "100";
163
+ super();
164
+ this._min = "0";
165
+ this._max = "100";
169
166
  this.readonly = false;
170
167
  this._valueFraction = 0;
171
168
  this._didChange = new EventEmitter(this, SbbSliderElement.events.didChange, {
@@ -173,96 +170,128 @@ let SbbSliderElement = class extends SbbDisabledMixin(LitElement) {
173
170
  cancelable: true
174
171
  });
175
172
  this._abort = new SbbConnectedAbortController(this);
173
+ this.internals.role = "slider";
174
+ }
175
+ set value(value) {
176
+ if (this._isValidNumber(value)) {
177
+ super.value = this._boundBetweenMinMax(value);
178
+ } else {
179
+ super.value = this._getDefaultValue();
180
+ }
181
+ this.internals.ariaValueNow = this.value;
182
+ this._calculateValueFraction();
183
+ }
184
+ get value() {
185
+ return super.value;
186
+ }
187
+ set valueAsNumber(value) {
188
+ this.value = value == null ? void 0 : value.toString();
189
+ }
190
+ get valueAsNumber() {
191
+ return Number(this.value);
192
+ }
193
+ set min(value) {
194
+ if (!this._isValidNumber(value)) {
195
+ return;
196
+ }
197
+ this._min = value;
198
+ this.value = this._boundBetweenMinMax(this.value);
199
+ }
200
+ get min() {
201
+ return this._min;
202
+ }
203
+ set max(value) {
204
+ if (!this._isValidNumber(value)) {
205
+ return;
206
+ }
207
+ this._max = value;
208
+ this.value = this._boundBetweenMinMax(this.value);
209
+ }
210
+ get max() {
211
+ return this._max;
176
212
  }
177
213
  connectedCallback() {
178
214
  super.connectedCallback();
179
215
  const signal = this._abort.signal;
180
216
  this.addEventListener("keydown", (e) => this._handleKeydown(e), { signal });
181
- this._handleChange();
217
+ if (!this.value) {
218
+ this.value = this._getDefaultValue();
219
+ }
182
220
  }
183
221
  willUpdate(changedProperties) {
184
222
  super.willUpdate(changedProperties);
185
- if (changedProperties.has("value")) {
186
- this._handleChange(Number(this.value));
187
- } else if (changedProperties.has("valueAsNumber")) {
188
- this._handleChange(Number(this.valueAsNumber));
189
- }
190
223
  if (changedProperties.has("min")) {
191
- setOrRemoveAttribute(this, "aria-valuemin", this.min ?? null);
224
+ this.internals.ariaValueMin = this.min;
192
225
  }
193
226
  if (changedProperties.has("max")) {
194
- setOrRemoveAttribute(this, "aria-valuemax", this.max ?? null);
227
+ this.internals.ariaValueMax = this.max;
195
228
  }
196
229
  if (changedProperties.has("readonly")) {
197
- setOrRemoveAttribute(this, "aria-readonly", this.readonly ? "true" : null);
198
- }
199
- if (changedProperties.has("disabled")) {
200
- if (this.disabled) {
201
- this.setAttribute("aria-disabled", "true");
202
- this.removeAttribute("tabindex");
203
- } else {
204
- this.removeAttribute("aria-disabled");
205
- this.setAttribute("tabindex", "0");
206
- }
230
+ this.internals.ariaReadOnly = Boolean(this.readonly).toString();
207
231
  }
208
232
  }
209
- _syncValues(newValue) {
210
- if (newValue == null) {
211
- return;
212
- } else if (newValue && typeof newValue !== "number") {
213
- newValue = +newValue;
214
- }
215
- this.value = newValue.toString();
216
- this.valueAsNumber = newValue;
217
- setOrRemoveAttribute(this, "aria-valuenow", this.value || null);
233
+ /**
234
+ * The reset value is the attribute value (the setup value). If not present, calculates the default.
235
+ * @internal
236
+ */
237
+ formResetCallback() {
238
+ this.value = this.getAttribute("value") ?? this._getDefaultValue();
218
239
  }
219
240
  /**
220
- * Recalculates the `_valueFraction` on change to correctly display the slider knob and lines.
221
- * The first calculation happens in connectedCallback(...), so since `_rangeInput` is not yet available,
222
- * the `min` and `max` values are used; if `value` is not provided, the default value is halfway between min and max
223
- * (see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/range#value).
241
+ * @internal
224
242
  */
225
- _handleChange(value = ((_a) => (_a = this._rangeInput) == null ? void 0 : _a.valueAsNumber)()) {
226
- let min, max;
227
- if (this._rangeInput) {
228
- min = +this._rangeInput.min;
229
- max = +this._rangeInput.max;
230
- } else {
231
- min = +this.min;
232
- max = +this.max;
233
- value = this.value && this.value !== "" ? +this.value : this.valueAsNumber ? this.valueAsNumber : +this.min + (+this.max - +this.min) / 2;
234
- }
243
+ formStateRestoreCallback(state2, _reason) {
244
+ this.value = state2;
245
+ }
246
+ /**
247
+ * If no value is provided, default is the middle point between min and max
248
+ * (see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/range#value)
249
+ */
250
+ _getDefaultValue() {
251
+ return (+this.min + (+this.max - +this.min) / 2).toString();
252
+ }
253
+ _isValidNumber(value) {
254
+ return !!value && !isNaN(Number(value));
255
+ }
256
+ /**
257
+ * Restrains the value between the min and max
258
+ */
259
+ _boundBetweenMinMax(value) {
260
+ return Math.max(+this.min, Math.min(+this.max, +value)).toString();
261
+ }
262
+ _calculateValueFraction() {
263
+ const value = this.valueAsNumber;
264
+ const min = +this.min;
265
+ const max = +this.max;
235
266
  const mathFraction = (value - min) / (max - min);
236
- this._valueFraction = isNaN(mathFraction) || mathFraction < 0 ? 0 : mathFraction > 1 ? 1 : mathFraction;
237
- this._syncValues(value);
267
+ this._valueFraction = isNaN(mathFraction) ? 0 : Math.max(0, Math.min(1, mathFraction));
238
268
  }
239
269
  async _handleKeydown(event) {
240
270
  if (event.key !== "Tab") {
241
271
  event.preventDefault();
242
272
  }
243
- if (this.disabled || this.readonly) {
273
+ if (this.readonly) {
244
274
  return;
245
275
  }
246
276
  if (event.key === "Home") {
247
- this._rangeInput.value = this._rangeInput.min;
277
+ this._rangeInput.value = this.min;
248
278
  } else if (event.key === "End") {
249
- this._rangeInput.value = this._rangeInput.max;
279
+ this._rangeInput.value = this.max;
250
280
  } else if (event.key === "ArrowLeft" || event.key === "ArrowDown") {
251
281
  this._rangeInput.stepDown();
252
282
  } else if (event.key === "ArrowRight" || event.key === "ArrowUp") {
253
283
  this._rangeInput.stepUp();
254
284
  } else if (event.key === "PageDown") {
255
- this._rangeInput.stepDown((+this._rangeInput.max - +this._rangeInput.min) / 10);
285
+ this._rangeInput.stepDown((+this.max - +this.min) / 10);
256
286
  } else if (event.key === "PageUp") {
257
- this._rangeInput.stepUp((+this._rangeInput.max - +this._rangeInput.min) / 10);
287
+ this._rangeInput.stepUp((+this.max - +this.min) / 10);
258
288
  } else {
259
289
  return;
260
290
  }
261
- this._handleChange();
262
- this.dispatchEvent(
291
+ this._rangeInput.dispatchEvent(
263
292
  new InputEvent("input", { bubbles: true, cancelable: true, composed: true })
264
293
  );
265
- this._emitChange(new Event("change", { bubbles: true, cancelable: true, composed: true }));
294
+ this._rangeInput.dispatchEvent(new Event("change", { bubbles: true }));
266
295
  }
267
296
  /** Emits the change event. */
268
297
  _emitChange(event) {
@@ -282,16 +311,14 @@ let SbbSliderElement = class extends SbbDisabledMixin(LitElement) {
282
311
  >
283
312
  <input
284
313
  tabindex="-1"
285
- name=${this.name || nothing}
286
- min=${this.min || nothing}
287
- max=${this.max || nothing}
288
- ?disabled=${this.disabled || this.readonly || nothing}
289
- .valueAsNumber=${this.valueAsNumber || nothing}
314
+ min=${this.min}
315
+ max=${this.max}
316
+ ?disabled=${this.disabled || this.formDisabled || this.readonly}
290
317
  value=${this.value || nothing}
291
318
  class="sbb-slider__range-input"
292
319
  type="range"
293
320
  @change=${(event) => this._emitChange(event)}
294
- @input=${() => this._handleChange()}
321
+ @input=${() => this.value = this._rangeInput.value}
295
322
  ${ref((input) => this._rangeInput = input)}
296
323
  />
297
324
  <div class="sbb-slider__line">
@@ -313,22 +340,16 @@ SbbSliderElement.events = {
313
340
  };
314
341
  __decorateClass([
315
342
  property()
316
- ], SbbSliderElement.prototype, "value", 2);
343
+ ], SbbSliderElement.prototype, "value", 1);
317
344
  __decorateClass([
318
345
  property({ attribute: "value-as-number", type: Number })
319
- ], SbbSliderElement.prototype, "valueAsNumber", 2);
320
- __decorateClass([
321
- property({ reflect: true })
322
- ], SbbSliderElement.prototype, "name", 2);
323
- __decorateClass([
324
- property()
325
- ], SbbSliderElement.prototype, "form", 2);
346
+ ], SbbSliderElement.prototype, "valueAsNumber", 1);
326
347
  __decorateClass([
327
348
  property()
328
- ], SbbSliderElement.prototype, "min", 2);
349
+ ], SbbSliderElement.prototype, "min", 1);
329
350
  __decorateClass([
330
351
  property()
331
- ], SbbSliderElement.prototype, "max", 2);
352
+ ], SbbSliderElement.prototype, "max", 1);
332
353
  __decorateClass([
333
354
  property({ type: Boolean })
334
355
  ], SbbSliderElement.prototype, "readonly", 2);
@@ -344,10 +365,10 @@ __decorateClass([
344
365
  SbbSliderElement = __decorateClass([
345
366
  customElement("sbb-slider"),
346
367
  hostAttributes({
347
- role: "slider"
368
+ tabindex: "0"
348
369
  })
349
370
  ], SbbSliderElement);
350
371
  export {
351
372
  SbbSliderElement
352
373
  };
353
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"slider.js","sources":["../../../src/elements/slider/slider.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, LitElement, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\nimport { styleMap } from 'lit/directives/style-map.js';\n\nimport { SbbConnectedAbortController } from '../core/controllers.js';\nimport { hostAttributes } from '../core/decorators.js';\nimport { setOrRemoveAttribute } from '../core/dom.js';\nimport { EventEmitter, forwardEventToHost } from '../core/eventing.js';\nimport { SbbDisabledMixin } from '../core/mixins.js';\n\nimport style from './slider.scss?lit&inline';\n\nimport '../icon.js';\n\n/**\n * It displays an input knob that can be moved in a range.\n *\n * @slot prefix - Use this slot to render an icon on the left side of the input.\n * @slot suffix - Use this slot to render an icon on the right side of the input.\n * @event {CustomEvent<void>} didChange - Deprecated. used for React. Will probably be removed once React 19 is available.\n */\n@customElement('sbb-slider')\n@hostAttributes({\n  role: 'slider',\n})\nexport class SbbSliderElement extends SbbDisabledMixin(LitElement) {\n  public static override styles: CSSResultGroup = style;\n  public static readonly events = {\n    didChange: 'didChange',\n  } as const;\n\n  /** Value for the inner HTMLInputElement. */\n  @property() public value?: string = '';\n\n  /** Numeric value for the inner HTMLInputElement. */\n  @property({ attribute: 'value-as-number', type: Number }) public valueAsNumber?: number;\n\n  /** Name of the inner HTMLInputElement. */\n  @property({ reflect: true }) public name?: string = '';\n\n  /** The <form> element to associate the inner HTMLInputElement with. */\n  @property() public form?: string;\n\n  /** Minimum acceptable value for the inner HTMLInputElement. */\n  @property() public min?: string = '0';\n\n  /** Maximum acceptable value for the inner HTMLInputElement. */\n  @property() public max?: string = '100';\n\n  /**\n   * Readonly state for the inner HTMLInputElement.\n   * Since the input range does not allow this attribute, it will be merged with the `disabled` one.\n   */\n  @property({ type: Boolean }) public readonly?: boolean = false;\n\n  /** Name of the icon at component's start, which will be forward to the nested `sbb-icon`. */\n  @property({ attribute: 'start-icon' }) public startIcon?: string;\n\n  /** Name of the icon at component's end, which will be forward to the nested `sbb-icon`. */\n  @property({ attribute: 'end-icon' }) public endIcon?: string;\n\n  /**\n   * The ratio between the absolute value and the validity interval.\n   * E.g. given `min=0`, `max=100` and `value=50`, then `_valueFraction=0.5`\n   */\n  @state() private _valueFraction = 0;\n\n  /**\n   * @deprecated only used for React. Will probably be removed once React 19 is available.\n   */\n  private _didChange: EventEmitter = new EventEmitter(this, SbbSliderElement.events.didChange, {\n    bubbles: true,\n    cancelable: true,\n  });\n\n  /** Reference to the inner HTMLInputElement with type='range'. */\n  private _rangeInput!: HTMLInputElement;\n\n  private _abort = new SbbConnectedAbortController(this);\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    const signal = this._abort.signal;\n    this.addEventListener('keydown', (e) => this._handleKeydown(e), { signal });\n    this._handleChange();\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('value')) {\n      this._handleChange(Number(this.value));\n    } else if (changedProperties.has('valueAsNumber')) {\n      this._handleChange(Number(this.valueAsNumber));\n    }\n\n    if (changedProperties.has('min')) {\n      setOrRemoveAttribute(this, 'aria-valuemin', this.min ?? null);\n    }\n    if (changedProperties.has('max')) {\n      setOrRemoveAttribute(this, 'aria-valuemax', this.max ?? null);\n    }\n    if (changedProperties.has('readonly')) {\n      setOrRemoveAttribute(this, 'aria-readonly', this.readonly ? 'true' : null);\n    }\n\n    if (changedProperties.has('disabled')) {\n      if (this.disabled) {\n        this.setAttribute('aria-disabled', 'true');\n        this.removeAttribute('tabindex');\n      } else {\n        this.removeAttribute('aria-disabled');\n        this.setAttribute('tabindex', '0');\n      }\n    }\n  }\n\n  private _syncValues(newValue: string | number): void {\n    if (newValue == null) {\n      return;\n    } else if (newValue && typeof newValue !== 'number') {\n      newValue = +newValue;\n    }\n\n    this.value = newValue.toString();\n    this.valueAsNumber = newValue as number;\n    setOrRemoveAttribute(this, 'aria-valuenow', this.value || null);\n  }\n\n  /**\n   * Recalculates the `_valueFraction` on change to correctly display the slider knob and lines.\n   * The first calculation happens in connectedCallback(...), so since `_rangeInput` is not yet available,\n   * the `min` and `max` values are used; if `value` is not provided, the default value is halfway between min and max\n   * (see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/range#value).\n   */\n  private _handleChange(value: number = this._rangeInput?.valueAsNumber): void {\n    let min: number, max: number;\n    if (this._rangeInput) {\n      min = +this._rangeInput.min;\n      max = +this._rangeInput.max;\n    } else {\n      min = +(this.min as string);\n      max = +(this.max as string);\n      value =\n        this.value && this.value !== ''\n          ? +this.value\n          : this.valueAsNumber\n            ? this.valueAsNumber\n            : +(this.min as string) + (+(this.max as string) - +(this.min as string)) / 2;\n    }\n    const mathFraction: number = (value - min) / (max - min);\n    this._valueFraction =\n      isNaN(mathFraction) || mathFraction < 0 ? 0 : mathFraction > 1 ? 1 : mathFraction;\n    this._syncValues(value);\n  }\n\n  private async _handleKeydown(event: KeyboardEvent): Promise<void> {\n    if (event.key !== 'Tab') {\n      event.preventDefault();\n    }\n\n    if (this.disabled || this.readonly) {\n      return;\n    }\n\n    if (event.key === 'Home') {\n      this._rangeInput.value = this._rangeInput.min;\n    } else if (event.key === 'End') {\n      this._rangeInput.value = this._rangeInput.max;\n    } else if (event.key === 'ArrowLeft' || event.key === 'ArrowDown') {\n      this._rangeInput.stepDown();\n    } else if (event.key === 'ArrowRight' || event.key === 'ArrowUp') {\n      this._rangeInput.stepUp();\n    } else if (event.key === 'PageDown') {\n      this._rangeInput.stepDown((+this._rangeInput.max - +this._rangeInput.min) / 10);\n    } else if (event.key === 'PageUp') {\n      this._rangeInput.stepUp((+this._rangeInput.max - +this._rangeInput.min) / 10);\n    } else {\n      return;\n    }\n\n    this._handleChange();\n    this.dispatchEvent(\n      new InputEvent('input', { bubbles: true, cancelable: true, composed: true }),\n    );\n    this._emitChange(new Event('change', { bubbles: true, cancelable: true, composed: true }));\n  }\n\n  /** Emits the change event. */\n  private _emitChange(event: Event): void {\n    forwardEventToHost(event, this);\n    this._didChange.emit();\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-slider__height-container\">\n        <div class=\"sbb-slider__wrapper\">\n          <slot name=\"prefix\">\n            ${this.startIcon ? html`<sbb-icon name=\"${this.startIcon}\"></sbb-icon>` : nothing}\n          </slot>\n          <div\n            class=\"sbb-slider__container\"\n            style=${styleMap({ '--sbb-slider-value-fraction': this._valueFraction.toString() })}\n          >\n            <input\n              tabindex=\"-1\"\n              name=${this.name || nothing}\n              min=${this.min || nothing}\n              max=${this.max || nothing}\n              ?disabled=${this.disabled || this.readonly || nothing}\n              .valueAsNumber=${this.valueAsNumber || nothing}\n              value=${this.value || nothing}\n              class=\"sbb-slider__range-input\"\n              type=\"range\"\n              @change=${(event: Event) => this._emitChange(event)}\n              @input=${() => this._handleChange()}\n              ${ref((input?: Element) => (this._rangeInput = input as HTMLInputElement))}\n            />\n            <div class=\"sbb-slider__line\">\n              <div class=\"sbb-slider__selected-line\"></div>\n            </div>\n            <div class=\"sbb-slider__knob\"></div>\n          </div>\n          <slot name=\"suffix\">\n            ${this.endIcon ? html`<sbb-icon name=\"${this.endIcon}\"></sbb-icon>` : nothing}\n          </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-slider': SbbSliderElement;\n  }\n}\n"],"names":[],"mappings":"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"}
374
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"slider.js","sources":["../../../src/elements/slider/slider.ts"],"sourcesContent":["import type { CSSResultGroup, PropertyValues, TemplateResult } from 'lit';\nimport { html, LitElement, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport { ref } from 'lit/directives/ref.js';\nimport { styleMap } from 'lit/directives/style-map.js';\n\nimport { SbbConnectedAbortController } from '../core/controllers.js';\nimport { hostAttributes } from '../core/decorators.js';\nimport { EventEmitter, forwardEventToHost } from '../core/eventing.js';\nimport {\n  type FormRestoreReason,\n  type FormRestoreState,\n  SbbDisabledMixin,\n  SbbFormAssociatedMixin,\n} from '../core/mixins.js';\n\nimport style from './slider.scss?lit&inline';\n\nimport '../icon.js';\n\n/**\n * It displays an input knob that can be moved in a range.\n *\n * @slot prefix - Use this slot to render an icon on the left side of the input.\n * @slot suffix - Use this slot to render an icon on the right side of the input.\n * @event {CustomEvent<void>} didChange - Deprecated. used for React. Will probably be removed once React 19 is available.\n */\n@customElement('sbb-slider')\n@hostAttributes({\n  tabindex: '0',\n})\nexport class SbbSliderElement extends SbbDisabledMixin(SbbFormAssociatedMixin(LitElement)) {\n  public static override styles: CSSResultGroup = style;\n  public static readonly events = {\n    didChange: 'didChange',\n  } as const;\n\n  /**\n   * Value of the form element.\n   * If no value is provided, default is the middle point between min and max.\n   */\n  @property()\n  public override set value(value: string | null) {\n    if (this._isValidNumber(value)) {\n      super.value = this._boundBetweenMinMax(value!);\n    } else {\n      super.value = this._getDefaultValue();\n    }\n    this.internals.ariaValueNow = this.value;\n    this._calculateValueFraction();\n  }\n  public override get value(): string {\n    return super.value!;\n  }\n\n  /** Numeric value for the inner HTMLInputElement. */\n  @property({ attribute: 'value-as-number', type: Number })\n  public set valueAsNumber(value: number) {\n    this.value = value?.toString();\n  }\n  public get valueAsNumber(): number | null {\n    return Number(this.value);\n  }\n\n  /** Minimum acceptable value for the inner HTMLInputElement. */\n  @property()\n  public set min(value: string) {\n    if (!this._isValidNumber(value!)) {\n      return;\n    }\n\n    this._min = value;\n    this.value = this._boundBetweenMinMax(this.value);\n  }\n  public get min(): string {\n    return this._min;\n  }\n  private _min: string = '0';\n\n  /** Maximum acceptable value for the inner HTMLInputElement. */\n  @property()\n  public set max(value: string) {\n    if (!this._isValidNumber(value!)) {\n      return;\n    }\n\n    this._max = value;\n    this.value = this._boundBetweenMinMax(this.value);\n  }\n  public get max(): string {\n    return this._max;\n  }\n  private _max: string = '100';\n\n  /**\n   * Readonly state for the inner HTMLInputElement.\n   * Since the input range does not allow this attribute, it will be merged with the `disabled` one.\n   */\n  @property({ type: Boolean }) public readonly?: boolean = false;\n\n  /** Name of the icon at component's start, which will be forward to the nested `sbb-icon`. */\n  @property({ attribute: 'start-icon' }) public startIcon?: string;\n\n  /** Name of the icon at component's end, which will be forward to the nested `sbb-icon`. */\n  @property({ attribute: 'end-icon' }) public endIcon?: string;\n\n  /**\n   * The ratio between the absolute value and the validity interval.\n   * E.g. given `min=0`, `max=100` and `value=50`, then `_valueFraction=0.5`\n   */\n  @state() private _valueFraction = 0;\n\n  /**\n   * @deprecated only used for React. Will probably be removed once React 19 is available.\n   */\n  private _didChange: EventEmitter = new EventEmitter(this, SbbSliderElement.events.didChange, {\n    bubbles: true,\n    cancelable: true,\n  });\n\n  /** Reference to the inner HTMLInputElement with type='range'. */\n  private _rangeInput!: HTMLInputElement;\n\n  private _abort = new SbbConnectedAbortController(this);\n\n  public constructor() {\n    super();\n    /** @internal */\n    this.internals.role = 'slider';\n  }\n\n  public override connectedCallback(): void {\n    super.connectedCallback();\n    const signal = this._abort.signal;\n    this.addEventListener('keydown', (e) => this._handleKeydown(e), { signal });\n\n    if (!this.value) {\n      this.value = this._getDefaultValue();\n    }\n  }\n\n  protected override willUpdate(changedProperties: PropertyValues<this>): void {\n    super.willUpdate(changedProperties);\n\n    if (changedProperties.has('min')) {\n      this.internals.ariaValueMin = this.min;\n    }\n    if (changedProperties.has('max')) {\n      this.internals.ariaValueMax = this.max;\n    }\n    if (changedProperties.has('readonly')) {\n      this.internals.ariaReadOnly = Boolean(this.readonly).toString();\n    }\n  }\n\n  /**\n   * The reset value is the attribute value (the setup value). If not present, calculates the default.\n   * @internal\n   */\n  public formResetCallback(): void {\n    this.value = this.getAttribute('value') ?? this._getDefaultValue();\n  }\n\n  /**\n   * @internal\n   */\n  public formStateRestoreCallback(\n    state: FormRestoreState | null,\n    _reason: FormRestoreReason,\n  ): void {\n    this.value = state as string | null;\n  }\n\n  /**\n   *  If no value is provided, default is the middle point between min and max\n   *  (see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/range#value)\n   */\n  private _getDefaultValue(): string {\n    return (+this.min + (+this.max - +this.min) / 2).toString();\n  }\n\n  private _isValidNumber(value: string | null): boolean {\n    return !!value && !isNaN(Number(value));\n  }\n\n  /**\n   * Restrains the value between the min and max\n   */\n  private _boundBetweenMinMax(value: string): string {\n    return Math.max(+this.min, Math.min(+this.max, +value)).toString();\n  }\n\n  private _calculateValueFraction(): void {\n    const value = this.valueAsNumber!;\n    const min = +this.min;\n    const max = +this.max;\n\n    const mathFraction: number = (value - min) / (max - min);\n    this._valueFraction = isNaN(mathFraction) ? 0 : Math.max(0, Math.min(1, mathFraction));\n  }\n\n  private async _handleKeydown(event: KeyboardEvent): Promise<void> {\n    if (event.key !== 'Tab') {\n      event.preventDefault();\n    }\n\n    if (this.readonly) {\n      return;\n    }\n\n    if (event.key === 'Home') {\n      this._rangeInput.value = this.min;\n    } else if (event.key === 'End') {\n      this._rangeInput.value = this.max;\n    } else if (event.key === 'ArrowLeft' || event.key === 'ArrowDown') {\n      this._rangeInput.stepDown();\n    } else if (event.key === 'ArrowRight' || event.key === 'ArrowUp') {\n      this._rangeInput.stepUp();\n    } else if (event.key === 'PageDown') {\n      this._rangeInput.stepDown((+this.max - +this.min) / 10);\n    } else if (event.key === 'PageUp') {\n      this._rangeInput.stepUp((+this.max - +this.min) / 10);\n    } else {\n      return;\n    }\n\n    // We have to manually fire events because programmatic changes don't trigger them\n    this._rangeInput.dispatchEvent(\n      new InputEvent('input', { bubbles: true, cancelable: true, composed: true }),\n    );\n    this._rangeInput.dispatchEvent(new Event('change', { bubbles: true }));\n  }\n\n  /** Emits the change event. */\n  private _emitChange(event: Event): void {\n    forwardEventToHost(event, this);\n    this._didChange.emit();\n  }\n\n  protected override render(): TemplateResult {\n    return html`\n      <div class=\"sbb-slider__height-container\">\n        <div class=\"sbb-slider__wrapper\">\n          <slot name=\"prefix\">\n            ${this.startIcon ? html`<sbb-icon name=\"${this.startIcon}\"></sbb-icon>` : nothing}\n          </slot>\n          <div\n            class=\"sbb-slider__container\"\n            style=${styleMap({ '--sbb-slider-value-fraction': this._valueFraction.toString() })}\n          >\n            <input\n              tabindex=\"-1\"\n              min=${this.min}\n              max=${this.max}\n              ?disabled=${this.disabled || this.formDisabled || this.readonly}\n              value=${this.value || nothing}\n              class=\"sbb-slider__range-input\"\n              type=\"range\"\n              @change=${(event: Event) => this._emitChange(event)}\n              @input=${() => (this.value = this._rangeInput.value)}\n              ${ref((input?: Element) => (this._rangeInput = input as HTMLInputElement))}\n            />\n            <div class=\"sbb-slider__line\">\n              <div class=\"sbb-slider__selected-line\"></div>\n            </div>\n            <div class=\"sbb-slider__knob\"></div>\n          </div>\n          <slot name=\"suffix\">\n            ${this.endIcon ? html`<sbb-icon name=\"${this.endIcon}\"></sbb-icon>` : nothing}\n          </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-slider': SbbSliderElement;\n  }\n}\n"],"names":["state"],"mappings":"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"}
package/index.d.ts CHANGED
@@ -89,6 +89,7 @@ import { SbbNotificationElement } from "./notification.js";
89
89
  import { SbbOptGroupElement } from "./option/optgroup.js";
90
90
  import { SbbOptionElement } from "./option/option.js";
91
91
  import { SbbOverlayElement } from "./overlay.js";
92
+ import { SbbPaginatorElement } from "./paginator.js";
92
93
  import { SbbPopoverElement } from "./popover/popover.js";
93
94
  import { SbbPopoverTriggerElement } from "./popover/popover-trigger.js";
94
95
  import { SbbRadioButtonElement } from "./radio-button/radio-button.js";
@@ -220,6 +221,7 @@ declare global {
220
221
  var SbbOptGroupElement: SbbOptGroupElement;
221
222
  var SbbOptionElement: SbbOptionElement;
222
223
  var SbbOverlayElement: SbbOverlayElement;
224
+ var SbbPaginatorElement: SbbPaginatorElement;
223
225
  var SbbPopoverElement: SbbPopoverElement;
224
226
  var SbbPopoverTriggerElement: SbbPopoverTriggerElement;
225
227
  var SbbRadioButtonElement: SbbRadioButtonElement;
package/index.js CHANGED
@@ -89,6 +89,7 @@ import { SbbNotificationElement } from "./notification.js";
89
89
  import { SbbOptGroupElement } from "./option/optgroup.js";
90
90
  import { SbbOptionElement } from "./option/option.js";
91
91
  import { SbbOverlayElement } from "./overlay.js";
92
+ import { SbbPaginatorElement } from "./paginator.js";
92
93
  import { SbbPopoverElement } from "./popover/popover.js";
93
94
  import { SbbPopoverTriggerElement } from "./popover/popover-trigger.js";
94
95
  import { SbbRadioButtonElement } from "./radio-button/radio-button.js";
@@ -219,6 +220,7 @@ globalThis.SbbNotificationElement = SbbNotificationElement;
219
220
  globalThis.SbbOptGroupElement = SbbOptGroupElement;
220
221
  globalThis.SbbOptionElement = SbbOptionElement;
221
222
  globalThis.SbbOverlayElement = SbbOverlayElement;
223
+ globalThis.SbbPaginatorElement = SbbPaginatorElement;
222
224
  globalThis.SbbPopoverElement = SbbPopoverElement;
223
225
  globalThis.SbbPopoverTriggerElement = SbbPopoverTriggerElement;
224
226
  globalThis.SbbRadioButtonElement = SbbRadioButtonElement;
package/lists.css CHANGED
@@ -66,24 +66,56 @@
66
66
  }
67
67
 
68
68
  .sbb-step-list {
69
- --sbb-step-list-marker-dimensions: 2.125rem;
70
- --sbb-step-list-padding-block: var(--sbb-spacing-fixed-3x);
69
+ /* stylelint-disable-next-line no-descending-specificity */
70
+ }
71
+ .sbb-step-list {
72
+ --sbb-step-list-dimensions: 2.125rem;
71
73
  --sbb-step-list-padding-inline: var(--sbb-spacing-responsive-xxs);
72
- --sbb-step-list-marker-to-text-gap: var(--sbb-spacing-responsive-xxxs);
73
- --sbb-step-list-border-radius: var(--sbb-border-radius-4x);
74
- --sbb-step-list-text-to-marker-block-offset: calc(
75
- 0.5 *
76
- (
77
- var(--sbb-step-list-marker-dimensions) - var(--sbb-typo-line-height-body-text) *
78
- var(--sbb-text-font-size)
79
- )
80
- );
81
- --sbb-step-list-vertical-gap: var(--sbb-spacing-fixed-1x);
74
+ --sbb-step-list-to-text-gap: var(--sbb-spacing-responsive-xxxs);
75
+ --sbb-step-list-text-to-marker-block-offset: calc(0.5 * (var(--sbb-step-list-dimensions) - var(--sbb-typo-line-height-body-text) * var(--sbb-text-font-size)));
82
76
  list-style: none;
83
77
  margin: 0;
84
78
  padding: 0;
79
+ }
80
+ @media (min-width: calc(52.5rem)) {
81
+ .sbb-step-list:where(.sbb-text-xl) {
82
+ --sbb-step-list-dimensions: 2.5625rem;
83
+ }
84
+ }
85
+ .sbb-step-list > li {
86
+ position: relative;
87
+ padding-inline: calc(var(--sbb-step-list-padding-inline) + var(--sbb-step-list-dimensions) + var(--sbb-step-list-to-text-gap)) var(--sbb-step-list-padding-inline);
88
+ min-height: var(--sbb-step-list-dimensions);
89
+ }
90
+ .sbb-step-list > li::before {
91
+ --sbb-text-font-size: var(--sbb-font-size-text-xxs);
92
+ font-family: var(--sbb-typo-font-family);
93
+ font-weight: normal;
94
+ line-height: var(--sbb-typo-line-height-body-text);
95
+ letter-spacing: var(--sbb-typo-letter-spacing-body-text);
96
+ font-size: var(--sbb-text-font-size);
97
+ font-weight: bold;
98
+ position: absolute;
99
+ height: var(--sbb-step-list-dimensions);
100
+ width: var(--sbb-step-list-dimensions);
101
+ margin-block-start: calc(-1 * var(--sbb-step-list-text-to-marker-block-offset));
102
+ inset-inline-start: var(--sbb-step-list-padding-inline);
103
+ border-radius: 50%;
104
+ }
105
+ .sbb-step-list > li + li {
106
+ margin-block-start: var(--sbb-step-list-vertical-gap);
107
+ }
108
+ .sbb-step-list > li p {
109
+ margin-block: 0;
110
+ }
111
+ .sbb-step-list > li > * + p {
112
+ margin-block-start: var(--sbb-spacing-responsive-xxxs);
113
+ }
114
+ .sbb-step-list {
115
+ --sbb-step-list-padding-block: var(--sbb-spacing-fixed-3x);
116
+ --sbb-step-list-vertical-gap: var(--sbb-spacing-fixed-1x);
117
+ --sbb-step-list-border-radius: var(--sbb-border-radius-4x);
85
118
  counter-reset: steps;
86
- /* stylelint-disable-next-line no-descending-specificity */
87
119
  }
88
120
  .sbb-step-list:where(.sbb-text-s) {
89
121
  --sbb-step-list-padding-block: var(--sbb-spacing-fixed-4x);
@@ -91,18 +123,20 @@
91
123
  .sbb-step-list:where(.sbb-text-m, .sbb-text-l, .sbb-text-xl) {
92
124
  --sbb-step-list-padding-block: var(--sbb-spacing-fixed-5x);
93
125
  }
94
- @media (min-width: calc(52.5rem)) {
95
- .sbb-step-list:where(.sbb-text-xl) {
96
- --sbb-step-list-marker-dimensions: 2.5625rem;
97
- }
98
- }
99
126
  .sbb-step-list > li {
100
- position: relative;
101
127
  counter-increment: steps;
102
128
  background-color: var(--sbb-color-milk);
103
129
  padding-block: calc(var(--sbb-step-list-padding-block) + var(--sbb-step-list-text-to-marker-block-offset)) var(--sbb-step-list-padding-block);
104
- padding-inline: calc(var(--sbb-step-list-padding-inline) + var(--sbb-step-list-marker-dimensions) + var(--sbb-step-list-marker-to-text-gap)) var(--sbb-step-list-padding-inline);
105
- min-height: calc(var(--sbb-step-list-marker-dimensions) + 2 * var(--sbb-step-list-padding-block));
130
+ padding-inline: calc(var(--sbb-step-list-padding-inline) + var(--sbb-step-list-dimensions) + var(--sbb-step-list-to-text-gap)) var(--sbb-step-list-padding-inline);
131
+ min-height: calc(var(--sbb-step-list-dimensions) + 2 * var(--sbb-step-list-padding-block));
132
+ }
133
+ .sbb-step-list > li::before {
134
+ content: counter(steps);
135
+ display: flex;
136
+ align-items: center;
137
+ justify-content: center;
138
+ border: var(--sbb-border-width-1x) solid var(--sbb-color-cement);
139
+ color: var(--sbb-color-charcoal);
106
140
  }
107
141
  .sbb-step-list > li:first-of-type {
108
142
  border-start-start-radius: var(--sbb-step-list-border-radius);
@@ -112,7 +146,30 @@
112
146
  border-end-start-radius: var(--sbb-step-list-border-radius);
113
147
  border-end-end-radius: var(--sbb-step-list-border-radius);
114
148
  }
115
- .sbb-step-list > li::before {
149
+
150
+ .sbb-icon-list {
151
+ /* stylelint-disable-next-line no-descending-specificity */
152
+ }
153
+ .sbb-icon-list {
154
+ --sbb-icon-list-dimensions: 2.125rem;
155
+ --sbb-icon-list-padding-inline: var(--sbb-spacing-responsive-xxs);
156
+ --sbb-icon-list-to-text-gap: var(--sbb-spacing-responsive-xxxs);
157
+ --sbb-icon-list-text-to-marker-block-offset: calc(0.5 * (var(--sbb-icon-list-dimensions) - var(--sbb-typo-line-height-body-text) * var(--sbb-text-font-size)));
158
+ list-style: none;
159
+ margin: 0;
160
+ padding: 0;
161
+ }
162
+ @media (min-width: calc(52.5rem)) {
163
+ .sbb-icon-list:where(.sbb-text-xl) {
164
+ --sbb-icon-list-dimensions: 2.5625rem;
165
+ }
166
+ }
167
+ .sbb-icon-list > li {
168
+ position: relative;
169
+ padding-inline: calc(var(--sbb-icon-list-padding-inline) + var(--sbb-icon-list-dimensions) + var(--sbb-icon-list-to-text-gap)) var(--sbb-icon-list-padding-inline);
170
+ min-height: var(--sbb-icon-list-dimensions);
171
+ }
172
+ .sbb-icon-list > li::before {
116
173
  --sbb-text-font-size: var(--sbb-font-size-text-xxs);
117
174
  font-family: var(--sbb-typo-font-family);
118
175
  font-weight: normal;
@@ -120,25 +177,34 @@
120
177
  letter-spacing: var(--sbb-typo-letter-spacing-body-text);
121
178
  font-size: var(--sbb-text-font-size);
122
179
  font-weight: bold;
123
- content: counter(steps);
124
180
  position: absolute;
125
- display: flex;
126
- align-items: center;
127
- justify-content: center;
128
- height: var(--sbb-step-list-marker-dimensions);
129
- width: var(--sbb-step-list-marker-dimensions);
130
- margin-block-start: calc(-1 * var(--sbb-step-list-text-to-marker-block-offset));
131
- inset-inline-start: var(--sbb-step-list-padding-inline);
132
- border: var(--sbb-border-width-1x) solid var(--sbb-color-cement);
181
+ height: var(--sbb-icon-list-dimensions);
182
+ width: var(--sbb-icon-list-dimensions);
183
+ margin-block-start: calc(-1 * var(--sbb-icon-list-text-to-marker-block-offset));
184
+ inset-inline-start: var(--sbb-icon-list-padding-inline);
133
185
  border-radius: 50%;
134
- color: var(--sbb-color-charcoal);
135
186
  }
136
- .sbb-step-list > li + li {
137
- margin-block-start: var(--sbb-step-list-vertical-gap);
187
+ .sbb-icon-list > li + li {
188
+ margin-block-start: var(--sbb-icon-list-vertical-gap);
138
189
  }
139
- .sbb-step-list > li p {
190
+ .sbb-icon-list > li p {
140
191
  margin-block: 0;
141
192
  }
142
- .sbb-step-list > li > * + p {
193
+ .sbb-icon-list > li > * + p {
143
194
  margin-block-start: var(--sbb-spacing-responsive-xxxs);
195
+ }
196
+ .sbb-icon-list {
197
+ --sbb-icon-list-marker-icon: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" fill="none" viewBox="0 0 24 24"><path fill="%23000" fill-rule="evenodd" d="M3 12a9 9 0 0 1 9-9 9 9 0 0 1 9 9 9 9 0 0 1-9 9 9 9 0 0 1-9-9m9-10C6.477 2 2 6.477 2 12s4.477 10 10 10 10-4.477 10-10S17.523 2 12 2m-1.104 13.305 5-6.5-.792-.61-4.69 6.096-2.102-1.681-.624.78 2.5 2 .398.319z" clip-rule="evenodd"></path></svg>');
198
+ --sbb-icon-list-marker-icon-color: currentcolor;
199
+ --sbb-icon-list-vertical-gap: var(--sbb-spacing-fixed-2x);
200
+ --sbb-icon-list-dimensions: var(--sbb-size-icon-ui-small);
201
+ --sbb-icon-list-to-text-gap: var(--sbb-spacing-fixed-2x);
202
+ }
203
+ .sbb-icon-list > li::before {
204
+ content: "";
205
+ background-color: var(--sbb-icon-list-marker-icon-color);
206
+ mask-image: var(--sbb-icon-list-marker-icon);
207
+ mask-repeat: no-repeat;
208
+ mask-position: center;
209
+ mask-size: 100%;
144
210
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sbb-esta/lyne-elements",
3
- "version": "1.11.3",
3
+ "version": "1.12.0",
4
4
  "description": "Lyne Design System",
5
5
  "keywords": [
6
6
  "design system",
@@ -735,6 +735,11 @@
735
735
  "development": "./development/overlay.js",
736
736
  "default": "./overlay.js"
737
737
  },
738
+ "./paginator.js": {
739
+ "types": "./development/paginator.d.ts",
740
+ "development": "./development/paginator.js",
741
+ "default": "./paginator.js"
742
+ },
738
743
  "./popover.js": {
739
744
  "types": "./development/popover.d.ts",
740
745
  "development": "./development/popover.js",
@@ -0,0 +1,2 @@
1
+ export * from './paginator.js';
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/elements/paginator/index.ts"],"names":[],"mappings":"AAAA,cAAc,gBAAgB,CAAC"}