valtech-components 2.0.536 → 2.0.538

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,6 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, Input, Output, EventEmitter } from '@angular/core';
2
+ import { Component, inject, Input, Output, EventEmitter } from '@angular/core';
3
+ import { I18nService } from '../../../services/i18n';
3
4
  import { DEFAULT_COUNTDOWN_LABELS, } from './types';
4
5
  import * as i0 from "@angular/core";
5
6
  /**
@@ -48,6 +49,7 @@ export class CountdownComponent {
48
49
  constructor() {
49
50
  this.complete = new EventEmitter();
50
51
  this.tick = new EventEmitter();
52
+ this.i18n = inject(I18nService);
51
53
  this.time = {
52
54
  days: 0,
53
55
  hours: 0,
@@ -180,7 +182,7 @@ export class CountdownComponent {
180
182
  return labels[unit] || unit;
181
183
  }
182
184
  getExpiredMessage() {
183
- return this.props.expiredMessage || 'Tiempo agotado';
185
+ return this.props.expiredMessage || this.i18n.t('timeExpired');
184
186
  }
185
187
  getColor() {
186
188
  if (this.props.color) {
@@ -335,4 +337,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
335
337
  }], tick: [{
336
338
  type: Output
337
339
  }] } });
338
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"countdown.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/atoms/countdown/countdown.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, Output, EventEmitter, OnInit, OnDestroy } from '@angular/core';\nimport {\n  CountdownMetadata,\n  CountdownTime,\n  CountdownCompleteEvent,\n  CountdownTickEvent,\n  DEFAULT_COUNTDOWN_LABELS,\n} from './types';\n\n@Component({\n  selector: 'val-countdown',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <div\n      class=\"countdown-container\"\n      [class]=\"props.cssClass\"\n      [class.expired]=\"isExpired\"\n      [class.format-full]=\"props.format === 'full' || !props.format\"\n      [class.format-compact]=\"props.format === 'compact'\"\n      [class.format-minimal]=\"props.format === 'minimal'\"\n      [class.format-digital]=\"props.format === 'digital'\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [style.--countdown-color]=\"getColor()\"\n    >\n      @if (isExpired && props.showExpiredMessage !== false) {\n        <div class=\"expired-message\">\n          {{ getExpiredMessage() }}\n        </div>\n      } @else if (!isExpired) {\n        <div class=\"countdown-units\">\n          @if (shouldShowDays()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.days) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('days', time.days) }}</span>\n              }\n            </div>\n            @if (props.showSeparators !== false && shouldShowHours()) {\n              <span class=\"countdown-separator\">{{ props.separator || ':' }}</span>\n            }\n          }\n\n          @if (shouldShowHours()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.hours) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('hours', time.hours) }}</span>\n              }\n            </div>\n            @if (props.showSeparators !== false && shouldShowMinutes()) {\n              <span class=\"countdown-separator\">{{ props.separator || ':' }}</span>\n            }\n          }\n\n          @if (shouldShowMinutes()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.minutes) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('minutes', time.minutes) }}</span>\n              }\n            </div>\n            @if (props.showSeparators !== false && shouldShowSeconds()) {\n              <span class=\"countdown-separator\">{{ props.separator || ':' }}</span>\n            }\n          }\n\n          @if (shouldShowSeconds()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.seconds) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('seconds', time.seconds) }}</span>\n              }\n            </div>\n          }\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./countdown.component.scss'],\n})\n/**\n * val-countdown\n *\n * A countdown timer component that counts down to a target date.\n *\n * @example Basic usage\n * ```html\n * <val-countdown\n *   [props]=\"{\n *     targetDate: '2024-12-31T23:59:59',\n *     format: 'full'\n *   }\"\n *   (complete)=\"onCountdownComplete()\"\n * ></val-countdown>\n * ```\n *\n * @example Compact format\n * ```html\n * <val-countdown\n *   [props]=\"{\n *     targetDate: raffleEndDate,\n *     format: 'compact',\n *     showDays: true,\n *     showSeconds: false,\n *     color: 'danger',\n *     expiredMessage: '¡Sorteo finalizado!'\n *   }\"\n * ></val-countdown>\n * ```\n *\n * @example Digital clock style\n * ```html\n * <val-countdown\n *   [props]=\"{\n *     targetDate: deadline,\n *     format: 'digital',\n *     showLabels: false,\n *     size: 'large'\n *   }\"\n * ></val-countdown>\n * ```\n */\nexport class CountdownComponent implements OnInit, OnDestroy {\n  @Input() props: CountdownMetadata;\n\n  @Output() complete = new EventEmitter<CountdownCompleteEvent>();\n  @Output() tick = new EventEmitter<CountdownTickEvent>();\n\n  time: CountdownTime = {\n    days: 0,\n    hours: 0,\n    minutes: 0,\n    seconds: 0,\n    totalSeconds: 0,\n    isExpired: false,\n  };\n\n  isExpired = false;\n\n  private intervalId: any = null;\n\n  ngOnInit(): void {\n    if (this.props.autoStart !== false) {\n      this.start();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.stop();\n  }\n\n  /**\n   * Start the countdown timer.\n   */\n  start(): void {\n    this.updateTime();\n    const interval = this.props.updateInterval || 1000;\n    this.intervalId = setInterval(() => this.updateTime(), interval);\n  }\n\n  /**\n   * Stop the countdown timer.\n   */\n  stop(): void {\n    if (this.intervalId) {\n      clearInterval(this.intervalId);\n      this.intervalId = null;\n    }\n  }\n\n  /**\n   * Reset and restart the countdown.\n   */\n  restart(): void {\n    this.stop();\n    this.isExpired = false;\n    this.start();\n  }\n\n  private updateTime(): void {\n    const target = this.getTargetDate();\n    const now = new Date();\n    const diff = target.getTime() - now.getTime();\n\n    if (diff <= 0) {\n      this.time = {\n        days: 0,\n        hours: 0,\n        minutes: 0,\n        seconds: 0,\n        totalSeconds: 0,\n        isExpired: true,\n      };\n      this.isExpired = true;\n      this.stop();\n      this.complete.emit({\n        targetDate: target,\n        completedAt: now,\n      });\n      return;\n    }\n\n    const totalSeconds = Math.floor(diff / 1000);\n    const days = Math.floor(totalSeconds / 86400);\n    const hours = Math.floor((totalSeconds % 86400) / 3600);\n    const minutes = Math.floor((totalSeconds % 3600) / 60);\n    const seconds = totalSeconds % 60;\n\n    this.time = {\n      days,\n      hours,\n      minutes,\n      seconds,\n      totalSeconds,\n      isExpired: false,\n    };\n\n    this.tick.emit({ time: this.time });\n  }\n\n  private getTargetDate(): Date {\n    const target = this.props.targetDate;\n    if (target instanceof Date) {\n      return target;\n    }\n    if (typeof target === 'number') {\n      return new Date(target);\n    }\n    return new Date(target);\n  }\n\n  shouldShowDays(): boolean {\n    if (this.props.showDays === false) return false;\n    if (this.props.showDays === true) return true;\n    // Auto: show if more than 0 days or format is full\n    return this.time.days > 0 || this.props.format === 'full';\n  }\n\n  shouldShowHours(): boolean {\n    if (this.props.showHours === false) return false;\n    return true;\n  }\n\n  shouldShowMinutes(): boolean {\n    if (this.props.showMinutes === false) return false;\n    return true;\n  }\n\n  shouldShowSeconds(): boolean {\n    if (this.props.showSeconds === false) return false;\n    if (this.props.format === 'minimal') return false;\n    return true;\n  }\n\n  formatNumber(value: number): string {\n    if (this.props.padNumbers === false) {\n      return String(value);\n    }\n    return String(value).padStart(2, '0');\n  }\n\n  getLabel(unit: 'days' | 'hours' | 'minutes' | 'seconds', value: number): string {\n    const labels = this.props.labels || DEFAULT_COUNTDOWN_LABELS;\n    const isSingular = value === 1;\n\n    // Get singular or plural form\n    const singularKey = unit.slice(0, -1) as 'day' | 'hour' | 'minute' | 'second';\n\n    if (isSingular && labels[singularKey]) {\n      return labels[singularKey]!;\n    }\n\n    return labels[unit] || unit;\n  }\n\n  getExpiredMessage(): string {\n    return this.props.expiredMessage || 'Tiempo agotado';\n  }\n\n  getColor(): string {\n    if (this.props.color) {\n      return `var(--ion-color-${this.props.color})`;\n    }\n    return 'var(--ion-color-primary)';\n  }\n}\n"]}
340
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"countdown.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/atoms/countdown/countdown.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, inject, Input, Output, EventEmitter, OnInit, OnDestroy } from '@angular/core';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  CountdownMetadata,\n  CountdownTime,\n  CountdownCompleteEvent,\n  CountdownTickEvent,\n  DEFAULT_COUNTDOWN_LABELS,\n} from './types';\n\n@Component({\n  selector: 'val-countdown',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <div\n      class=\"countdown-container\"\n      [class]=\"props.cssClass\"\n      [class.expired]=\"isExpired\"\n      [class.format-full]=\"props.format === 'full' || !props.format\"\n      [class.format-compact]=\"props.format === 'compact'\"\n      [class.format-minimal]=\"props.format === 'minimal'\"\n      [class.format-digital]=\"props.format === 'digital'\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [style.--countdown-color]=\"getColor()\"\n    >\n      @if (isExpired && props.showExpiredMessage !== false) {\n        <div class=\"expired-message\">\n          {{ getExpiredMessage() }}\n        </div>\n      } @else if (!isExpired) {\n        <div class=\"countdown-units\">\n          @if (shouldShowDays()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.days) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('days', time.days) }}</span>\n              }\n            </div>\n            @if (props.showSeparators !== false && shouldShowHours()) {\n              <span class=\"countdown-separator\">{{ props.separator || ':' }}</span>\n            }\n          }\n\n          @if (shouldShowHours()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.hours) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('hours', time.hours) }}</span>\n              }\n            </div>\n            @if (props.showSeparators !== false && shouldShowMinutes()) {\n              <span class=\"countdown-separator\">{{ props.separator || ':' }}</span>\n            }\n          }\n\n          @if (shouldShowMinutes()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.minutes) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('minutes', time.minutes) }}</span>\n              }\n            </div>\n            @if (props.showSeparators !== false && shouldShowSeconds()) {\n              <span class=\"countdown-separator\">{{ props.separator || ':' }}</span>\n            }\n          }\n\n          @if (shouldShowSeconds()) {\n            <div class=\"countdown-unit\">\n              <span class=\"countdown-value\">{{ formatNumber(time.seconds) }}</span>\n              @if (props.showLabels !== false) {\n                <span class=\"countdown-label\">{{ getLabel('seconds', time.seconds) }}</span>\n              }\n            </div>\n          }\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./countdown.component.scss'],\n})\n/**\n * val-countdown\n *\n * A countdown timer component that counts down to a target date.\n *\n * @example Basic usage\n * ```html\n * <val-countdown\n *   [props]=\"{\n *     targetDate: '2024-12-31T23:59:59',\n *     format: 'full'\n *   }\"\n *   (complete)=\"onCountdownComplete()\"\n * ></val-countdown>\n * ```\n *\n * @example Compact format\n * ```html\n * <val-countdown\n *   [props]=\"{\n *     targetDate: raffleEndDate,\n *     format: 'compact',\n *     showDays: true,\n *     showSeconds: false,\n *     color: 'danger',\n *     expiredMessage: '¡Sorteo finalizado!'\n *   }\"\n * ></val-countdown>\n * ```\n *\n * @example Digital clock style\n * ```html\n * <val-countdown\n *   [props]=\"{\n *     targetDate: deadline,\n *     format: 'digital',\n *     showLabels: false,\n *     size: 'large'\n *   }\"\n * ></val-countdown>\n * ```\n */\nexport class CountdownComponent implements OnInit, OnDestroy {\n  @Input() props: CountdownMetadata;\n\n  @Output() complete = new EventEmitter<CountdownCompleteEvent>();\n  @Output() tick = new EventEmitter<CountdownTickEvent>();\n\n  private i18n = inject(I18nService);\n\n  time: CountdownTime = {\n    days: 0,\n    hours: 0,\n    minutes: 0,\n    seconds: 0,\n    totalSeconds: 0,\n    isExpired: false,\n  };\n\n  isExpired = false;\n\n  private intervalId: any = null;\n\n  ngOnInit(): void {\n    if (this.props.autoStart !== false) {\n      this.start();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.stop();\n  }\n\n  /**\n   * Start the countdown timer.\n   */\n  start(): void {\n    this.updateTime();\n    const interval = this.props.updateInterval || 1000;\n    this.intervalId = setInterval(() => this.updateTime(), interval);\n  }\n\n  /**\n   * Stop the countdown timer.\n   */\n  stop(): void {\n    if (this.intervalId) {\n      clearInterval(this.intervalId);\n      this.intervalId = null;\n    }\n  }\n\n  /**\n   * Reset and restart the countdown.\n   */\n  restart(): void {\n    this.stop();\n    this.isExpired = false;\n    this.start();\n  }\n\n  private updateTime(): void {\n    const target = this.getTargetDate();\n    const now = new Date();\n    const diff = target.getTime() - now.getTime();\n\n    if (diff <= 0) {\n      this.time = {\n        days: 0,\n        hours: 0,\n        minutes: 0,\n        seconds: 0,\n        totalSeconds: 0,\n        isExpired: true,\n      };\n      this.isExpired = true;\n      this.stop();\n      this.complete.emit({\n        targetDate: target,\n        completedAt: now,\n      });\n      return;\n    }\n\n    const totalSeconds = Math.floor(diff / 1000);\n    const days = Math.floor(totalSeconds / 86400);\n    const hours = Math.floor((totalSeconds % 86400) / 3600);\n    const minutes = Math.floor((totalSeconds % 3600) / 60);\n    const seconds = totalSeconds % 60;\n\n    this.time = {\n      days,\n      hours,\n      minutes,\n      seconds,\n      totalSeconds,\n      isExpired: false,\n    };\n\n    this.tick.emit({ time: this.time });\n  }\n\n  private getTargetDate(): Date {\n    const target = this.props.targetDate;\n    if (target instanceof Date) {\n      return target;\n    }\n    if (typeof target === 'number') {\n      return new Date(target);\n    }\n    return new Date(target);\n  }\n\n  shouldShowDays(): boolean {\n    if (this.props.showDays === false) return false;\n    if (this.props.showDays === true) return true;\n    // Auto: show if more than 0 days or format is full\n    return this.time.days > 0 || this.props.format === 'full';\n  }\n\n  shouldShowHours(): boolean {\n    if (this.props.showHours === false) return false;\n    return true;\n  }\n\n  shouldShowMinutes(): boolean {\n    if (this.props.showMinutes === false) return false;\n    return true;\n  }\n\n  shouldShowSeconds(): boolean {\n    if (this.props.showSeconds === false) return false;\n    if (this.props.format === 'minimal') return false;\n    return true;\n  }\n\n  formatNumber(value: number): string {\n    if (this.props.padNumbers === false) {\n      return String(value);\n    }\n    return String(value).padStart(2, '0');\n  }\n\n  getLabel(unit: 'days' | 'hours' | 'minutes' | 'seconds', value: number): string {\n    const labels = this.props.labels || DEFAULT_COUNTDOWN_LABELS;\n    const isSingular = value === 1;\n\n    // Get singular or plural form\n    const singularKey = unit.slice(0, -1) as 'day' | 'hour' | 'minute' | 'second';\n\n    if (isSingular && labels[singularKey]) {\n      return labels[singularKey]!;\n    }\n\n    return labels[unit] || unit;\n  }\n\n  getExpiredMessage(): string {\n    return this.props.expiredMessage || this.i18n.t('timeExpired');\n  }\n\n  getColor(): string {\n    if (this.props.color) {\n      return `var(--ion-color-${this.props.color})`;\n    }\n    return 'var(--ion-color-primary)';\n  }\n}\n"]}
@@ -1,5 +1,6 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, Input } from '@angular/core';
2
+ import { Component, inject, Input } from '@angular/core';
3
+ import { I18nService } from '../../../services/i18n';
3
4
  import { TextComponent } from '../../atoms/text/text.component';
4
5
  import * as i0 from "@angular/core";
5
6
  import * as i1 from "@angular/common";
@@ -14,7 +15,9 @@ import * as i1 from "@angular/common";
14
15
  * @input props: InputMetadata - Configuration for the input (form control, errors, etc.)
15
16
  */
16
17
  export class HintComponent {
17
- constructor() { }
18
+ constructor() {
19
+ this.i18n = inject(I18nService);
20
+ }
18
21
  ngOnInit() { }
19
22
  get shouldShowErrors() {
20
23
  if (this.props.control) {
@@ -39,8 +42,8 @@ export class HintComponent {
39
42
  }
40
43
  else if (this.props.fromControl && this.props.toControl) {
41
44
  // NUMBER_FROM_TO field - check both controls
42
- const fromLabel = this.props.fromLabel || 'Inicial';
43
- const toLabel = this.props.toLabel || 'Final';
45
+ const fromLabel = this.props.fromLabel || this.i18n.t('from');
46
+ const toLabel = this.props.toLabel || this.i18n.t('to');
44
47
  if (this.props.fromControl.hasError(e)) {
45
48
  errors.push(`${fromLabel}: ${this.props.errors[e]}`);
46
49
  }
@@ -84,4 +87,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
84
87
  }], ctorParameters: () => [], propDecorators: { props: [{
85
88
  type: Input
86
89
  }] } });
87
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGludC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2hpbnQvaGludC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQ3pELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQzs7O0FBc0JoRTs7Ozs7Ozs7O0dBU0c7QUFDSCxNQUFNLE9BQU8sYUFBYTtJQVN4QixnQkFBZSxDQUFDO0lBRWhCLFFBQVEsS0FBSSxDQUFDO0lBRWIsSUFBSSxnQkFBZ0I7UUFDbEIsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ3ZCLG1DQUFtQztZQUNuQyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNoRyxDQUFDO2FBQU0sSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQzFELDhDQUE4QztZQUM5QyxNQUFNLFdBQVcsR0FDZixJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxPQUFPLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDckcsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsT0FBTyxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQy9HLE9BQU8sV0FBVyxJQUFJLFNBQVMsQ0FBQztRQUNsQyxDQUFDO1FBQ0QsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0lBRUQsSUFBSSxNQUFNO1FBQ1IsTUFBTSxJQUFJLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sSUFBSSxFQUFFLENBQUMsQ0FBQztRQUNsRCxNQUFNLE1BQU0sR0FBRyxFQUFFLENBQUM7UUFFbEIsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQVMsRUFBRSxFQUFFO1lBQ3JCLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7Z0JBQ3pELGVBQWU7Z0JBQ2YsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3BDLENBQUM7aUJBQU0sSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUMxRCw2Q0FBNkM7Z0JBQzdDLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxJQUFJLFNBQVMsQ0FBQztnQkFDcEQsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLElBQUksT0FBTyxDQUFDO2dCQUU5QyxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO29CQUN2QyxNQUFNLENBQUMsSUFBSSxDQUFDLEdBQUcsU0FBUyxLQUFLLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQztnQkFDdkQsQ0FBQztnQkFDRCxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO29CQUNyQyxNQUFNLENBQUMsSUFBSSxDQUFDLEdBQUcsT0FBTyxLQUFLLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQztnQkFDckQsQ0FBQztZQUNILENBQUM7UUFDSCxDQUFDLENBQUMsQ0FBQztRQUVILE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUM7K0dBbERVLGFBQWE7bUdBQWIsYUFBYSxnR0F6QmQ7Ozs7Ozs7Ozs7OztHQVlULGt0RkFiUyxZQUFZLGdRQUFFLGFBQWE7OzRGQTBCMUIsYUFBYTtrQkE3QnpCLFNBQVM7K0JBQ0UsVUFBVSxjQUNSLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxhQUFhLENBQUMsWUFDNUI7Ozs7Ozs7Ozs7OztHQVlUO3dEQW9CUSxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBUZXh0Q29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYXRvbXMvdGV4dC90ZXh0LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBJbnB1dE1ldGFkYXRhIH0gZnJvbSAnLi4vLi4vdHlwZXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtaGludCcsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIFRleHRDb21wb25lbnRdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkaXYgY2xhc3M9XCJoaW50LWNvbnRhaW5lclwiICpuZ0lmPVwic2hvdWxkU2hvd0Vycm9yc1wiPlxuICAgICAgPHZhbC10ZXh0XG4gICAgICAgICpuZ0Zvcj1cImxldCBlIG9mIEVycm9yc1wiXG4gICAgICAgIFtwcm9wc109XCJ7XG4gICAgICAgICAgY29udGVudDogZSxcbiAgICAgICAgICBjb2xvcjogJ2RhbmdlcicsXG4gICAgICAgICAgYm9sZDogZmFsc2UsXG4gICAgICAgICAgc2l6ZTogJ3NtYWxsJyxcbiAgICAgICAgfVwiXG4gICAgICA+PC92YWwtdGV4dD5cbiAgICA8L2Rpdj5cbiAgYCxcbiAgc3R5bGVVcmxzOiBbJy4vaGludC5jb21wb25lbnQuc2NzcyddLFxufSlcbi8qKlxuICogdmFsLWhpbnRcbiAqXG4gKiBEaXNwbGF5cyB2YWxpZGF0aW9uIGVycm9yIG1lc3NhZ2VzIGZvciBhIGZvcm0gaW5wdXQsIHVzaW5nIEFuZ3VsYXIgZm9ybXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIDx2YWwtaGludCBbcHJvcHNdPVwieyBjb250cm9sOiBteUNvbnRyb2wsIGVycm9yczogeyByZXF1aXJlZDogJ1JlcXVpcmVkIGZpZWxkJyB9IH1cIj48L3ZhbC1oaW50PlxuICpcbiAqIEBpbnB1dCBwcm9wczogSW5wdXRNZXRhZGF0YSAtIENvbmZpZ3VyYXRpb24gZm9yIHRoZSBpbnB1dCAoZm9ybSBjb250cm9sLCBlcnJvcnMsIGV0Yy4pXG4gKi9cbmV4cG9ydCBjbGFzcyBIaW50Q29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0IHtcbiAgLyoqXG4gICAqIElucHV0IGNvbmZpZ3VyYXRpb24gb2JqZWN0LlxuICAgKiBAdHlwZSB7SW5wdXRNZXRhZGF0YX1cbiAgICogQHByb3BlcnR5IGNvbnRyb2wgLSBUaGUgQW5ndWxhciBGb3JtQ29udHJvbCBmb3IgdGhlIGlucHV0LlxuICAgKiBAcHJvcGVydHkgZXJyb3JzIC0gVGhlIGVycm9yIG1lc3NhZ2VzIG1hcHBpbmcuXG4gICAqL1xuICBASW5wdXQoKSBwcm9wczogSW5wdXRNZXRhZGF0YTtcblxuICBjb25zdHJ1Y3RvcigpIHt9XG5cbiAgbmdPbkluaXQoKSB7fVxuXG4gIGdldCBzaG91bGRTaG93RXJyb3JzKCk6IGJvb2xlYW4ge1xuICAgIGlmICh0aGlzLnByb3BzLmNvbnRyb2wpIHtcbiAgICAgIC8vIE5vcm1hbCBmaWVsZCB3aXRoIHNpbmdsZSBjb250cm9sXG4gICAgICByZXR1cm4gdGhpcy5wcm9wcy5jb250cm9sLmludmFsaWQgJiYgKHRoaXMucHJvcHMuY29udHJvbC50b3VjaGVkIHx8IHRoaXMucHJvcHMuY29udHJvbC5kaXJ0eSk7XG4gICAgfSBlbHNlIGlmICh0aGlzLnByb3BzLmZyb21Db250cm9sICYmIHRoaXMucHJvcHMudG9Db250cm9sKSB7XG4gICAgICAvLyBOVU1CRVJfRlJPTV9UTyBmaWVsZCB3aXRoIHNlcGFyYXRlIGNvbnRyb2xzXG4gICAgICBjb25zdCBmcm9tSW52YWxpZCA9XG4gICAgICAgIHRoaXMucHJvcHMuZnJvbUNvbnRyb2wuaW52YWxpZCAmJiAodGhpcy5wcm9wcy5mcm9tQ29udHJvbC50b3VjaGVkIHx8IHRoaXMucHJvcHMuZnJvbUNvbnRyb2wuZGlydHkpO1xuICAgICAgY29uc3QgdG9JbnZhbGlkID0gdGhpcy5wcm9wcy50b0NvbnRyb2wuaW52YWxpZCAmJiAodGhpcy5wcm9wcy50b0NvbnRyb2wudG91Y2hlZCB8fCB0aGlzLnByb3BzLnRvQ29udHJvbC5kaXJ0eSk7XG4gICAgICByZXR1cm4gZnJvbUludmFsaWQgfHwgdG9JbnZhbGlkO1xuICAgIH1cbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICBnZXQgRXJyb3JzKCk6IHN0cmluZ1tdIHtcbiAgICBjb25zdCBrZXlzID0gT2JqZWN0LmtleXModGhpcy5wcm9wcy5lcnJvcnMgfHwge30pO1xuICAgIGNvbnN0IGVycm9ycyA9IFtdO1xuXG4gICAga2V5cy5tYXAoKGU6IHN0cmluZykgPT4ge1xuICAgICAgaWYgKHRoaXMucHJvcHMuY29udHJvbCAmJiB0aGlzLnByb3BzLmNvbnRyb2wuaGFzRXJyb3IoZSkpIHtcbiAgICAgICAgLy8gTm9ybWFsIGZpZWxkXG4gICAgICAgIGVycm9ycy5wdXNoKHRoaXMucHJvcHMuZXJyb3JzW2VdKTtcbiAgICAgIH0gZWxzZSBpZiAodGhpcy5wcm9wcy5mcm9tQ29udHJvbCAmJiB0aGlzLnByb3BzLnRvQ29udHJvbCkge1xuICAgICAgICAvLyBOVU1CRVJfRlJPTV9UTyBmaWVsZCAtIGNoZWNrIGJvdGggY29udHJvbHNcbiAgICAgICAgY29uc3QgZnJvbUxhYmVsID0gdGhpcy5wcm9wcy5mcm9tTGFiZWwgfHwgJ0luaWNpYWwnO1xuICAgICAgICBjb25zdCB0b0xhYmVsID0gdGhpcy5wcm9wcy50b0xhYmVsIHx8ICdGaW5hbCc7XG5cbiAgICAgICAgaWYgKHRoaXMucHJvcHMuZnJvbUNvbnRyb2wuaGFzRXJyb3IoZSkpIHtcbiAgICAgICAgICBlcnJvcnMucHVzaChgJHtmcm9tTGFiZWx9OiAke3RoaXMucHJvcHMuZXJyb3JzW2VdfWApO1xuICAgICAgICB9XG4gICAgICAgIGlmICh0aGlzLnByb3BzLnRvQ29udHJvbC5oYXNFcnJvcihlKSkge1xuICAgICAgICAgIGVycm9ycy5wdXNoKGAke3RvTGFiZWx9OiAke3RoaXMucHJvcHMuZXJyb3JzW2VdfWApO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICByZXR1cm4gZXJyb3JzO1xuICB9XG59XG4iXX0=
90
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGludC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2hpbnQvaGludC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBVSxNQUFNLGVBQWUsQ0FBQztBQUNqRSxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDckQsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGlDQUFpQyxDQUFDOzs7QUFzQmhFOzs7Ozs7Ozs7R0FTRztBQUNILE1BQU0sT0FBTyxhQUFhO0lBV3hCO1FBRlEsU0FBSSxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUVwQixDQUFDO0lBRWhCLFFBQVEsS0FBSSxDQUFDO0lBRWIsSUFBSSxnQkFBZ0I7UUFDbEIsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ3ZCLG1DQUFtQztZQUNuQyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNoRyxDQUFDO2FBQU0sSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQzFELDhDQUE4QztZQUM5QyxNQUFNLFdBQVcsR0FDZixJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxPQUFPLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDckcsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsT0FBTyxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQy9HLE9BQU8sV0FBVyxJQUFJLFNBQVMsQ0FBQztRQUNsQyxDQUFDO1FBQ0QsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0lBRUQsSUFBSSxNQUFNO1FBQ1IsTUFBTSxJQUFJLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sSUFBSSxFQUFFLENBQUMsQ0FBQztRQUNsRCxNQUFNLE1BQU0sR0FBRyxFQUFFLENBQUM7UUFFbEIsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQVMsRUFBRSxFQUFFO1lBQ3JCLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7Z0JBQ3pELGVBQWU7Z0JBQ2YsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3BDLENBQUM7aUJBQU0sSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUMxRCw2Q0FBNkM7Z0JBQzdDLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDO2dCQUM5RCxNQUFNLE9BQU8sR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFFeEQsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztvQkFDdkMsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLFNBQVMsS0FBSyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUM7Z0JBQ3ZELENBQUM7Z0JBQ0QsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztvQkFDckMsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLE9BQU8sS0FBSyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUM7Z0JBQ3JELENBQUM7WUFDSCxDQUFDO1FBQ0gsQ0FBQyxDQUFDLENBQUM7UUFFSCxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDOytHQXBEVSxhQUFhO21HQUFiLGFBQWEsZ0dBekJkOzs7Ozs7Ozs7Ozs7R0FZVCxrdEZBYlMsWUFBWSxnUUFBRSxhQUFhOzs0RkEwQjFCLGFBQWE7a0JBN0J6QixTQUFTOytCQUNFLFVBQVUsY0FDUixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsYUFBYSxDQUFDLFlBQzVCOzs7Ozs7Ozs7Ozs7R0FZVDt3REFvQlEsS0FBSztzQkFBYixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgaW5qZWN0LCBJbnB1dCwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBJMThuU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2kxOG4nO1xuaW1wb3J0IHsgVGV4dENvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL3RleHQvdGV4dC5jb21wb25lbnQnO1xuaW1wb3J0IHsgSW5wdXRNZXRhZGF0YSB9IGZyb20gJy4uLy4uL3R5cGVzJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLWhpbnQnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBUZXh0Q29tcG9uZW50XSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8ZGl2IGNsYXNzPVwiaGludC1jb250YWluZXJcIiAqbmdJZj1cInNob3VsZFNob3dFcnJvcnNcIj5cbiAgICAgIDx2YWwtdGV4dFxuICAgICAgICAqbmdGb3I9XCJsZXQgZSBvZiBFcnJvcnNcIlxuICAgICAgICBbcHJvcHNdPVwie1xuICAgICAgICAgIGNvbnRlbnQ6IGUsXG4gICAgICAgICAgY29sb3I6ICdkYW5nZXInLFxuICAgICAgICAgIGJvbGQ6IGZhbHNlLFxuICAgICAgICAgIHNpemU6ICdzbWFsbCcsXG4gICAgICAgIH1cIlxuICAgICAgPjwvdmFsLXRleHQ+XG4gICAgPC9kaXY+XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL2hpbnQuY29tcG9uZW50LnNjc3MnXSxcbn0pXG4vKipcbiAqIHZhbC1oaW50XG4gKlxuICogRGlzcGxheXMgdmFsaWRhdGlvbiBlcnJvciBtZXNzYWdlcyBmb3IgYSBmb3JtIGlucHV0LCB1c2luZyBBbmd1bGFyIGZvcm1zLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLWhpbnQgW3Byb3BzXT1cInsgY29udHJvbDogbXlDb250cm9sLCBlcnJvcnM6IHsgcmVxdWlyZWQ6ICdSZXF1aXJlZCBmaWVsZCcgfSB9XCI+PC92YWwtaGludD5cbiAqXG4gKiBAaW5wdXQgcHJvcHM6IElucHV0TWV0YWRhdGEgLSBDb25maWd1cmF0aW9uIGZvciB0aGUgaW5wdXQgKGZvcm0gY29udHJvbCwgZXJyb3JzLCBldGMuKVxuICovXG5leHBvcnQgY2xhc3MgSGludENvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIC8qKlxuICAgKiBJbnB1dCBjb25maWd1cmF0aW9uIG9iamVjdC5cbiAgICogQHR5cGUge0lucHV0TWV0YWRhdGF9XG4gICAqIEBwcm9wZXJ0eSBjb250cm9sIC0gVGhlIEFuZ3VsYXIgRm9ybUNvbnRyb2wgZm9yIHRoZSBpbnB1dC5cbiAgICogQHByb3BlcnR5IGVycm9ycyAtIFRoZSBlcnJvciBtZXNzYWdlcyBtYXBwaW5nLlxuICAgKi9cbiAgQElucHV0KCkgcHJvcHM6IElucHV0TWV0YWRhdGE7XG5cbiAgcHJpdmF0ZSBpMThuID0gaW5qZWN0KEkxOG5TZXJ2aWNlKTtcblxuICBjb25zdHJ1Y3RvcigpIHt9XG5cbiAgbmdPbkluaXQoKSB7fVxuXG4gIGdldCBzaG91bGRTaG93RXJyb3JzKCk6IGJvb2xlYW4ge1xuICAgIGlmICh0aGlzLnByb3BzLmNvbnRyb2wpIHtcbiAgICAgIC8vIE5vcm1hbCBmaWVsZCB3aXRoIHNpbmdsZSBjb250cm9sXG4gICAgICByZXR1cm4gdGhpcy5wcm9wcy5jb250cm9sLmludmFsaWQgJiYgKHRoaXMucHJvcHMuY29udHJvbC50b3VjaGVkIHx8IHRoaXMucHJvcHMuY29udHJvbC5kaXJ0eSk7XG4gICAgfSBlbHNlIGlmICh0aGlzLnByb3BzLmZyb21Db250cm9sICYmIHRoaXMucHJvcHMudG9Db250cm9sKSB7XG4gICAgICAvLyBOVU1CRVJfRlJPTV9UTyBmaWVsZCB3aXRoIHNlcGFyYXRlIGNvbnRyb2xzXG4gICAgICBjb25zdCBmcm9tSW52YWxpZCA9XG4gICAgICAgIHRoaXMucHJvcHMuZnJvbUNvbnRyb2wuaW52YWxpZCAmJiAodGhpcy5wcm9wcy5mcm9tQ29udHJvbC50b3VjaGVkIHx8IHRoaXMucHJvcHMuZnJvbUNvbnRyb2wuZGlydHkpO1xuICAgICAgY29uc3QgdG9JbnZhbGlkID0gdGhpcy5wcm9wcy50b0NvbnRyb2wuaW52YWxpZCAmJiAodGhpcy5wcm9wcy50b0NvbnRyb2wudG91Y2hlZCB8fCB0aGlzLnByb3BzLnRvQ29udHJvbC5kaXJ0eSk7XG4gICAgICByZXR1cm4gZnJvbUludmFsaWQgfHwgdG9JbnZhbGlkO1xuICAgIH1cbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICBnZXQgRXJyb3JzKCk6IHN0cmluZ1tdIHtcbiAgICBjb25zdCBrZXlzID0gT2JqZWN0LmtleXModGhpcy5wcm9wcy5lcnJvcnMgfHwge30pO1xuICAgIGNvbnN0IGVycm9ycyA9IFtdO1xuXG4gICAga2V5cy5tYXAoKGU6IHN0cmluZykgPT4ge1xuICAgICAgaWYgKHRoaXMucHJvcHMuY29udHJvbCAmJiB0aGlzLnByb3BzLmNvbnRyb2wuaGFzRXJyb3IoZSkpIHtcbiAgICAgICAgLy8gTm9ybWFsIGZpZWxkXG4gICAgICAgIGVycm9ycy5wdXNoKHRoaXMucHJvcHMuZXJyb3JzW2VdKTtcbiAgICAgIH0gZWxzZSBpZiAodGhpcy5wcm9wcy5mcm9tQ29udHJvbCAmJiB0aGlzLnByb3BzLnRvQ29udHJvbCkge1xuICAgICAgICAvLyBOVU1CRVJfRlJPTV9UTyBmaWVsZCAtIGNoZWNrIGJvdGggY29udHJvbHNcbiAgICAgICAgY29uc3QgZnJvbUxhYmVsID0gdGhpcy5wcm9wcy5mcm9tTGFiZWwgfHwgdGhpcy5pMThuLnQoJ2Zyb20nKTtcbiAgICAgICAgY29uc3QgdG9MYWJlbCA9IHRoaXMucHJvcHMudG9MYWJlbCB8fCB0aGlzLmkxOG4udCgndG8nKTtcblxuICAgICAgICBpZiAodGhpcy5wcm9wcy5mcm9tQ29udHJvbC5oYXNFcnJvcihlKSkge1xuICAgICAgICAgIGVycm9ycy5wdXNoKGAke2Zyb21MYWJlbH06ICR7dGhpcy5wcm9wcy5lcnJvcnNbZV19YCk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHRoaXMucHJvcHMudG9Db250cm9sLmhhc0Vycm9yKGUpKSB7XG4gICAgICAgICAgZXJyb3JzLnB1c2goYCR7dG9MYWJlbH06ICR7dGhpcy5wcm9wcy5lcnJvcnNbZV19YCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9KTtcblxuICAgIHJldHVybiBlcnJvcnM7XG4gIH1cbn1cbiJdfQ==
@@ -1,6 +1,7 @@
1
- import { Component, Input } from '@angular/core';
1
+ import { Component, inject, Input } from '@angular/core';
2
2
  import { ReactiveFormsModule } from '@angular/forms';
3
3
  import { IonInput, IonLabel } from '@ionic/angular/standalone';
4
+ import { I18nService } from '../../../services/i18n';
4
5
  import * as i0 from "@angular/core";
5
6
  import * as i1 from "@angular/forms";
6
7
  /**
@@ -14,7 +15,17 @@ import * as i1 from "@angular/forms";
14
15
  * @input props: NumberFromToMetadata - Configuration for the number range input.
15
16
  */
16
17
  export class NumberFromToComponent {
17
- constructor() { }
18
+ /** Get from label with i18n fallback */
19
+ getFromLabel() {
20
+ return this.props.fromLabel || this.i18n.t('from');
21
+ }
22
+ /** Get to label with i18n fallback */
23
+ getToLabel() {
24
+ return this.props.toLabel || this.i18n.t('to');
25
+ }
26
+ constructor() {
27
+ this.i18n = inject(I18nService);
28
+ }
18
29
  ngOnInit() {
19
30
  // Apply default values if configured
20
31
  if (this.props?.withDefault || this.props?.value) {
@@ -75,7 +86,7 @@ export class NumberFromToComponent {
75
86
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: NumberFromToComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
76
87
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: NumberFromToComponent, isStandalone: true, selector: "val-number-from-to", inputs: { props: "props" }, ngImport: i0, template: `
77
88
  <div class="number-from-to-container">
78
- <ion-label position="stacked">{{ props.fromLabel || 'Inicial' }}</ion-label>
89
+ <ion-label position="stacked">{{ getFromLabel() }}</ion-label>
79
90
  <ion-input
80
91
  [formControl]="fromControl"
81
92
  type="number"
@@ -86,7 +97,7 @@ export class NumberFromToComponent {
86
97
  >
87
98
  </ion-input>
88
99
 
89
- <ion-label position="stacked">{{ props.toLabel || 'Final' }}</ion-label>
100
+ <ion-label position="stacked">{{ getToLabel() }}</ion-label>
90
101
  <ion-input
91
102
  [formControl]="toControl"
92
103
  type="number"
@@ -103,7 +114,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
103
114
  type: Component,
104
115
  args: [{ selector: 'val-number-from-to', standalone: true, imports: [ReactiveFormsModule, IonInput, IonLabel], template: `
105
116
  <div class="number-from-to-container">
106
- <ion-label position="stacked">{{ props.fromLabel || 'Inicial' }}</ion-label>
117
+ <ion-label position="stacked">{{ getFromLabel() }}</ion-label>
107
118
  <ion-input
108
119
  [formControl]="fromControl"
109
120
  type="number"
@@ -114,7 +125,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
114
125
  >
115
126
  </ion-input>
116
127
 
117
- <ion-label position="stacked">{{ props.toLabel || 'Final' }}</ion-label>
128
+ <ion-label position="stacked">{{ getToLabel() }}</ion-label>
118
129
  <ion-input
119
130
  [formControl]="toControl"
120
131
  type="number"
@@ -129,4 +140,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
129
140
  }], ctorParameters: () => [], propDecorators: { props: [{
130
141
  type: Input
131
142
  }] } });
132
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"number-from-to.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/number-from-to/number-from-to.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\nimport { FormControl, ReactiveFormsModule } from '@angular/forms';\nimport { IonInput, IonLabel } from '@ionic/angular/standalone';\nimport { NumberFromToMetadata } from './types';\n\n@Component({\n  selector: 'val-number-from-to',\n  standalone: true,\n  imports: [ReactiveFormsModule, IonInput, IonLabel],\n  template: `\n    <div class=\"number-from-to-container\">\n      <ion-label position=\"stacked\">{{ props.fromLabel || 'Inicial' }}</ion-label>\n      <ion-input\n        [formControl]=\"fromControl\"\n        type=\"number\"\n        [placeholder]=\"props.fromPlaceholder || props.placeholder\"\n        [min]=\"props.min\"\n        [max]=\"props.max\"\n        [step]=\"props.step\"\n      >\n      </ion-input>\n\n      <ion-label position=\"stacked\">{{ props.toLabel || 'Final' }}</ion-label>\n      <ion-input\n        [formControl]=\"toControl\"\n        type=\"number\"\n        [placeholder]=\"props.toPlaceholder || props.placeholder\"\n        [min]=\"props.min\"\n        [max]=\"props.max\"\n        [step]=\"props.step\"\n      >\n      </ion-input>\n    </div>\n  `,\n  styleUrls: ['./number-from-to.component.scss'],\n})\n/**\n * val-number-from-to\n *\n * A dual number input for specifying a range (from/to values), integrated with Angular forms.\n *\n * @example\n * <val-number-from-to [props]=\"{ fromControl: fromCtrl, toControl: toCtrl, min: 0, max: 100 }\"></val-number-from-to>\n *\n * @input props: NumberFromToMetadata - Configuration for the number range input.\n */\nexport class NumberFromToComponent implements OnInit {\n  /**\n   * Input configuration object.\n   * @type {NumberFromToMetadata}\n   * @property fromControl - The Angular FormControl for the \"from\" value.\n   * @property toControl - The Angular FormControl for the \"to\" value.\n   */\n  @Input() props: NumberFromToMetadata;\n\n  constructor() {}\n\n  ngOnInit() {\n    // Apply default values if configured\n    if (this.props?.withDefault || this.props?.value) {\n      this.applyDefaultValues();\n    }\n  }\n\n  private applyDefaultValues() {\n    const defaultValue = this.resolveDefaultValue();\n    if (defaultValue !== null) {\n      if (typeof defaultValue === 'object' && defaultValue.from !== undefined) {\n        this.fromControl?.setValue(defaultValue.from);\n        this.toControl?.setValue(defaultValue.to);\n      } else {\n        // Si es un valor simple, aplicarlo a ambos controles\n        this.fromControl?.setValue(defaultValue);\n        this.toControl?.setValue(defaultValue);\n      }\n      this.fromControl?.markAsPristine();\n      this.toControl?.markAsPristine();\n      this.fromControl?.updateValueAndValidity();\n      this.toControl?.updateValueAndValidity();\n    }\n  }\n\n  private resolveDefaultValue(): any {\n    // Explicit value takes precedence\n    if (this.props.value !== undefined && this.props.value !== null && this.props.value !== '') {\n      return this.parseValue(this.props.value);\n    }\n\n    // No default configuration\n    if (!this.props.withDefault) {\n      return null;\n    }\n\n    // Custom default value\n    if (typeof this.props.withDefault === 'string') {\n      return this.parseValue(this.props.withDefault);\n    }\n\n    // Auto default value (withDefault === true)\n    return { from: 0, to: 0 };\n  }\n\n  private parseValue(value: string): any {\n    try {\n      // Try to parse as JSON first (for object values)\n      return JSON.parse(value);\n    } catch {\n      // If not JSON, treat as number\n      const numValue = parseFloat(value) || 0;\n      return { from: numValue, to: numValue };\n    }\n  }\n\n  get fromControl(): FormControl {\n    return this.props.fromControl;\n  }\n\n  get toControl(): FormControl {\n    return this.props.toControl;\n  }\n}\n"]}
143
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"number-from-to.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/number-from-to/number-from-to.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnInit } from '@angular/core';\nimport { FormControl, ReactiveFormsModule } from '@angular/forms';\nimport { IonInput, IonLabel } from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport { NumberFromToMetadata } from './types';\n\n@Component({\n  selector: 'val-number-from-to',\n  standalone: true,\n  imports: [ReactiveFormsModule, IonInput, IonLabel],\n  template: `\n    <div class=\"number-from-to-container\">\n      <ion-label position=\"stacked\">{{ getFromLabel() }}</ion-label>\n      <ion-input\n        [formControl]=\"fromControl\"\n        type=\"number\"\n        [placeholder]=\"props.fromPlaceholder || props.placeholder\"\n        [min]=\"props.min\"\n        [max]=\"props.max\"\n        [step]=\"props.step\"\n      >\n      </ion-input>\n\n      <ion-label position=\"stacked\">{{ getToLabel() }}</ion-label>\n      <ion-input\n        [formControl]=\"toControl\"\n        type=\"number\"\n        [placeholder]=\"props.toPlaceholder || props.placeholder\"\n        [min]=\"props.min\"\n        [max]=\"props.max\"\n        [step]=\"props.step\"\n      >\n      </ion-input>\n    </div>\n  `,\n  styleUrls: ['./number-from-to.component.scss'],\n})\n/**\n * val-number-from-to\n *\n * A dual number input for specifying a range (from/to values), integrated with Angular forms.\n *\n * @example\n * <val-number-from-to [props]=\"{ fromControl: fromCtrl, toControl: toCtrl, min: 0, max: 100 }\"></val-number-from-to>\n *\n * @input props: NumberFromToMetadata - Configuration for the number range input.\n */\nexport class NumberFromToComponent implements OnInit {\n  /**\n   * Input configuration object.\n   * @type {NumberFromToMetadata}\n   * @property fromControl - The Angular FormControl for the \"from\" value.\n   * @property toControl - The Angular FormControl for the \"to\" value.\n   */\n  @Input() props: NumberFromToMetadata;\n\n  private i18n = inject(I18nService);\n\n  /** Get from label with i18n fallback */\n  getFromLabel(): string {\n    return this.props.fromLabel || this.i18n.t('from');\n  }\n\n  /** Get to label with i18n fallback */\n  getToLabel(): string {\n    return this.props.toLabel || this.i18n.t('to');\n  }\n\n  constructor() {}\n\n  ngOnInit() {\n    // Apply default values if configured\n    if (this.props?.withDefault || this.props?.value) {\n      this.applyDefaultValues();\n    }\n  }\n\n  private applyDefaultValues() {\n    const defaultValue = this.resolveDefaultValue();\n    if (defaultValue !== null) {\n      if (typeof defaultValue === 'object' && defaultValue.from !== undefined) {\n        this.fromControl?.setValue(defaultValue.from);\n        this.toControl?.setValue(defaultValue.to);\n      } else {\n        // Si es un valor simple, aplicarlo a ambos controles\n        this.fromControl?.setValue(defaultValue);\n        this.toControl?.setValue(defaultValue);\n      }\n      this.fromControl?.markAsPristine();\n      this.toControl?.markAsPristine();\n      this.fromControl?.updateValueAndValidity();\n      this.toControl?.updateValueAndValidity();\n    }\n  }\n\n  private resolveDefaultValue(): any {\n    // Explicit value takes precedence\n    if (this.props.value !== undefined && this.props.value !== null && this.props.value !== '') {\n      return this.parseValue(this.props.value);\n    }\n\n    // No default configuration\n    if (!this.props.withDefault) {\n      return null;\n    }\n\n    // Custom default value\n    if (typeof this.props.withDefault === 'string') {\n      return this.parseValue(this.props.withDefault);\n    }\n\n    // Auto default value (withDefault === true)\n    return { from: 0, to: 0 };\n  }\n\n  private parseValue(value: string): any {\n    try {\n      // Try to parse as JSON first (for object values)\n      return JSON.parse(value);\n    } catch {\n      // If not JSON, treat as number\n      const numValue = parseFloat(value) || 0;\n      return { from: numValue, to: numValue };\n    }\n  }\n\n  get fromControl(): FormControl {\n    return this.props.fromControl;\n  }\n\n  get toControl(): FormControl {\n    return this.props.toControl;\n  }\n}\n"]}
@@ -1,9 +1,10 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, Input, Output, EventEmitter } from '@angular/core';
2
+ import { Component, inject, Input, Output, EventEmitter } from '@angular/core';
3
3
  import { ReactiveFormsModule } from '@angular/forms';
4
4
  import { IonButton, IonIcon, IonInput } from '@ionic/angular/standalone';
5
5
  import { addIcons } from 'ionicons';
6
6
  import { addOutline, removeOutline } from 'ionicons/icons';
7
+ import { I18nService } from '../../../services/i18n';
7
8
  import { ComponentStates } from '../../types';
8
9
  import * as i0 from "@angular/core";
9
10
  import * as i1 from "@angular/forms";
@@ -64,8 +65,17 @@ export class NumberStepperComponent {
64
65
  constructor() {
65
66
  this.valueChange = new EventEmitter();
66
67
  this.states = ComponentStates;
68
+ this.i18n = inject(I18nService);
67
69
  this.valueSubscription = null;
68
70
  }
71
+ /** Get decrement button aria-label with i18n fallback */
72
+ getDecrementLabel() {
73
+ return this.props.decrementLabel || this.i18n.t('decrease');
74
+ }
75
+ /** Get increment button aria-label with i18n fallback */
76
+ getIncrementLabel() {
77
+ return this.props.incrementLabel || this.i18n.t('increase');
78
+ }
69
79
  ngOnInit() {
70
80
  // Ensure initial value is within bounds
71
81
  this.clampValue();
@@ -206,7 +216,7 @@ export class NumberStepperComponent {
206
216
  [shape]="props.buttonShape || 'round'"
207
217
  [disabled]="isAtMin || props.state === states.DISABLED"
208
218
  (click)="decrement()"
209
- [attr.aria-label]="props.decrementLabel || 'Disminuir'"
219
+ [attr.aria-label]="getDecrementLabel()"
210
220
  >
211
221
  <ion-icon slot="icon-only" name="remove-outline"></ion-icon>
212
222
  </ion-button>
@@ -239,7 +249,7 @@ export class NumberStepperComponent {
239
249
  [shape]="props.buttonShape || 'round'"
240
250
  [disabled]="isAtMax || props.state === states.DISABLED"
241
251
  (click)="increment()"
242
- [attr.aria-label]="props.incrementLabel || 'Aumentar'"
252
+ [attr.aria-label]="getIncrementLabel()"
243
253
  >
244
254
  <ion-icon slot="icon-only" name="add-outline"></ion-icon>
245
255
  </ion-button>
@@ -306,7 +316,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
306
316
  [shape]="props.buttonShape || 'round'"
307
317
  [disabled]="isAtMin || props.state === states.DISABLED"
308
318
  (click)="decrement()"
309
- [attr.aria-label]="props.decrementLabel || 'Disminuir'"
319
+ [attr.aria-label]="getDecrementLabel()"
310
320
  >
311
321
  <ion-icon slot="icon-only" name="remove-outline"></ion-icon>
312
322
  </ion-button>
@@ -339,7 +349,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
339
349
  [shape]="props.buttonShape || 'round'"
340
350
  [disabled]="isAtMax || props.state === states.DISABLED"
341
351
  (click)="increment()"
342
- [attr.aria-label]="props.incrementLabel || 'Aumentar'"
352
+ [attr.aria-label]="getIncrementLabel()"
343
353
  >
344
354
  <ion-icon slot="icon-only" name="add-outline"></ion-icon>
345
355
  </ion-button>
@@ -366,4 +376,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
366
376
  }], valueChange: [{
367
377
  type: Output
368
378
  }] } });
369
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"number-stepper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/number-stepper/number-stepper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, Output, EventEmitter, OnInit, OnDestroy } from '@angular/core';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { IonButton, IonIcon, IonInput } from '@ionic/angular/standalone';\nimport { Subscription } from 'rxjs';\nimport { addIcons } from 'ionicons';\nimport { addOutline, removeOutline } from 'ionicons/icons';\nimport { ComponentStates } from '../../types';\nimport { NumberStepperMetadata, NumberStepperChangeEvent } from './types';\n\naddIcons({ addOutline, removeOutline });\n\n@Component({\n  selector: 'val-number-stepper',\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, IonButton, IonIcon, IonInput],\n  template: `\n    <div\n      class=\"number-stepper-container\"\n      [class]=\"props.cssClass\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [class.layout-vertical]=\"props.layout === 'vertical'\"\n      [class.disabled]=\"props.state === states.DISABLED\"\n      [style.--stepper-color]=\"getColor()\"\n    >\n      @if (props.label) {\n        <label class=\"stepper-label\">{{ getLabel() }}</label>\n      }\n\n      <!-- Preset buttons -->\n      @if (props.showPresets && props.presets?.length) {\n        <div class=\"preset-buttons\">\n          @for (preset of props.presets; track preset.value) {\n            <button\n              type=\"button\"\n              class=\"preset-button\"\n              [class.active]=\"currentValue === preset.value\"\n              [class.popular]=\"preset.popular\"\n              (click)=\"setValue(preset.value)\"\n              [disabled]=\"props.state === states.DISABLED\"\n            >\n              {{ preset.label || preset.value }}\n            </button>\n          }\n        </div>\n      }\n\n      <!-- Stepper controls -->\n      <div class=\"stepper-controls\">\n        <ion-button\n          class=\"stepper-button decrement\"\n          [color]=\"props.color || 'primary'\"\n          fill=\"outline\"\n          [shape]=\"props.buttonShape || 'round'\"\n          [disabled]=\"isAtMin || props.state === states.DISABLED\"\n          (click)=\"decrement()\"\n          [attr.aria-label]=\"props.decrementLabel || 'Disminuir'\"\n        >\n          <ion-icon slot=\"icon-only\" name=\"remove-outline\"></ion-icon>\n        </ion-button>\n\n        <div class=\"value-display\">\n          @if (props.showInput !== false && props.editableInput) {\n            <ion-input\n              type=\"number\"\n              [formControl]=\"props.control\"\n              [min]=\"props.min\"\n              [max]=\"props.max\"\n              [step]=\"props.step || 1\"\n              [disabled]=\"props.state === states.DISABLED\"\n              class=\"value-input\"\n              (ionBlur)=\"onInputBlur()\"\n            ></ion-input>\n          } @else {\n            <span class=\"value-text\">{{ currentValue }}</span>\n          }\n\n          @if (props.unitLabel) {\n            <span class=\"unit-label\">{{ getUnitLabel() }}</span>\n          }\n        </div>\n\n        <ion-button\n          class=\"stepper-button increment\"\n          [color]=\"props.color || 'primary'\"\n          fill=\"outline\"\n          [shape]=\"props.buttonShape || 'round'\"\n          [disabled]=\"isAtMax || props.state === states.DISABLED\"\n          (click)=\"increment()\"\n          [attr.aria-label]=\"props.incrementLabel || 'Aumentar'\"\n        >\n          <ion-icon slot=\"icon-only\" name=\"add-outline\"></ion-icon>\n        </ion-button>\n      </div>\n\n      <!-- Price display -->\n      @if (props.showTotal && props.unitPrice) {\n        <div class=\"price-display\">\n          <span class=\"price-calculation\">\n            {{ currentValue }} × {{ formatPrice(props.unitPrice) }}\n          </span>\n          <span class=\"price-equals\">=</span>\n          <span class=\"price-total\">{{ formatPrice(totalPrice) }}</span>\n        </div>\n      }\n\n      @if (props.hint) {\n        <span class=\"stepper-hint\">{{ props.hint }}</span>\n      }\n    </div>\n  `,\n  styleUrls: ['./number-stepper.component.scss'],\n})\n/**\n * val-number-stepper\n *\n * A number input with increment/decrement buttons, ideal for quantity selection.\n *\n * @example Basic usage\n * ```html\n * <val-number-stepper\n *   [props]=\"{\n *     control: quantityControl,\n *     min: 1,\n *     max: 100\n *   }\"\n * ></val-number-stepper>\n * ```\n *\n * @example With pricing\n * ```html\n * <val-number-stepper\n *   [props]=\"{\n *     control: ticketControl,\n *     min: 1,\n *     max: 50,\n *     unitPrice: 10,\n *     currency: 'MXN',\n *     currencySymbol: '$',\n *     showTotal: true,\n *     unitLabel: 'boletos',\n *     unitLabelSingular: 'boleto'\n *   }\"\n *   (valueChange)=\"onQuantityChange($event)\"\n * ></val-number-stepper>\n * ```\n *\n * @example With presets\n * ```html\n * <val-number-stepper\n *   [props]=\"{\n *     control: ticketControl,\n *     min: 1,\n *     max: 100,\n *     showPresets: true,\n *     presets: [\n *       { value: 1 },\n *       { value: 5, popular: true },\n *       { value: 10 },\n *       { value: 25 }\n *     ]\n *   }\"\n * ></val-number-stepper>\n * ```\n */\nexport class NumberStepperComponent implements OnInit, OnDestroy {\n  @Input() props: NumberStepperMetadata;\n\n  @Output() valueChange = new EventEmitter<NumberStepperChangeEvent>();\n\n  states = ComponentStates;\n\n  private valueSubscription: Subscription | null = null;\n\n  ngOnInit(): void {\n    // Ensure initial value is within bounds\n    this.clampValue();\n\n    // Subscribe to value changes\n    this.valueSubscription = this.props.control.valueChanges.subscribe(() => {\n      this.clampValue();\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.valueSubscription?.unsubscribe();\n  }\n\n  get currentValue(): number {\n    return this.props.control.value ?? this.props.min ?? 0;\n  }\n\n  get isAtMin(): boolean {\n    if (this.props.min === undefined) return false;\n    return this.currentValue <= this.props.min;\n  }\n\n  get isAtMax(): boolean {\n    if (this.props.max === undefined) return false;\n    return this.currentValue >= this.props.max;\n  }\n\n  get totalPrice(): number {\n    return this.currentValue * (this.props.unitPrice || 0);\n  }\n\n  increment(): void {\n    const step = this.props.step || 1;\n    const newValue = this.currentValue + step;\n    this.setValue(newValue);\n  }\n\n  decrement(): void {\n    const step = this.props.step || 1;\n    const newValue = this.currentValue - step;\n    this.setValue(newValue);\n  }\n\n  setValue(value: number): void {\n    const previousValue = this.currentValue;\n    const clampedValue = this.clamp(value);\n\n    if (clampedValue !== previousValue) {\n      this.props.control.setValue(clampedValue);\n      this.emitChange(clampedValue, previousValue);\n    }\n  }\n\n  onInputBlur(): void {\n    this.clampValue();\n  }\n\n  private clampValue(): void {\n    const currentVal = this.props.control.value;\n    const clampedVal = this.clamp(currentVal);\n    if (currentVal !== clampedVal) {\n      this.props.control.setValue(clampedVal);\n    }\n  }\n\n  private clamp(value: number): number {\n    let result = value;\n    if (this.props.min !== undefined && result < this.props.min) {\n      result = this.props.min;\n    }\n    if (this.props.max !== undefined && result > this.props.max) {\n      result = this.props.max;\n    }\n    return result;\n  }\n\n  private emitChange(value: number, previousValue: number): void {\n    this.valueChange.emit({\n      value,\n      previousValue,\n      totalPrice: this.props.unitPrice ? value * this.props.unitPrice : undefined,\n      atMin: this.isAtMin,\n      atMax: this.isAtMax,\n    });\n  }\n\n  getLabel(): string {\n    return this.props.label || '';\n  }\n\n  getUnitLabel(): string {\n    if (this.currentValue === 1 && this.props.unitLabelSingular) {\n      return this.props.unitLabelSingular;\n    }\n    return this.props.unitLabel || '';\n  }\n\n  formatPrice(amount: number): string {\n    const symbol = this.props.currencySymbol || '$';\n    const formatted = amount.toLocaleString('es-MX', {\n      minimumFractionDigits: 2,\n      maximumFractionDigits: 2,\n    });\n    return `${symbol}${formatted}`;\n  }\n\n  getColor(): string {\n    if (this.props.color) {\n      return `var(--ion-color-${this.props.color})`;\n    }\n    return 'var(--ion-color-primary)';\n  }\n}\n"]}
379
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"number-stepper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/number-stepper/number-stepper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, inject, Input, Output, EventEmitter, OnInit, OnDestroy } from '@angular/core';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { IonButton, IonIcon, IonInput } from '@ionic/angular/standalone';\nimport { Subscription } from 'rxjs';\nimport { addIcons } from 'ionicons';\nimport { addOutline, removeOutline } from 'ionicons/icons';\nimport { I18nService } from '../../../services/i18n';\nimport { ComponentStates } from '../../types';\nimport { NumberStepperMetadata, NumberStepperChangeEvent } from './types';\n\naddIcons({ addOutline, removeOutline });\n\n@Component({\n  selector: 'val-number-stepper',\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, IonButton, IonIcon, IonInput],\n  template: `\n    <div\n      class=\"number-stepper-container\"\n      [class]=\"props.cssClass\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [class.layout-vertical]=\"props.layout === 'vertical'\"\n      [class.disabled]=\"props.state === states.DISABLED\"\n      [style.--stepper-color]=\"getColor()\"\n    >\n      @if (props.label) {\n        <label class=\"stepper-label\">{{ getLabel() }}</label>\n      }\n\n      <!-- Preset buttons -->\n      @if (props.showPresets && props.presets?.length) {\n        <div class=\"preset-buttons\">\n          @for (preset of props.presets; track preset.value) {\n            <button\n              type=\"button\"\n              class=\"preset-button\"\n              [class.active]=\"currentValue === preset.value\"\n              [class.popular]=\"preset.popular\"\n              (click)=\"setValue(preset.value)\"\n              [disabled]=\"props.state === states.DISABLED\"\n            >\n              {{ preset.label || preset.value }}\n            </button>\n          }\n        </div>\n      }\n\n      <!-- Stepper controls -->\n      <div class=\"stepper-controls\">\n        <ion-button\n          class=\"stepper-button decrement\"\n          [color]=\"props.color || 'primary'\"\n          fill=\"outline\"\n          [shape]=\"props.buttonShape || 'round'\"\n          [disabled]=\"isAtMin || props.state === states.DISABLED\"\n          (click)=\"decrement()\"\n          [attr.aria-label]=\"getDecrementLabel()\"\n        >\n          <ion-icon slot=\"icon-only\" name=\"remove-outline\"></ion-icon>\n        </ion-button>\n\n        <div class=\"value-display\">\n          @if (props.showInput !== false && props.editableInput) {\n            <ion-input\n              type=\"number\"\n              [formControl]=\"props.control\"\n              [min]=\"props.min\"\n              [max]=\"props.max\"\n              [step]=\"props.step || 1\"\n              [disabled]=\"props.state === states.DISABLED\"\n              class=\"value-input\"\n              (ionBlur)=\"onInputBlur()\"\n            ></ion-input>\n          } @else {\n            <span class=\"value-text\">{{ currentValue }}</span>\n          }\n\n          @if (props.unitLabel) {\n            <span class=\"unit-label\">{{ getUnitLabel() }}</span>\n          }\n        </div>\n\n        <ion-button\n          class=\"stepper-button increment\"\n          [color]=\"props.color || 'primary'\"\n          fill=\"outline\"\n          [shape]=\"props.buttonShape || 'round'\"\n          [disabled]=\"isAtMax || props.state === states.DISABLED\"\n          (click)=\"increment()\"\n          [attr.aria-label]=\"getIncrementLabel()\"\n        >\n          <ion-icon slot=\"icon-only\" name=\"add-outline\"></ion-icon>\n        </ion-button>\n      </div>\n\n      <!-- Price display -->\n      @if (props.showTotal && props.unitPrice) {\n        <div class=\"price-display\">\n          <span class=\"price-calculation\">\n            {{ currentValue }} × {{ formatPrice(props.unitPrice) }}\n          </span>\n          <span class=\"price-equals\">=</span>\n          <span class=\"price-total\">{{ formatPrice(totalPrice) }}</span>\n        </div>\n      }\n\n      @if (props.hint) {\n        <span class=\"stepper-hint\">{{ props.hint }}</span>\n      }\n    </div>\n  `,\n  styleUrls: ['./number-stepper.component.scss'],\n})\n/**\n * val-number-stepper\n *\n * A number input with increment/decrement buttons, ideal for quantity selection.\n *\n * @example Basic usage\n * ```html\n * <val-number-stepper\n *   [props]=\"{\n *     control: quantityControl,\n *     min: 1,\n *     max: 100\n *   }\"\n * ></val-number-stepper>\n * ```\n *\n * @example With pricing\n * ```html\n * <val-number-stepper\n *   [props]=\"{\n *     control: ticketControl,\n *     min: 1,\n *     max: 50,\n *     unitPrice: 10,\n *     currency: 'MXN',\n *     currencySymbol: '$',\n *     showTotal: true,\n *     unitLabel: 'boletos',\n *     unitLabelSingular: 'boleto'\n *   }\"\n *   (valueChange)=\"onQuantityChange($event)\"\n * ></val-number-stepper>\n * ```\n *\n * @example With presets\n * ```html\n * <val-number-stepper\n *   [props]=\"{\n *     control: ticketControl,\n *     min: 1,\n *     max: 100,\n *     showPresets: true,\n *     presets: [\n *       { value: 1 },\n *       { value: 5, popular: true },\n *       { value: 10 },\n *       { value: 25 }\n *     ]\n *   }\"\n * ></val-number-stepper>\n * ```\n */\nexport class NumberStepperComponent implements OnInit, OnDestroy {\n  @Input() props: NumberStepperMetadata;\n\n  @Output() valueChange = new EventEmitter<NumberStepperChangeEvent>();\n\n  states = ComponentStates;\n\n  private i18n = inject(I18nService);\n  private valueSubscription: Subscription | null = null;\n\n  /** Get decrement button aria-label with i18n fallback */\n  getDecrementLabel(): string {\n    return this.props.decrementLabel || this.i18n.t('decrease');\n  }\n\n  /** Get increment button aria-label with i18n fallback */\n  getIncrementLabel(): string {\n    return this.props.incrementLabel || this.i18n.t('increase');\n  }\n\n  ngOnInit(): void {\n    // Ensure initial value is within bounds\n    this.clampValue();\n\n    // Subscribe to value changes\n    this.valueSubscription = this.props.control.valueChanges.subscribe(() => {\n      this.clampValue();\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.valueSubscription?.unsubscribe();\n  }\n\n  get currentValue(): number {\n    return this.props.control.value ?? this.props.min ?? 0;\n  }\n\n  get isAtMin(): boolean {\n    if (this.props.min === undefined) return false;\n    return this.currentValue <= this.props.min;\n  }\n\n  get isAtMax(): boolean {\n    if (this.props.max === undefined) return false;\n    return this.currentValue >= this.props.max;\n  }\n\n  get totalPrice(): number {\n    return this.currentValue * (this.props.unitPrice || 0);\n  }\n\n  increment(): void {\n    const step = this.props.step || 1;\n    const newValue = this.currentValue + step;\n    this.setValue(newValue);\n  }\n\n  decrement(): void {\n    const step = this.props.step || 1;\n    const newValue = this.currentValue - step;\n    this.setValue(newValue);\n  }\n\n  setValue(value: number): void {\n    const previousValue = this.currentValue;\n    const clampedValue = this.clamp(value);\n\n    if (clampedValue !== previousValue) {\n      this.props.control.setValue(clampedValue);\n      this.emitChange(clampedValue, previousValue);\n    }\n  }\n\n  onInputBlur(): void {\n    this.clampValue();\n  }\n\n  private clampValue(): void {\n    const currentVal = this.props.control.value;\n    const clampedVal = this.clamp(currentVal);\n    if (currentVal !== clampedVal) {\n      this.props.control.setValue(clampedVal);\n    }\n  }\n\n  private clamp(value: number): number {\n    let result = value;\n    if (this.props.min !== undefined && result < this.props.min) {\n      result = this.props.min;\n    }\n    if (this.props.max !== undefined && result > this.props.max) {\n      result = this.props.max;\n    }\n    return result;\n  }\n\n  private emitChange(value: number, previousValue: number): void {\n    this.valueChange.emit({\n      value,\n      previousValue,\n      totalPrice: this.props.unitPrice ? value * this.props.unitPrice : undefined,\n      atMin: this.isAtMin,\n      atMax: this.isAtMax,\n    });\n  }\n\n  getLabel(): string {\n    return this.props.label || '';\n  }\n\n  getUnitLabel(): string {\n    if (this.currentValue === 1 && this.props.unitLabelSingular) {\n      return this.props.unitLabelSingular;\n    }\n    return this.props.unitLabel || '';\n  }\n\n  formatPrice(amount: number): string {\n    const symbol = this.props.currencySymbol || '$';\n    const formatted = amount.toLocaleString('es-MX', {\n      minimumFractionDigits: 2,\n      maximumFractionDigits: 2,\n    });\n    return `${symbol}${formatted}`;\n  }\n\n  getColor(): string {\n    if (this.props.color) {\n      return `var(--ion-color-${this.props.color})`;\n    }\n    return 'var(--ion-color-primary)';\n  }\n}\n"]}
@@ -41,8 +41,9 @@ export class DocsTocComponent {
41
41
  // Reactive state
42
42
  this.activeId = signal('');
43
43
  this.flatItems = signal([]);
44
- this.observer = null;
45
44
  this.headingElements = [];
45
+ this.scrollHandler = null;
46
+ this.rafId = null;
46
47
  }
47
48
  ngOnInit() {
48
49
  this.updateFlatItems();
@@ -132,34 +133,57 @@ export class DocsTocComponent {
132
133
  .filter((el) => el !== null);
133
134
  if (this.headingElements.length === 0)
134
135
  return;
135
- const offsetTop = this.props.offsetTop ?? 100;
136
- // Use IntersectionObserver for scroll spy
136
+ // Set initial active item
137
+ this.updateActiveHeading();
138
+ // Use scroll event with requestAnimationFrame for performance
137
139
  this.ngZone.runOutsideAngular(() => {
138
- this.observer = new IntersectionObserver(entries => {
139
- // Find the first visible heading
140
- const visibleEntries = entries
141
- .filter(entry => entry.isIntersecting)
142
- .sort((a, b) => a.boundingClientRect.top - b.boundingClientRect.top);
143
- if (visibleEntries.length > 0) {
144
- const activeEntry = visibleEntries[0];
145
- this.ngZone.run(() => {
146
- this.activeId.set(activeEntry.target.id);
147
- this.sectionChange.emit(activeEntry.target.id);
148
- });
140
+ this.scrollHandler = () => {
141
+ if (this.rafId) {
142
+ cancelAnimationFrame(this.rafId);
149
143
  }
150
- }, {
151
- rootMargin: `-${offsetTop}px 0px -70% 0px`,
152
- threshold: 0,
153
- });
154
- this.headingElements.forEach(el => {
155
- this.observer?.observe(el);
156
- });
144
+ this.rafId = requestAnimationFrame(() => {
145
+ this.updateActiveHeading();
146
+ });
147
+ };
148
+ window.addEventListener('scroll', this.scrollHandler, { passive: true });
157
149
  });
158
150
  }
151
+ updateActiveHeading() {
152
+ const offsetTop = this.props.offsetTop ?? 100;
153
+ const scrollY = window.scrollY;
154
+ // Find the heading that is currently at or above the offset point
155
+ let activeHeading = null;
156
+ for (const heading of this.headingElements) {
157
+ const rect = heading.getBoundingClientRect();
158
+ const headingTop = rect.top + scrollY;
159
+ // If heading is at or above the trigger point (scrollY + offset)
160
+ if (headingTop <= scrollY + offsetTop + 10) {
161
+ activeHeading = heading;
162
+ }
163
+ else {
164
+ // Headings are in order, so we can stop once we find one below
165
+ break;
166
+ }
167
+ }
168
+ // If no heading is above the trigger point, use the first one
169
+ if (!activeHeading && this.headingElements.length > 0) {
170
+ activeHeading = this.headingElements[0];
171
+ }
172
+ if (activeHeading && activeHeading.id !== this.activeId()) {
173
+ this.ngZone.run(() => {
174
+ this.activeId.set(activeHeading.id);
175
+ this.sectionChange.emit(activeHeading.id);
176
+ });
177
+ }
178
+ }
159
179
  destroyScrollSpy() {
160
- if (this.observer) {
161
- this.observer.disconnect();
162
- this.observer = null;
180
+ if (this.scrollHandler) {
181
+ window.removeEventListener('scroll', this.scrollHandler);
182
+ this.scrollHandler = null;
183
+ }
184
+ if (this.rafId) {
185
+ cancelAnimationFrame(this.rafId);
186
+ this.rafId = null;
163
187
  }
164
188
  this.headingElements = [];
165
189
  }
@@ -252,4 +276,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
252
276
  }], sectionChange: [{
253
277
  type: Output
254
278
  }] } });
255
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-toc.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/docs-toc/docs-toc.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  AfterViewInit,\n  NgZone,\n  signal,\n  computed,\n} from '@angular/core';\nimport { DocsTocMetadata, DocsTocItem } from './types';\n\n/**\n * val-docs-toc\n *\n * A table of contents component with scroll spy functionality.\n * Automatically highlights the current section based on scroll position.\n *\n * @example Manual items\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     title: 'On this page',\n *     items: [\n *       { id: 'overview', label: 'Overview', level: 1 },\n *       { id: 'installation', label: 'Installation', level: 1 },\n *       { id: 'npm', label: 'Using npm', level: 2 },\n *       { id: 'yarn', label: 'Using yarn', level: 2 }\n *     ]\n *   }\"\n * ></val-docs-toc>\n * ```\n *\n * @example Auto-generated from headings\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     autoGenerate: true,\n *     containerSelector: '.article-content',\n *     headingLevels: [2, 3]\n *   }\"\n * ></val-docs-toc>\n * ```\n */\n@Component({\n  selector: 'val-docs-toc',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <nav\n      class=\"docs-toc\"\n      [class]=\"props.cssClass\"\n      role=\"navigation\"\n      aria-label=\"Table of contents\"\n    >\n      @if (!props.hideTitle) {\n        <h4 class=\"docs-toc__title\">{{ props.title || 'Contents' }}</h4>\n      }\n\n      <ul class=\"docs-toc__list\">\n        @for (item of flatItems(); track item.id) {\n          <li\n            class=\"docs-toc__item\"\n            [class.docs-toc__item--level-1]=\"item.level === 1\"\n            [class.docs-toc__item--level-2]=\"item.level === 2\"\n            [class.docs-toc__item--level-3]=\"item.level === 3\"\n            [class.docs-toc__item--active]=\"activeId() === item.id\"\n          >\n            <a\n              class=\"docs-toc__link\"\n              [href]=\"'#' + item.id\"\n              (click)=\"scrollToSection($event, item.id)\"\n            >\n              {{ item.label }}\n            </a>\n          </li>\n        }\n      </ul>\n    </nav>\n  `,\n  styleUrls: ['./docs-toc.component.scss'],\n})\nexport class DocsTocComponent implements OnInit, OnDestroy, OnChanges, AfterViewInit {\n  @Input() props: DocsTocMetadata = {};\n\n  @Output() sectionChange = new EventEmitter<string>();\n\n  // Reactive state\n  protected activeId = signal<string>('');\n  protected flatItems = signal<DocsTocItem[]>([]);\n\n  private observer: IntersectionObserver | null = null;\n  private headingElements: HTMLElement[] = [];\n\n  constructor(private ngZone: NgZone) {}\n\n  ngOnInit(): void {\n    this.updateFlatItems();\n  }\n\n  ngAfterViewInit(): void {\n    // Delay to ensure DOM is ready\n    setTimeout(() => {\n      if (this.props.autoGenerate) {\n        this.generateTocFromHeadings();\n      }\n      this.setupScrollSpy();\n    }, 100);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['props']) {\n      this.updateFlatItems();\n      if (this.props.autoGenerate) {\n        setTimeout(() => this.generateTocFromHeadings(), 100);\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroyScrollSpy();\n  }\n\n  private updateFlatItems(): void {\n    if (this.props.items) {\n      this.flatItems.set(this.flattenItems(this.props.items));\n    }\n  }\n\n  private flattenItems(items: DocsTocItem[]): DocsTocItem[] {\n    const result: DocsTocItem[] = [];\n    const flatten = (itemList: DocsTocItem[]) => {\n      itemList.forEach(item => {\n        result.push(item);\n        if (item.children?.length) {\n          flatten(item.children);\n        }\n      });\n    };\n    flatten(items);\n    return result;\n  }\n\n  private generateTocFromHeadings(): void {\n    const selector = this.props.containerSelector || '.docs-layout__content';\n    const container = document.querySelector(selector);\n\n    if (!container) {\n      console.warn(`[docs-toc] Container not found: ${selector}`);\n      return;\n    }\n\n    const levels = this.props.headingLevels || [2, 3];\n    const headingSelector = levels.map(l => `h${l}`).join(', ');\n    const headings = container.querySelectorAll(headingSelector);\n\n    const items: DocsTocItem[] = [];\n    headings.forEach((heading: Element) => {\n      const el = heading as HTMLElement;\n      // Generate ID if not present\n      if (!el.id) {\n        el.id = this.generateId(el.textContent || '');\n      }\n\n      const level = parseInt(el.tagName[1], 10);\n      items.push({\n        id: el.id,\n        label: el.textContent?.trim() || '',\n        level: level - 1, // h2 = level 1, h3 = level 2\n      });\n    });\n\n    this.flatItems.set(items);\n    // Re-setup scroll spy with new items\n    this.setupScrollSpy();\n  }\n\n  private generateId(text: string): string {\n    return text\n      .toLowerCase()\n      .trim()\n      .replace(/[^\\w\\s-]/g, '')\n      .replace(/\\s+/g, '-')\n      .substring(0, 50);\n  }\n\n  private setupScrollSpy(): void {\n    this.destroyScrollSpy();\n\n    const items = this.flatItems();\n    if (items.length === 0) return;\n\n    // Collect heading elements\n    this.headingElements = items\n      .map(item => document.getElementById(item.id))\n      .filter((el): el is HTMLElement => el !== null);\n\n    if (this.headingElements.length === 0) return;\n\n    const offsetTop = this.props.offsetTop ?? 100;\n\n    // Use IntersectionObserver for scroll spy\n    this.ngZone.runOutsideAngular(() => {\n      this.observer = new IntersectionObserver(\n        entries => {\n          // Find the first visible heading\n          const visibleEntries = entries\n            .filter(entry => entry.isIntersecting)\n            .sort((a, b) => a.boundingClientRect.top - b.boundingClientRect.top);\n\n          if (visibleEntries.length > 0) {\n            const activeEntry = visibleEntries[0];\n            this.ngZone.run(() => {\n              this.activeId.set(activeEntry.target.id);\n              this.sectionChange.emit(activeEntry.target.id);\n            });\n          }\n        },\n        {\n          rootMargin: `-${offsetTop}px 0px -70% 0px`,\n          threshold: 0,\n        }\n      );\n\n      this.headingElements.forEach(el => {\n        this.observer?.observe(el);\n      });\n    });\n  }\n\n  private destroyScrollSpy(): void {\n    if (this.observer) {\n      this.observer.disconnect();\n      this.observer = null;\n    }\n    this.headingElements = [];\n  }\n\n  scrollToSection(event: Event, id: string): void {\n    event.preventDefault();\n\n    const element = document.getElementById(id);\n    if (!element) return;\n\n    const offsetTop = this.props.offsetTop ?? 100;\n\n    const top = element.getBoundingClientRect().top + window.scrollY - offsetTop;\n\n    window.scrollTo({\n      top,\n      behavior: 'smooth',\n    });\n\n    // Update active immediately for better UX\n    this.activeId.set(id);\n\n    // Update URL hash without jumping\n    history.pushState(null, '', `#${id}`);\n  }\n}\n"]}
279
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-toc.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/docs-toc/docs-toc.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  AfterViewInit,\n  NgZone,\n  signal,\n} from '@angular/core';\nimport { DocsTocMetadata, DocsTocItem } from './types';\n\n/**\n * val-docs-toc\n *\n * A table of contents component with scroll spy functionality.\n * Automatically highlights the current section based on scroll position.\n *\n * @example Manual items\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     title: 'On this page',\n *     items: [\n *       { id: 'overview', label: 'Overview', level: 1 },\n *       { id: 'installation', label: 'Installation', level: 1 },\n *       { id: 'npm', label: 'Using npm', level: 2 },\n *       { id: 'yarn', label: 'Using yarn', level: 2 }\n *     ]\n *   }\"\n * ></val-docs-toc>\n * ```\n *\n * @example Auto-generated from headings\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     autoGenerate: true,\n *     containerSelector: '.article-content',\n *     headingLevels: [2, 3]\n *   }\"\n * ></val-docs-toc>\n * ```\n */\n@Component({\n  selector: 'val-docs-toc',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <nav\n      class=\"docs-toc\"\n      [class]=\"props.cssClass\"\n      role=\"navigation\"\n      aria-label=\"Table of contents\"\n    >\n      @if (!props.hideTitle) {\n        <h4 class=\"docs-toc__title\">{{ props.title || 'Contents' }}</h4>\n      }\n\n      <ul class=\"docs-toc__list\">\n        @for (item of flatItems(); track item.id) {\n          <li\n            class=\"docs-toc__item\"\n            [class.docs-toc__item--level-1]=\"item.level === 1\"\n            [class.docs-toc__item--level-2]=\"item.level === 2\"\n            [class.docs-toc__item--level-3]=\"item.level === 3\"\n            [class.docs-toc__item--active]=\"activeId() === item.id\"\n          >\n            <a\n              class=\"docs-toc__link\"\n              [href]=\"'#' + item.id\"\n              (click)=\"scrollToSection($event, item.id)\"\n            >\n              {{ item.label }}\n            </a>\n          </li>\n        }\n      </ul>\n    </nav>\n  `,\n  styleUrls: ['./docs-toc.component.scss'],\n})\nexport class DocsTocComponent implements OnInit, OnDestroy, OnChanges, AfterViewInit {\n  @Input() props: DocsTocMetadata = {};\n\n  @Output() sectionChange = new EventEmitter<string>();\n\n  // Reactive state\n  protected activeId = signal<string>('');\n  protected flatItems = signal<DocsTocItem[]>([]);\n\n  private headingElements: HTMLElement[] = [];\n  private scrollHandler: (() => void) | null = null;\n  private rafId: number | null = null;\n\n  constructor(private ngZone: NgZone) {}\n\n  ngOnInit(): void {\n    this.updateFlatItems();\n  }\n\n  ngAfterViewInit(): void {\n    // Delay to ensure DOM is ready\n    setTimeout(() => {\n      if (this.props.autoGenerate) {\n        this.generateTocFromHeadings();\n      }\n      this.setupScrollSpy();\n    }, 100);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['props']) {\n      this.updateFlatItems();\n      if (this.props.autoGenerate) {\n        setTimeout(() => this.generateTocFromHeadings(), 100);\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroyScrollSpy();\n  }\n\n  private updateFlatItems(): void {\n    if (this.props.items) {\n      this.flatItems.set(this.flattenItems(this.props.items));\n    }\n  }\n\n  private flattenItems(items: DocsTocItem[]): DocsTocItem[] {\n    const result: DocsTocItem[] = [];\n    const flatten = (itemList: DocsTocItem[]) => {\n      itemList.forEach(item => {\n        result.push(item);\n        if (item.children?.length) {\n          flatten(item.children);\n        }\n      });\n    };\n    flatten(items);\n    return result;\n  }\n\n  private generateTocFromHeadings(): void {\n    const selector = this.props.containerSelector || '.docs-layout__content';\n    const container = document.querySelector(selector);\n\n    if (!container) {\n      console.warn(`[docs-toc] Container not found: ${selector}`);\n      return;\n    }\n\n    const levels = this.props.headingLevels || [2, 3];\n    const headingSelector = levels.map(l => `h${l}`).join(', ');\n    const headings = container.querySelectorAll(headingSelector);\n\n    const items: DocsTocItem[] = [];\n    headings.forEach((heading: Element) => {\n      const el = heading as HTMLElement;\n      // Generate ID if not present\n      if (!el.id) {\n        el.id = this.generateId(el.textContent || '');\n      }\n\n      const level = parseInt(el.tagName[1], 10);\n      items.push({\n        id: el.id,\n        label: el.textContent?.trim() || '',\n        level: level - 1, // h2 = level 1, h3 = level 2\n      });\n    });\n\n    this.flatItems.set(items);\n    // Re-setup scroll spy with new items\n    this.setupScrollSpy();\n  }\n\n  private generateId(text: string): string {\n    return text\n      .toLowerCase()\n      .trim()\n      .replace(/[^\\w\\s-]/g, '')\n      .replace(/\\s+/g, '-')\n      .substring(0, 50);\n  }\n\n  private setupScrollSpy(): void {\n    this.destroyScrollSpy();\n\n    const items = this.flatItems();\n    if (items.length === 0) return;\n\n    // Collect heading elements\n    this.headingElements = items\n      .map(item => document.getElementById(item.id))\n      .filter((el): el is HTMLElement => el !== null);\n\n    if (this.headingElements.length === 0) return;\n\n    // Set initial active item\n    this.updateActiveHeading();\n\n    // Use scroll event with requestAnimationFrame for performance\n    this.ngZone.runOutsideAngular(() => {\n      this.scrollHandler = () => {\n        if (this.rafId) {\n          cancelAnimationFrame(this.rafId);\n        }\n        this.rafId = requestAnimationFrame(() => {\n          this.updateActiveHeading();\n        });\n      };\n\n      window.addEventListener('scroll', this.scrollHandler, { passive: true });\n    });\n  }\n\n  private updateActiveHeading(): void {\n    const offsetTop = this.props.offsetTop ?? 100;\n    const scrollY = window.scrollY;\n\n    // Find the heading that is currently at or above the offset point\n    let activeHeading: HTMLElement | null = null;\n\n    for (const heading of this.headingElements) {\n      const rect = heading.getBoundingClientRect();\n      const headingTop = rect.top + scrollY;\n\n      // If heading is at or above the trigger point (scrollY + offset)\n      if (headingTop <= scrollY + offsetTop + 10) {\n        activeHeading = heading;\n      } else {\n        // Headings are in order, so we can stop once we find one below\n        break;\n      }\n    }\n\n    // If no heading is above the trigger point, use the first one\n    if (!activeHeading && this.headingElements.length > 0) {\n      activeHeading = this.headingElements[0];\n    }\n\n    if (activeHeading && activeHeading.id !== this.activeId()) {\n      this.ngZone.run(() => {\n        this.activeId.set(activeHeading!.id);\n        this.sectionChange.emit(activeHeading!.id);\n      });\n    }\n  }\n\n  private destroyScrollSpy(): void {\n    if (this.scrollHandler) {\n      window.removeEventListener('scroll', this.scrollHandler);\n      this.scrollHandler = null;\n    }\n    if (this.rafId) {\n      cancelAnimationFrame(this.rafId);\n      this.rafId = null;\n    }\n    this.headingElements = [];\n  }\n\n  scrollToSection(event: Event, id: string): void {\n    event.preventDefault();\n\n    const element = document.getElementById(id);\n    if (!element) return;\n\n    const offsetTop = this.props.offsetTop ?? 100;\n\n    const top = element.getBoundingClientRect().top + window.scrollY - offsetTop;\n\n    window.scrollTo({\n      top,\n      behavior: 'smooth',\n    });\n\n    // Update active immediately for better UX\n    this.activeId.set(id);\n\n    // Update URL hash without jumping\n    history.pushState(null, '', `#${id}`);\n  }\n}\n"]}