valtech-components 2.0.589 → 2.0.591

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.
@@ -48,6 +48,12 @@ export class LinksAccordionComponent {
48
48
  */
49
49
  this.navigate = new EventEmitter();
50
50
  }
51
+ /**
52
+ * Gets a unique value for the accordion section.
53
+ */
54
+ getSectionValue(section, index) {
55
+ return section.titleKey || section.title || `section-${index}`;
56
+ }
51
57
  /**
52
58
  * Gets the section title, supporting i18n via titleKey.
53
59
  */
@@ -91,9 +97,10 @@ export class LinksAccordionComponent {
91
97
  <div class="links-accordion">
92
98
  <ion-accordion-group #accordionGroup>
93
99
  <ion-accordion
94
- *ngFor="let section of props.sections"
95
- [value]="section.title"
96
- [toggleIcon]="section.links?.length ? 'chevron-down-outline' : 'chevron-forward-outline'"
100
+ *ngFor="let section of props.sections; let i = index"
101
+ [value]="getSectionValue(section, i)"
102
+ [toggleIcon]="section.links?.length ? 'chevron-down-outline' : ''"
103
+ [readonly]="!section.links?.length"
97
104
  >
98
105
  <ion-item
99
106
  slot="header"
@@ -132,9 +139,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
132
139
  <div class="links-accordion">
133
140
  <ion-accordion-group #accordionGroup>
134
141
  <ion-accordion
135
- *ngFor="let section of props.sections"
136
- [value]="section.title"
137
- [toggleIcon]="section.links?.length ? 'chevron-down-outline' : 'chevron-forward-outline'"
142
+ *ngFor="let section of props.sections; let i = index"
143
+ [value]="getSectionValue(section, i)"
144
+ [toggleIcon]="section.links?.length ? 'chevron-down-outline' : ''"
145
+ [readonly]="!section.links?.length"
138
146
  >
139
147
  <ion-item
140
148
  slot="header"
@@ -174,4 +182,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
174
182
  type: ViewChild,
175
183
  args: ['accordionGroup']
176
184
  }] } });
177
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"links-accordion.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/links-accordion/links-accordion.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output, ViewChild } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { IonAccordion, IonAccordionGroup, IonItem, IonLabel, IonList, MenuController } from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport { LinksAccordionLink, LinksAccordionMetadata, LinksAccordionSection } from './types';\n\n/**\n * val-links-accordion\n *\n * A hierarchical navigation component using Ionic accordions.\n * Supports sections with optional sub-links and automatic menu closing.\n *\n * @example\n * <val-links-accordion\n *   [props]=\"{\n *     sections: [\n *       { title: 'Home', route: ['/'] },\n *       { title: 'Products', links: [\n *         { text: 'Catalog', route: ['/products'] },\n *         { text: 'New Arrivals', route: ['/products', 'new'] }\n *       ]}\n *     ]\n *   }\"\n *   (navigate)=\"onNavigate($event)\"\n * ></val-links-accordion>\n *\n * @input props - Accordion configuration\n * @output navigate - Emits route when a link is clicked\n */\n@Component({\n  selector: 'val-links-accordion',\n  standalone: true,\n  imports: [CommonModule, IonAccordionGroup, IonAccordion, IonItem, IonLabel, IonList],\n  template: `\n    <div class=\"links-accordion\">\n      <ion-accordion-group #accordionGroup>\n        <ion-accordion\n          *ngFor=\"let section of props.sections\"\n          [value]=\"section.title\"\n          [toggleIcon]=\"section.links?.length ? 'chevron-down-outline' : 'chevron-forward-outline'\"\n        >\n          <ion-item\n            slot=\"header\"\n            color=\"light\"\n            [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n            (click)=\"section.route ? onLinkClick(section.route, $event) : null\"\n            [button]=\"!!section.route\"\n            lines=\"none\"\n          >\n            <ion-label class=\"accordion-label\">{{ getSectionTitle(section) }}</ion-label>\n          </ion-item>\n          <div\n            slot=\"content\"\n            [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n            *ngIf=\"section.links?.length\"\n          >\n            <ion-list lines=\"none\">\n              <ion-item\n                *ngFor=\"let link of section.links\"\n                [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n                (click)=\"onLinkClick(link.route, $event)\"\n                button\n              >\n                <ion-label class=\"accordion-item-label\">{{ getLinkText(link) }}</ion-label>\n              </ion-item>\n            </ion-list>\n          </div>\n        </ion-accordion>\n      </ion-accordion-group>\n    </div>\n  `,\n  styleUrls: ['./links-accordion.component.scss'],\n})\nexport class LinksAccordionComponent {\n  private i18n = inject(I18nService);\n\n  /**\n   * Accordion configuration.\n   */\n  @Input() props: LinksAccordionMetadata = {\n    sections: [],\n    backgroundColor: 'var(--ion-background-color)',\n    closeMenuOnClick: true,\n    closeAccordionOnClick: true,\n  };\n\n  /**\n   * Emits the route when a link is clicked.\n   */\n  @Output() navigate = new EventEmitter<string[]>();\n\n  @ViewChild('accordionGroup') accordionGroup!: IonAccordionGroup;\n\n  constructor(\n    private router: Router,\n    private menuCtrl: MenuController\n  ) {}\n\n  /**\n   * Gets the section title, supporting i18n via titleKey.\n   */\n  getSectionTitle(section: LinksAccordionSection): string {\n    if (section.titleKey && this.props.i18nNamespace) {\n      return this.i18n.t(section.titleKey, this.props.i18nNamespace);\n    }\n    return section.title || '';\n  }\n\n  /**\n   * Gets the link text, supporting i18n via textKey.\n   */\n  getLinkText(link: LinksAccordionLink): string {\n    if (link.textKey && this.props.i18nNamespace) {\n      return this.i18n.t(link.textKey, this.props.i18nNamespace);\n    }\n    return link.text || '';\n  }\n\n  /**\n   * Handles link click - closes accordion/menu and navigates.\n   */\n  async onLinkClick(route: string[], event?: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    // Close accordion before navigating\n    if (this.props.closeAccordionOnClick !== false && this.accordionGroup) {\n      this.accordionGroup.value = undefined;\n    }\n\n    // Close menu if enabled\n    if (this.props.closeMenuOnClick !== false) {\n      await this.menuCtrl.close();\n    }\n\n    // Emit event\n    this.navigate.emit(route);\n\n    // Navigate\n    this.router.navigate(route);\n  }\n}\n"]}
185
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"links-accordion.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/links-accordion/links-accordion.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAE1F,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAkB,MAAM,2BAA2B,CAAC;AACxH,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAC;;;;;AAGrD;;;;;;;;;;;;;;;;;;;;;;GAsBG;AA8CH,MAAM,OAAO,uBAAuB;IAoBlC,YACU,MAAc,EACd,QAAwB;QADxB,WAAM,GAAN,MAAM,CAAQ;QACd,aAAQ,GAAR,QAAQ,CAAgB;QArB1B,SAAI,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAEnC;;WAEG;QACM,UAAK,GAA2B;YACvC,QAAQ,EAAE,EAAE;YACZ,eAAe,EAAE,6BAA6B;YAC9C,gBAAgB,EAAE,IAAI;YACtB,qBAAqB,EAAE,IAAI;SAC5B,CAAC;QAEF;;WAEG;QACO,aAAQ,GAAG,IAAI,YAAY,EAAY,CAAC;IAO/C,CAAC;IAEJ;;OAEG;IACH,eAAe,CAAC,OAA8B,EAAE,KAAa;QAC3D,OAAO,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,KAAK,IAAI,WAAW,KAAK,EAAE,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,OAA8B;QAC5C,IAAI,OAAO,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;YACjD,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QACjE,CAAC;QACD,OAAO,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC;IAC7B,CAAC;IAED;;OAEG;IACH,WAAW,CAAC,IAAwB;QAClC,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;YAC7C,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QAC7D,CAAC;QACD,OAAO,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC;IACzB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CAAC,KAAe,EAAE,KAAa;QAC9C,IAAI,KAAK,EAAE,CAAC;YACV,KAAK,CAAC,eAAe,EAAE,CAAC;QAC1B,CAAC;QAED,oCAAoC;QACpC,IAAI,IAAI,CAAC,KAAK,CAAC,qBAAqB,KAAK,KAAK,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACtE,IAAI,CAAC,cAAc,CAAC,KAAK,GAAG,SAAS,CAAC;QACxC,CAAC;QAED,wBAAwB;QACxB,IAAI,IAAI,CAAC,KAAK,CAAC,gBAAgB,KAAK,KAAK,EAAE,CAAC;YAC1C,MAAM,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;QAC9B,CAAC;QAED,aAAa;QACb,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAE1B,WAAW;QACX,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC9B,CAAC;+GA3EU,uBAAuB;mGAAvB,uBAAuB,gQAzCxB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsCT,gTAvCS,YAAY,qVAAE,iBAAiB,qJAAE,YAAY,6IAAE,OAAO,0NAAE,QAAQ,6FAAE,OAAO;;4FA0CxE,uBAAuB;kBA7CnC,SAAS;+BACE,qBAAqB,cACnB,IAAI,WACP,CAAC,YAAY,EAAE,iBAAiB,EAAE,YAAY,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,CAAC,YAC1E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsCT;wGASQ,KAAK;sBAAb,KAAK;gBAUI,QAAQ;sBAAjB,MAAM;gBAEsB,cAAc;sBAA1C,SAAS;uBAAC,gBAAgB","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output, ViewChild } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { IonAccordion, IonAccordionGroup, IonItem, IonLabel, IonList, MenuController } from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport { LinksAccordionLink, LinksAccordionMetadata, LinksAccordionSection } from './types';\n\n/**\n * val-links-accordion\n *\n * A hierarchical navigation component using Ionic accordions.\n * Supports sections with optional sub-links and automatic menu closing.\n *\n * @example\n * <val-links-accordion\n *   [props]=\"{\n *     sections: [\n *       { title: 'Home', route: ['/'] },\n *       { title: 'Products', links: [\n *         { text: 'Catalog', route: ['/products'] },\n *         { text: 'New Arrivals', route: ['/products', 'new'] }\n *       ]}\n *     ]\n *   }\"\n *   (navigate)=\"onNavigate($event)\"\n * ></val-links-accordion>\n *\n * @input props - Accordion configuration\n * @output navigate - Emits route when a link is clicked\n */\n@Component({\n  selector: 'val-links-accordion',\n  standalone: true,\n  imports: [CommonModule, IonAccordionGroup, IonAccordion, IonItem, IonLabel, IonList],\n  template: `\n    <div class=\"links-accordion\">\n      <ion-accordion-group #accordionGroup>\n        <ion-accordion\n          *ngFor=\"let section of props.sections; let i = index\"\n          [value]=\"getSectionValue(section, i)\"\n          [toggleIcon]=\"section.links?.length ? 'chevron-down-outline' : ''\"\n          [readonly]=\"!section.links?.length\"\n        >\n          <ion-item\n            slot=\"header\"\n            color=\"light\"\n            [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n            (click)=\"section.route ? onLinkClick(section.route, $event) : null\"\n            [button]=\"!!section.route\"\n            lines=\"none\"\n          >\n            <ion-label class=\"accordion-label\">{{ getSectionTitle(section) }}</ion-label>\n          </ion-item>\n          <div\n            slot=\"content\"\n            [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n            *ngIf=\"section.links?.length\"\n          >\n            <ion-list lines=\"none\">\n              <ion-item\n                *ngFor=\"let link of section.links\"\n                [ngStyle]=\"{ background: props.backgroundColor || 'var(--ion-background-color)' }\"\n                (click)=\"onLinkClick(link.route, $event)\"\n                button\n              >\n                <ion-label class=\"accordion-item-label\">{{ getLinkText(link) }}</ion-label>\n              </ion-item>\n            </ion-list>\n          </div>\n        </ion-accordion>\n      </ion-accordion-group>\n    </div>\n  `,\n  styleUrls: ['./links-accordion.component.scss'],\n})\nexport class LinksAccordionComponent {\n  private i18n = inject(I18nService);\n\n  /**\n   * Accordion configuration.\n   */\n  @Input() props: LinksAccordionMetadata = {\n    sections: [],\n    backgroundColor: 'var(--ion-background-color)',\n    closeMenuOnClick: true,\n    closeAccordionOnClick: true,\n  };\n\n  /**\n   * Emits the route when a link is clicked.\n   */\n  @Output() navigate = new EventEmitter<string[]>();\n\n  @ViewChild('accordionGroup') accordionGroup!: IonAccordionGroup;\n\n  constructor(\n    private router: Router,\n    private menuCtrl: MenuController\n  ) {}\n\n  /**\n   * Gets a unique value for the accordion section.\n   */\n  getSectionValue(section: LinksAccordionSection, index: number): string {\n    return section.titleKey || section.title || `section-${index}`;\n  }\n\n  /**\n   * Gets the section title, supporting i18n via titleKey.\n   */\n  getSectionTitle(section: LinksAccordionSection): string {\n    if (section.titleKey && this.props.i18nNamespace) {\n      return this.i18n.t(section.titleKey, this.props.i18nNamespace);\n    }\n    return section.title || '';\n  }\n\n  /**\n   * Gets the link text, supporting i18n via textKey.\n   */\n  getLinkText(link: LinksAccordionLink): string {\n    if (link.textKey && this.props.i18nNamespace) {\n      return this.i18n.t(link.textKey, this.props.i18nNamespace);\n    }\n    return link.text || '';\n  }\n\n  /**\n   * Handles link click - closes accordion/menu and navigates.\n   */\n  async onLinkClick(route: string[], event?: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    // Close accordion before navigating\n    if (this.props.closeAccordionOnClick !== false && this.accordionGroup) {\n      this.accordionGroup.value = undefined;\n    }\n\n    // Close menu if enabled\n    if (this.props.closeMenuOnClick !== false) {\n      await this.menuCtrl.close();\n    }\n\n    // Emit event\n    this.navigate.emit(route);\n\n    // Navigate\n    this.router.navigate(route);\n  }\n}\n"]}
@@ -0,0 +1,180 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, Input, signal, computed } from '@angular/core';
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "@angular/common";
5
+ /**
6
+ * val-rotating-text
7
+ *
8
+ * A component that rotates through an array of text messages with smooth animations.
9
+ * Features entrance/exit animations with fade, blur, and scale effects.
10
+ *
11
+ * @example
12
+ * <val-rotating-text
13
+ * [props]="{
14
+ * messages: [
15
+ * { aboveTitle: 'Welcome', title: 'Hello World' },
16
+ * { aboveTitle: 'Tip:', title: 'Stay curious' }
17
+ * ],
18
+ * interval: 4000,
19
+ * showDots: true
20
+ * }"
21
+ * ></val-rotating-text>
22
+ *
23
+ * @input props - Configuration for the rotating text component
24
+ */
25
+ export class RotatingTextComponent {
26
+ constructor() {
27
+ this.intervalId = null;
28
+ /**
29
+ * Component configuration.
30
+ */
31
+ this.props = {
32
+ messages: [],
33
+ interval: 4000,
34
+ showDots: true,
35
+ aboveTitleColor: 'medium',
36
+ titleColor: 'dark',
37
+ };
38
+ // Animation state
39
+ this.currentIndex = signal(0);
40
+ this.isEntering = signal(true);
41
+ this.isExiting = signal(false);
42
+ // Current message computed from index
43
+ this.currentMessage = computed(() => this.props.messages[this.currentIndex()]);
44
+ }
45
+ // Color values
46
+ get aboveTitleColorValue() {
47
+ const color = this.props.aboveTitleColor || 'medium';
48
+ return color.startsWith('--') || color.startsWith('#') || color.startsWith('rgb')
49
+ ? color
50
+ : `var(--ion-color-${color})`;
51
+ }
52
+ get titleColorValue() {
53
+ const color = this.props.titleColor || 'dark';
54
+ return color.startsWith('--') || color.startsWith('#') || color.startsWith('rgb')
55
+ ? color
56
+ : `var(--ion-color-${color})`;
57
+ }
58
+ ngOnInit() {
59
+ if (this.props.messages.length > 1) {
60
+ this.startRotation();
61
+ }
62
+ }
63
+ ngOnDestroy() {
64
+ this.stopRotation();
65
+ }
66
+ startRotation() {
67
+ const interval = this.props.interval ?? 4000;
68
+ this.intervalId = setInterval(() => {
69
+ this.rotateToNext();
70
+ }, interval);
71
+ }
72
+ stopRotation() {
73
+ if (this.intervalId) {
74
+ clearInterval(this.intervalId);
75
+ this.intervalId = null;
76
+ }
77
+ }
78
+ rotateToNext() {
79
+ // Start exit animation
80
+ this.isEntering.set(false);
81
+ this.isExiting.set(true);
82
+ // After exit animation, change text and start enter animation
83
+ setTimeout(() => {
84
+ this.currentIndex.update((i) => (i + 1) % this.props.messages.length);
85
+ this.isExiting.set(false);
86
+ this.isEntering.set(true);
87
+ }, 400); // Match exit animation duration
88
+ }
89
+ /**
90
+ * Navigate to a specific message by index.
91
+ */
92
+ goToMessage(index) {
93
+ if (index === this.currentIndex())
94
+ return;
95
+ this.stopRotation();
96
+ this.isEntering.set(false);
97
+ this.isExiting.set(true);
98
+ setTimeout(() => {
99
+ this.currentIndex.set(index);
100
+ this.isExiting.set(false);
101
+ this.isEntering.set(true);
102
+ if (this.props.messages.length > 1) {
103
+ this.startRotation();
104
+ }
105
+ }, 400);
106
+ }
107
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RotatingTextComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
108
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: RotatingTextComponent, isStandalone: true, selector: "val-rotating-text", inputs: { props: "props" }, ngImport: i0, template: `
109
+ <div
110
+ class="rotating-banner"
111
+ [ngStyle]="{ background: props.backgroundColor || 'transparent' }"
112
+ >
113
+ <div
114
+ class="rotating-text"
115
+ [class.text-enter]="isEntering()"
116
+ [class.text-exit]="isExiting()"
117
+ >
118
+ <div
119
+ class="above-title"
120
+ [ngStyle]="{ color: aboveTitleColorValue }"
121
+ >
122
+ {{ currentMessage()?.aboveTitle }}
123
+ </div>
124
+ <div
125
+ class="title"
126
+ [ngStyle]="{ color: titleColorValue }"
127
+ >
128
+ {{ currentMessage()?.title }}
129
+ </div>
130
+ </div>
131
+ <div class="progress-dots" *ngIf="props.showDots !== false">
132
+ <div
133
+ *ngFor="let message of props.messages; let i = index"
134
+ class="progress-dot"
135
+ [class.active]="i === currentIndex()"
136
+ (click)="goToMessage(i)"
137
+ ></div>
138
+ </div>
139
+ </div>
140
+ `, isInline: true, styles: [".rotating-banner{text-align:center;padding:2rem 1rem;min-height:120px;display:flex;flex-direction:column;justify-content:center;align-items:center}.rotating-text{position:relative;overflow:hidden}.rotating-text .above-title{font-size:1.125rem;margin-bottom:.75rem;font-family:monospace}.rotating-text .title{font-size:2rem;font-weight:900}.text-enter{animation:textEnter .6s ease-out forwards}.text-exit{animation:textExit .4s ease-in forwards}@keyframes textEnter{0%{opacity:0;transform:translateY(20px) scale(.95);filter:blur(4px)}to{opacity:1;transform:translateY(0) scale(1);filter:blur(0)}}@keyframes textExit{0%{opacity:1;transform:translateY(0) scale(1);filter:blur(0)}to{opacity:0;transform:translateY(-20px) scale(.95);filter:blur(4px)}}.progress-dots{display:flex;gap:8px;margin-top:1.5rem}.progress-dot{width:8px;height:8px;border-radius:50%;background:var(--ion-color-medium-tint);transition:all .3s ease;cursor:pointer}.progress-dot.active{background:var(--ion-color-dark);transform:scale(1.3)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); }
141
+ }
142
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: RotatingTextComponent, decorators: [{
143
+ type: Component,
144
+ args: [{ selector: 'val-rotating-text', standalone: true, imports: [CommonModule], template: `
145
+ <div
146
+ class="rotating-banner"
147
+ [ngStyle]="{ background: props.backgroundColor || 'transparent' }"
148
+ >
149
+ <div
150
+ class="rotating-text"
151
+ [class.text-enter]="isEntering()"
152
+ [class.text-exit]="isExiting()"
153
+ >
154
+ <div
155
+ class="above-title"
156
+ [ngStyle]="{ color: aboveTitleColorValue }"
157
+ >
158
+ {{ currentMessage()?.aboveTitle }}
159
+ </div>
160
+ <div
161
+ class="title"
162
+ [ngStyle]="{ color: titleColorValue }"
163
+ >
164
+ {{ currentMessage()?.title }}
165
+ </div>
166
+ </div>
167
+ <div class="progress-dots" *ngIf="props.showDots !== false">
168
+ <div
169
+ *ngFor="let message of props.messages; let i = index"
170
+ class="progress-dot"
171
+ [class.active]="i === currentIndex()"
172
+ (click)="goToMessage(i)"
173
+ ></div>
174
+ </div>
175
+ </div>
176
+ `, styles: [".rotating-banner{text-align:center;padding:2rem 1rem;min-height:120px;display:flex;flex-direction:column;justify-content:center;align-items:center}.rotating-text{position:relative;overflow:hidden}.rotating-text .above-title{font-size:1.125rem;margin-bottom:.75rem;font-family:monospace}.rotating-text .title{font-size:2rem;font-weight:900}.text-enter{animation:textEnter .6s ease-out forwards}.text-exit{animation:textExit .4s ease-in forwards}@keyframes textEnter{0%{opacity:0;transform:translateY(20px) scale(.95);filter:blur(4px)}to{opacity:1;transform:translateY(0) scale(1);filter:blur(0)}}@keyframes textExit{0%{opacity:1;transform:translateY(0) scale(1);filter:blur(0)}to{opacity:0;transform:translateY(-20px) scale(.95);filter:blur(4px)}}.progress-dots{display:flex;gap:8px;margin-top:1.5rem}.progress-dot{width:8px;height:8px;border-radius:50%;background:var(--ion-color-medium-tint);transition:all .3s ease;cursor:pointer}.progress-dot.active{background:var(--ion-color-dark);transform:scale(1.3)}\n"] }]
177
+ }], propDecorators: { props: [{
178
+ type: Input
179
+ }] } });
180
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rotating-text.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/rotating-text/rotating-text.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, OnDestroy, OnInit, signal, computed } from '@angular/core';\nimport { RotatingTextMetadata, RotatingTextMessage } from './types';\n\n/**\n * val-rotating-text\n *\n * A component that rotates through an array of text messages with smooth animations.\n * Features entrance/exit animations with fade, blur, and scale effects.\n *\n * @example\n * <val-rotating-text\n *   [props]=\"{\n *     messages: [\n *       { aboveTitle: 'Welcome', title: 'Hello World' },\n *       { aboveTitle: 'Tip:', title: 'Stay curious' }\n *     ],\n *     interval: 4000,\n *     showDots: true\n *   }\"\n * ></val-rotating-text>\n *\n * @input props - Configuration for the rotating text component\n */\n@Component({\n  selector: 'val-rotating-text',\n  standalone: true,\n  imports: [CommonModule],\n  styles: [\n    `\n      .rotating-banner {\n        text-align: center;\n        padding: 2rem 1rem;\n        min-height: 120px;\n        display: flex;\n        flex-direction: column;\n        justify-content: center;\n        align-items: center;\n      }\n\n      .rotating-text {\n        position: relative;\n        overflow: hidden;\n      }\n\n      .rotating-text .above-title {\n        font-size: 1.125rem;\n        margin-bottom: 0.75rem;\n        font-family: monospace;\n      }\n\n      .rotating-text .title {\n        font-size: 2rem;\n        font-weight: 900;\n      }\n\n      .text-enter {\n        animation: textEnter 0.6s ease-out forwards;\n      }\n\n      .text-exit {\n        animation: textExit 0.4s ease-in forwards;\n      }\n\n      @keyframes textEnter {\n        0% {\n          opacity: 0;\n          transform: translateY(20px) scale(0.95);\n          filter: blur(4px);\n        }\n        100% {\n          opacity: 1;\n          transform: translateY(0) scale(1);\n          filter: blur(0);\n        }\n      }\n\n      @keyframes textExit {\n        0% {\n          opacity: 1;\n          transform: translateY(0) scale(1);\n          filter: blur(0);\n        }\n        100% {\n          opacity: 0;\n          transform: translateY(-20px) scale(0.95);\n          filter: blur(4px);\n        }\n      }\n\n      .progress-dots {\n        display: flex;\n        gap: 8px;\n        margin-top: 1.5rem;\n      }\n\n      .progress-dot {\n        width: 8px;\n        height: 8px;\n        border-radius: 50%;\n        background: var(--ion-color-medium-tint);\n        transition: all 0.3s ease;\n        cursor: pointer;\n      }\n\n      .progress-dot.active {\n        background: var(--ion-color-dark);\n        transform: scale(1.3);\n      }\n    `,\n  ],\n  template: `\n    <div\n      class=\"rotating-banner\"\n      [ngStyle]=\"{ background: props.backgroundColor || 'transparent' }\"\n    >\n      <div\n        class=\"rotating-text\"\n        [class.text-enter]=\"isEntering()\"\n        [class.text-exit]=\"isExiting()\"\n      >\n        <div\n          class=\"above-title\"\n          [ngStyle]=\"{ color: aboveTitleColorValue }\"\n        >\n          {{ currentMessage()?.aboveTitle }}\n        </div>\n        <div\n          class=\"title\"\n          [ngStyle]=\"{ color: titleColorValue }\"\n        >\n          {{ currentMessage()?.title }}\n        </div>\n      </div>\n      <div class=\"progress-dots\" *ngIf=\"props.showDots !== false\">\n        <div\n          *ngFor=\"let message of props.messages; let i = index\"\n          class=\"progress-dot\"\n          [class.active]=\"i === currentIndex()\"\n          (click)=\"goToMessage(i)\"\n        ></div>\n      </div>\n    </div>\n  `,\n})\nexport class RotatingTextComponent implements OnInit, OnDestroy {\n  private intervalId: ReturnType<typeof setInterval> | null = null;\n\n  /**\n   * Component configuration.\n   */\n  @Input() props: RotatingTextMetadata = {\n    messages: [],\n    interval: 4000,\n    showDots: true,\n    aboveTitleColor: 'medium',\n    titleColor: 'dark',\n  };\n\n  // Animation state\n  currentIndex = signal(0);\n  isEntering = signal(true);\n  isExiting = signal(false);\n\n  // Current message computed from index\n  currentMessage = computed(() => this.props.messages[this.currentIndex()]);\n\n  // Color values\n  get aboveTitleColorValue(): string {\n    const color = this.props.aboveTitleColor || 'medium';\n    return color.startsWith('--') || color.startsWith('#') || color.startsWith('rgb')\n      ? color\n      : `var(--ion-color-${color})`;\n  }\n\n  get titleColorValue(): string {\n    const color = this.props.titleColor || 'dark';\n    return color.startsWith('--') || color.startsWith('#') || color.startsWith('rgb')\n      ? color\n      : `var(--ion-color-${color})`;\n  }\n\n  ngOnInit(): void {\n    if (this.props.messages.length > 1) {\n      this.startRotation();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.stopRotation();\n  }\n\n  private startRotation(): void {\n    const interval = this.props.interval ?? 4000;\n    this.intervalId = setInterval(() => {\n      this.rotateToNext();\n    }, interval);\n  }\n\n  private stopRotation(): void {\n    if (this.intervalId) {\n      clearInterval(this.intervalId);\n      this.intervalId = null;\n    }\n  }\n\n  private rotateToNext(): void {\n    // Start exit animation\n    this.isEntering.set(false);\n    this.isExiting.set(true);\n\n    // After exit animation, change text and start enter animation\n    setTimeout(() => {\n      this.currentIndex.update((i) => (i + 1) % this.props.messages.length);\n      this.isExiting.set(false);\n      this.isEntering.set(true);\n    }, 400); // Match exit animation duration\n  }\n\n  /**\n   * Navigate to a specific message by index.\n   */\n  goToMessage(index: number): void {\n    if (index === this.currentIndex()) return;\n\n    this.stopRotation();\n    this.isEntering.set(false);\n    this.isExiting.set(true);\n\n    setTimeout(() => {\n      this.currentIndex.set(index);\n      this.isExiting.set(false);\n      this.isEntering.set(true);\n      if (this.props.messages.length > 1) {\n        this.startRotation();\n      }\n    }, 400);\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL3JvdGF0aW5nLXRleHQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQSBzaW5nbGUgbWVzc2FnZSB0byBkaXNwbGF5IGluIHRoZSByb3RhdGluZyB0ZXh0IGNvbXBvbmVudC5cbiAqXG4gKiBAcHJvcGVydHkgYWJvdmVUaXRsZSAtIFNtYWxsIHRleHQgZGlzcGxheWVkIGFib3ZlIHRoZSBtYWluIHRpdGxlLlxuICogQHByb3BlcnR5IHRpdGxlIC0gVGhlIG1haW4gdGl0bGUgdGV4dC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSb3RhdGluZ1RleHRNZXNzYWdlIHtcbiAgYWJvdmVUaXRsZTogc3RyaW5nO1xuICB0aXRsZTogc3RyaW5nO1xufVxuXG4vKipcbiAqIFByb3BzIGZvciB2YWwtcm90YXRpbmctdGV4dCBjb21wb25lbnQuXG4gKlxuICogQHByb3BlcnR5IG1lc3NhZ2VzIC0gQXJyYXkgb2YgbWVzc2FnZXMgdG8gcm90YXRlIHRocm91Z2guXG4gKiBAcHJvcGVydHkgaW50ZXJ2YWwgLSBUaW1lIGluIG1pbGxpc2Vjb25kcyBiZXR3ZWVuIHJvdGF0aW9ucyAoZGVmYXVsdDogNDAwMCkuXG4gKiBAcHJvcGVydHkgc2hvd0RvdHMgLSBXaGV0aGVyIHRvIHNob3cgbmF2aWdhdGlvbiBkb3RzIChkZWZhdWx0OiB0cnVlKS5cbiAqIEBwcm9wZXJ0eSBhYm92ZVRpdGxlQ29sb3IgLSBDb2xvciBmb3IgYWJvdmUgdGl0bGUgdGV4dCAoZGVmYXVsdDogJ21lZGl1bScpLlxuICogQHByb3BlcnR5IHRpdGxlQ29sb3IgLSBDb2xvciBmb3IgbWFpbiB0aXRsZSB0ZXh0IChkZWZhdWx0OiAnZGFyaycpLlxuICogQHByb3BlcnR5IGJhY2tncm91bmRDb2xvciAtIEJhY2tncm91bmQgY29sb3IgZm9yIHRoZSBzZWN0aW9uLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJvdGF0aW5nVGV4dE1ldGFkYXRhIHtcbiAgbWVzc2FnZXM6IFJvdGF0aW5nVGV4dE1lc3NhZ2VbXTtcbiAgaW50ZXJ2YWw/OiBudW1iZXI7XG4gIHNob3dEb3RzPzogYm9vbGVhbjtcbiAgYWJvdmVUaXRsZUNvbG9yPzogc3RyaW5nO1xuICB0aXRsZUNvbG9yPzogc3RyaW5nO1xuICBiYWNrZ3JvdW5kQ29sb3I/OiBzdHJpbmc7XG59XG4iXX0=