valtech-components 2.0.590 → 2.0.592
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.
- package/esm2022/lib/components/organisms/rotating-text/rotating-text.component.mjs +180 -0
- package/esm2022/lib/components/organisms/rotating-text/types.mjs +2 -0
- package/esm2022/lib/components/organisms/terminal-404/terminal-404.component.mjs +224 -0
- package/esm2022/lib/components/organisms/terminal-404/types.mjs +2 -0
- package/esm2022/lib/services/auth/auth.service.mjs +4 -4
- package/esm2022/lib/services/auth/index.mjs +1 -3
- package/esm2022/lib/services/auth/types.mjs +1 -1
- package/esm2022/public-api.mjs +5 -1
- package/fesm2022/valtech-components.mjs +398 -219
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/organisms/rotating-text/rotating-text.component.d.ts +47 -0
- package/lib/components/organisms/rotating-text/types.d.ts +28 -0
- package/lib/components/organisms/terminal-404/terminal-404.component.d.ts +36 -0
- package/lib/components/organisms/terminal-404/types.d.ts +22 -0
- package/lib/services/auth/auth.service.d.ts +3 -3
- package/lib/services/auth/index.d.ts +0 -1
- package/lib/services/auth/types.d.ts +0 -52
- package/package.json +1 -7
- package/public-api.d.ts +4 -0
- package/esm2022/lib/services/auth/google-auth.service.mjs +0 -221
- package/lib/services/auth/google-auth.service.d.ts +0 -102
|
@@ -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":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,KAAK,EAAqB,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;;;AAGtF;;;;;;;;;;;;;;;;;;;GAmBG;AA0HH,MAAM,OAAO,qBAAqB;IAzHlC;QA0HU,eAAU,GAA0C,IAAI,CAAC;QAEjE;;WAEG;QACM,UAAK,GAAyB;YACrC,QAAQ,EAAE,EAAE;YACZ,QAAQ,EAAE,IAAI;YACd,QAAQ,EAAE,IAAI;YACd,eAAe,EAAE,QAAQ;YACzB,UAAU,EAAE,MAAM;SACnB,CAAC;QAEF,kBAAkB;QAClB,iBAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QACzB,eAAU,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;QAC1B,cAAS,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAE1B,sCAAsC;QACtC,mBAAc,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC;KAyE3E;IAvEC,eAAe;IACf,IAAI,oBAAoB;QACtB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,IAAI,QAAQ,CAAC;QACrD,OAAO,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC;YAC/E,CAAC,CAAC,KAAK;YACP,CAAC,CAAC,mBAAmB,KAAK,GAAG,CAAC;IAClC,CAAC;IAED,IAAI,eAAe;QACjB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,IAAI,MAAM,CAAC;QAC9C,OAAO,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC;YAC/E,CAAC,CAAC,KAAK;YACP,CAAC,CAAC,mBAAmB,KAAK,GAAG,CAAC;IAClC,CAAC;IAED,QAAQ;QACN,IAAI,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACnC,IAAI,CAAC,aAAa,EAAE,CAAC;QACvB,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,YAAY,EAAE,CAAC;IACtB,CAAC;IAEO,aAAa;QACnB,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,IAAI,IAAI,CAAC;QAC7C,IAAI,CAAC,UAAU,GAAG,WAAW,CAAC,GAAG,EAAE;YACjC,IAAI,CAAC,YAAY,EAAE,CAAC;QACtB,CAAC,EAAE,QAAQ,CAAC,CAAC;IACf,CAAC;IAEO,YAAY;QAClB,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,aAAa,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;YAC/B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACzB,CAAC;IACH,CAAC;IAEO,YAAY;QAClB,uBAAuB;QACvB,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC3B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAEzB,8DAA8D;QAC9D,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YACtE,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAC1B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC5B,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC,gCAAgC;IAC3C,CAAC;IAED;;OAEG;IACH,WAAW,CAAC,KAAa;QACvB,IAAI,KAAK,KAAK,IAAI,CAAC,YAAY,EAAE;YAAE,OAAO;QAE1C,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC3B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAEzB,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAC7B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAC1B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YAC1B,IAAI,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACnC,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,CAAC;QACH,CAAC,EAAE,GAAG,CAAC,CAAC;IACV,CAAC;+GA5FU,qBAAqB;mGAArB,qBAAqB,yGAlCtB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCT,0jCApHS,YAAY;;4FAsHX,qBAAqB;kBAzHjC,SAAS;+BACE,mBAAmB,cACjB,IAAI,WACP,CAAC,YAAY,CAAC,YAoFb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCT;8BAQQ,KAAK;sBAAb,KAAK","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=
|
|
@@ -0,0 +1,224 @@
|
|
|
1
|
+
import { Component, Input, signal } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { RouterLink } from '@angular/router';
|
|
4
|
+
import { IonButton } from '@ionic/angular/standalone';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/common";
|
|
7
|
+
/**
|
|
8
|
+
* val-terminal-404
|
|
9
|
+
*
|
|
10
|
+
* A fun, terminal-styled 404 page component with typing animations.
|
|
11
|
+
*
|
|
12
|
+
* Features:
|
|
13
|
+
* - Terminal window aesthetic with colored dots
|
|
14
|
+
* - Typing animation effect with sequential line reveals
|
|
15
|
+
* - Blinking cursor
|
|
16
|
+
* - Customizable colors to match brand
|
|
17
|
+
* - Shows the attempted path
|
|
18
|
+
*
|
|
19
|
+
* @example
|
|
20
|
+
* <val-terminal-404
|
|
21
|
+
* [props]="{
|
|
22
|
+
* primaryColor: '#4a1d96',
|
|
23
|
+
* homeRoute: '/',
|
|
24
|
+
* homeButtonText: 'cd /home'
|
|
25
|
+
* }"
|
|
26
|
+
* ></val-terminal-404>
|
|
27
|
+
*/
|
|
28
|
+
export class Terminal404Component {
|
|
29
|
+
constructor() {
|
|
30
|
+
this.props = {};
|
|
31
|
+
this.requestedPath = signal('page-not-found');
|
|
32
|
+
}
|
|
33
|
+
get backgroundGradient() {
|
|
34
|
+
const bg = this.props.backgroundColor;
|
|
35
|
+
if (bg) {
|
|
36
|
+
return `linear-gradient(135deg, ${bg.start} 0%, ${bg.middle} 50%, ${bg.end} 100%)`;
|
|
37
|
+
}
|
|
38
|
+
return 'linear-gradient(135deg, #ede9fe 0%, #ddd6fe 50%, #c4b5fd 100%)';
|
|
39
|
+
}
|
|
40
|
+
ngOnInit() {
|
|
41
|
+
if (typeof window !== 'undefined') {
|
|
42
|
+
const path = window.location.pathname.slice(1) || 'unknown';
|
|
43
|
+
this.requestedPath.set(path);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Lighten a hex color for hover states
|
|
48
|
+
*/
|
|
49
|
+
lightenColor(hex) {
|
|
50
|
+
// Simple lighten by adding to RGB values
|
|
51
|
+
const num = parseInt(hex.replace('#', ''), 16);
|
|
52
|
+
const r = Math.min(255, (num >> 16) + 20);
|
|
53
|
+
const g = Math.min(255, ((num >> 8) & 0x00ff) + 20);
|
|
54
|
+
const b = Math.min(255, (num & 0x0000ff) + 20);
|
|
55
|
+
return `#${((r << 16) | (g << 8) | b).toString(16).padStart(6, '0')}`;
|
|
56
|
+
}
|
|
57
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: Terminal404Component, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
58
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: Terminal404Component, isStandalone: true, selector: "val-terminal-404", inputs: { props: "props" }, ngImport: i0, template: `
|
|
59
|
+
<div
|
|
60
|
+
class="terminal-container"
|
|
61
|
+
[ngStyle]="{
|
|
62
|
+
background: backgroundGradient
|
|
63
|
+
}"
|
|
64
|
+
>
|
|
65
|
+
<div
|
|
66
|
+
class="error-code"
|
|
67
|
+
[ngStyle]="{
|
|
68
|
+
color: props.primaryColor || '#4a1d96',
|
|
69
|
+
textShadow: '0 0 40px ' + (props.primaryColor || '#4a1d96') + '4d'
|
|
70
|
+
}"
|
|
71
|
+
>
|
|
72
|
+
404
|
|
73
|
+
</div>
|
|
74
|
+
|
|
75
|
+
<div
|
|
76
|
+
class="terminal"
|
|
77
|
+
[ngStyle]="{ background: props.terminalBackground || '#0d1117' }"
|
|
78
|
+
>
|
|
79
|
+
<div class="terminal-header">
|
|
80
|
+
<div class="terminal-dot dot-red"></div>
|
|
81
|
+
<div class="terminal-dot dot-yellow"></div>
|
|
82
|
+
<div class="terminal-dot dot-green"></div>
|
|
83
|
+
<span class="terminal-title">bash - 404</span>
|
|
84
|
+
</div>
|
|
85
|
+
|
|
86
|
+
<div class="terminal-body">
|
|
87
|
+
<div class="line">
|
|
88
|
+
<span class="prompt">user@app</span>
|
|
89
|
+
<span class="command">:~$ </span>
|
|
90
|
+
<span class="command">cd /{{ requestedPath() }}</span>
|
|
91
|
+
</div>
|
|
92
|
+
|
|
93
|
+
<div class="line">
|
|
94
|
+
<span class="error"
|
|
95
|
+
>bash: cd: /{{ requestedPath() }}: No existe el directorio</span
|
|
96
|
+
>
|
|
97
|
+
</div>
|
|
98
|
+
|
|
99
|
+
<div class="line">
|
|
100
|
+
<span class="prompt">user@app</span>
|
|
101
|
+
<span class="command">:~$ </span>
|
|
102
|
+
<span class="command">sudo find / -name "pagina"</span>
|
|
103
|
+
</div>
|
|
104
|
+
|
|
105
|
+
<div class="line">
|
|
106
|
+
<span class="info"
|
|
107
|
+
><span class="search-icon">🔍</span> Buscando...</span
|
|
108
|
+
>
|
|
109
|
+
</div>
|
|
110
|
+
|
|
111
|
+
<div class="line">
|
|
112
|
+
<span class="warning">find: No se encontraron resultados.</span>
|
|
113
|
+
</div>
|
|
114
|
+
|
|
115
|
+
<div class="line">
|
|
116
|
+
<span class="prompt">user@app</span>
|
|
117
|
+
<span class="command">:~$ </span>
|
|
118
|
+
<span class="cursor"></span>
|
|
119
|
+
</div>
|
|
120
|
+
</div>
|
|
121
|
+
</div>
|
|
122
|
+
|
|
123
|
+
<div class="action-container">
|
|
124
|
+
<ion-button
|
|
125
|
+
class="home-button"
|
|
126
|
+
[routerLink]="props.homeRoute || '/'"
|
|
127
|
+
expand="block"
|
|
128
|
+
[ngStyle]="{
|
|
129
|
+
'--background': props.primaryColor || '#4a1d96',
|
|
130
|
+
'--background-hover': lightenColor(props.primaryColor || '#4a1d96')
|
|
131
|
+
}"
|
|
132
|
+
>
|
|
133
|
+
{{ props.homeButtonText || 'cd /home' }}
|
|
134
|
+
</ion-button>
|
|
135
|
+
</div>
|
|
136
|
+
</div>
|
|
137
|
+
`, isInline: true, styles: [":host{display:block;width:100%}.terminal-container{min-height:70vh;display:flex;flex-direction:column;justify-content:center;align-items:center;padding:2rem;border-radius:16px}.terminal{border-radius:12px;padding:0;max-width:600px;width:100%;box-shadow:0 20px 60px #0000004d;overflow:hidden}.terminal-header{background:#161b22;padding:12px 16px;display:flex;align-items:center;gap:8px}.terminal-dot{width:12px;height:12px;border-radius:50%}.dot-red{background:#ff5f56}.dot-yellow{background:#ffbd2e}.dot-green{background:#27ca40}.terminal-title{flex:1;text-align:center;color:#8b949e;font-size:.85rem;font-family:monospace}.terminal-body{padding:24px;font-family:Fira Code,Monaco,Consolas,monospace;font-size:.95rem;line-height:1.8}.line{margin-bottom:8px;opacity:0;animation:fadeIn .3s ease forwards}.line:nth-child(1){animation-delay:.2s}.line:nth-child(2){animation-delay:.8s}.line:nth-child(3){animation-delay:1.4s}.line:nth-child(4){animation-delay:2s}.line:nth-child(5){animation-delay:2.6s}.line:nth-child(6){animation-delay:3.2s}@keyframes fadeIn{to{opacity:1}}.prompt{color:#7ee787}.command{color:#c9d1d9}.error{color:#f85149}.info{color:#58a6ff}.warning{color:#d29922}.cursor{display:inline-block;width:10px;height:18px;background:#7ee787;margin-left:4px;animation:blink 1s step-end infinite;vertical-align:text-bottom}@keyframes blink{0%,to{opacity:1}50%{opacity:0}}.error-code{font-size:6rem;font-weight:900;text-align:center;margin:.5rem 0 1rem;font-family:monospace}.action-container{margin-top:2rem;text-align:center;opacity:0;animation:fadeIn .5s ease forwards;animation-delay:3.8s}.home-button{--border-radius: 8px;font-family:monospace}.search-icon{animation:searching 2s ease-in-out infinite;display:inline-block}@keyframes searching{0%,to{transform:translate(0)}25%{transform:translate(5px)}75%{transform:translate(-5px)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }] }); }
|
|
138
|
+
}
|
|
139
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: Terminal404Component, decorators: [{
|
|
140
|
+
type: Component,
|
|
141
|
+
args: [{ selector: 'val-terminal-404', standalone: true, imports: [CommonModule, IonButton, RouterLink], template: `
|
|
142
|
+
<div
|
|
143
|
+
class="terminal-container"
|
|
144
|
+
[ngStyle]="{
|
|
145
|
+
background: backgroundGradient
|
|
146
|
+
}"
|
|
147
|
+
>
|
|
148
|
+
<div
|
|
149
|
+
class="error-code"
|
|
150
|
+
[ngStyle]="{
|
|
151
|
+
color: props.primaryColor || '#4a1d96',
|
|
152
|
+
textShadow: '0 0 40px ' + (props.primaryColor || '#4a1d96') + '4d'
|
|
153
|
+
}"
|
|
154
|
+
>
|
|
155
|
+
404
|
|
156
|
+
</div>
|
|
157
|
+
|
|
158
|
+
<div
|
|
159
|
+
class="terminal"
|
|
160
|
+
[ngStyle]="{ background: props.terminalBackground || '#0d1117' }"
|
|
161
|
+
>
|
|
162
|
+
<div class="terminal-header">
|
|
163
|
+
<div class="terminal-dot dot-red"></div>
|
|
164
|
+
<div class="terminal-dot dot-yellow"></div>
|
|
165
|
+
<div class="terminal-dot dot-green"></div>
|
|
166
|
+
<span class="terminal-title">bash - 404</span>
|
|
167
|
+
</div>
|
|
168
|
+
|
|
169
|
+
<div class="terminal-body">
|
|
170
|
+
<div class="line">
|
|
171
|
+
<span class="prompt">user@app</span>
|
|
172
|
+
<span class="command">:~$ </span>
|
|
173
|
+
<span class="command">cd /{{ requestedPath() }}</span>
|
|
174
|
+
</div>
|
|
175
|
+
|
|
176
|
+
<div class="line">
|
|
177
|
+
<span class="error"
|
|
178
|
+
>bash: cd: /{{ requestedPath() }}: No existe el directorio</span
|
|
179
|
+
>
|
|
180
|
+
</div>
|
|
181
|
+
|
|
182
|
+
<div class="line">
|
|
183
|
+
<span class="prompt">user@app</span>
|
|
184
|
+
<span class="command">:~$ </span>
|
|
185
|
+
<span class="command">sudo find / -name "pagina"</span>
|
|
186
|
+
</div>
|
|
187
|
+
|
|
188
|
+
<div class="line">
|
|
189
|
+
<span class="info"
|
|
190
|
+
><span class="search-icon">🔍</span> Buscando...</span
|
|
191
|
+
>
|
|
192
|
+
</div>
|
|
193
|
+
|
|
194
|
+
<div class="line">
|
|
195
|
+
<span class="warning">find: No se encontraron resultados.</span>
|
|
196
|
+
</div>
|
|
197
|
+
|
|
198
|
+
<div class="line">
|
|
199
|
+
<span class="prompt">user@app</span>
|
|
200
|
+
<span class="command">:~$ </span>
|
|
201
|
+
<span class="cursor"></span>
|
|
202
|
+
</div>
|
|
203
|
+
</div>
|
|
204
|
+
</div>
|
|
205
|
+
|
|
206
|
+
<div class="action-container">
|
|
207
|
+
<ion-button
|
|
208
|
+
class="home-button"
|
|
209
|
+
[routerLink]="props.homeRoute || '/'"
|
|
210
|
+
expand="block"
|
|
211
|
+
[ngStyle]="{
|
|
212
|
+
'--background': props.primaryColor || '#4a1d96',
|
|
213
|
+
'--background-hover': lightenColor(props.primaryColor || '#4a1d96')
|
|
214
|
+
}"
|
|
215
|
+
>
|
|
216
|
+
{{ props.homeButtonText || 'cd /home' }}
|
|
217
|
+
</ion-button>
|
|
218
|
+
</div>
|
|
219
|
+
</div>
|
|
220
|
+
`, styles: [":host{display:block;width:100%}.terminal-container{min-height:70vh;display:flex;flex-direction:column;justify-content:center;align-items:center;padding:2rem;border-radius:16px}.terminal{border-radius:12px;padding:0;max-width:600px;width:100%;box-shadow:0 20px 60px #0000004d;overflow:hidden}.terminal-header{background:#161b22;padding:12px 16px;display:flex;align-items:center;gap:8px}.terminal-dot{width:12px;height:12px;border-radius:50%}.dot-red{background:#ff5f56}.dot-yellow{background:#ffbd2e}.dot-green{background:#27ca40}.terminal-title{flex:1;text-align:center;color:#8b949e;font-size:.85rem;font-family:monospace}.terminal-body{padding:24px;font-family:Fira Code,Monaco,Consolas,monospace;font-size:.95rem;line-height:1.8}.line{margin-bottom:8px;opacity:0;animation:fadeIn .3s ease forwards}.line:nth-child(1){animation-delay:.2s}.line:nth-child(2){animation-delay:.8s}.line:nth-child(3){animation-delay:1.4s}.line:nth-child(4){animation-delay:2s}.line:nth-child(5){animation-delay:2.6s}.line:nth-child(6){animation-delay:3.2s}@keyframes fadeIn{to{opacity:1}}.prompt{color:#7ee787}.command{color:#c9d1d9}.error{color:#f85149}.info{color:#58a6ff}.warning{color:#d29922}.cursor{display:inline-block;width:10px;height:18px;background:#7ee787;margin-left:4px;animation:blink 1s step-end infinite;vertical-align:text-bottom}@keyframes blink{0%,to{opacity:1}50%{opacity:0}}.error-code{font-size:6rem;font-weight:900;text-align:center;margin:.5rem 0 1rem;font-family:monospace}.action-container{margin-top:2rem;text-align:center;opacity:0;animation:fadeIn .5s ease forwards;animation-delay:3.8s}.home-button{--border-radius: 8px;font-family:monospace}.search-icon{animation:searching 2s ease-in-out infinite;display:inline-block}@keyframes searching{0%,to{transform:translate(0)}25%{transform:translate(5px)}75%{transform:translate(-5px)}}\n"] }]
|
|
221
|
+
}], propDecorators: { props: [{
|
|
222
|
+
type: Input
|
|
223
|
+
}] } });
|
|
224
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"terminal-404.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/terminal-404/terminal-404.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AACjE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;;;AAGtD;;;;;;;;;;;;;;;;;;;;GAoBG;AAsQH,MAAM,OAAO,oBAAoB;IArQjC;QAsQW,UAAK,GAAwB,EAAE,CAAC;QAEzC,kBAAa,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;KA4B1C;IA1BC,IAAI,kBAAkB;QACpB,MAAM,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC;QACtC,IAAI,EAAE,EAAE,CAAC;YACP,OAAO,2BAA2B,EAAE,CAAC,KAAK,QAAQ,EAAE,CAAC,MAAM,SAAS,EAAE,CAAC,GAAG,QAAQ,CAAC;QACrF,CAAC;QACD,OAAO,gEAAgE,CAAC;IAC1E,CAAC;IAED,QAAQ;QACN,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE,CAAC;YAClC,MAAM,IAAI,GAAG,MAAM,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC;YAC5D,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC/B,CAAC;IACH,CAAC;IAED;;OAEG;IACH,YAAY,CAAC,GAAW;QACtB,yCAAyC;QACzC,MAAM,GAAG,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;QAC/C,MAAM,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,GAAG,IAAI,EAAE,CAAC,GAAG,EAAE,CAAC,CAAC;QAC1C,MAAM,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,GAAG,MAAM,CAAC,GAAG,EAAE,CAAC,CAAC;QACpD,MAAM,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,GAAG,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC;QAC/C,OAAO,IAAI,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;IACxE,CAAC;+GA9BU,oBAAoB;mGAApB,oBAAoB,wGAjFrB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+ET,63DAhQS,YAAY,oHAAE,SAAS,oPAAE,UAAU;;4FAkQlC,oBAAoB;kBArQhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP,CAAC,YAAY,EAAE,SAAS,EAAE,UAAU,CAAC,YAiLpC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+ET;8BAGQ,KAAK;sBAAb,KAAK","sourcesContent":["import { Component, Input, OnInit, signal } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { RouterLink } from '@angular/router';\nimport { IonButton } from '@ionic/angular/standalone';\nimport { Terminal404Metadata } from './types';\n\n/**\n * val-terminal-404\n *\n * A fun, terminal-styled 404 page component with typing animations.\n *\n * Features:\n * - Terminal window aesthetic with colored dots\n * - Typing animation effect with sequential line reveals\n * - Blinking cursor\n * - Customizable colors to match brand\n * - Shows the attempted path\n *\n * @example\n * <val-terminal-404\n *   [props]=\"{\n *     primaryColor: '#4a1d96',\n *     homeRoute: '/',\n *     homeButtonText: 'cd /home'\n *   }\"\n * ></val-terminal-404>\n */\n@Component({\n  selector: 'val-terminal-404',\n  standalone: true,\n  imports: [CommonModule, IonButton, RouterLink],\n  styles: [\n    `\n      :host {\n        display: block;\n        width: 100%;\n      }\n\n      .terminal-container {\n        min-height: 70vh;\n        display: flex;\n        flex-direction: column;\n        justify-content: center;\n        align-items: center;\n        padding: 2rem;\n        border-radius: 16px;\n      }\n\n      .terminal {\n        border-radius: 12px;\n        padding: 0;\n        max-width: 600px;\n        width: 100%;\n        box-shadow: 0 20px 60px rgba(0, 0, 0, 0.3);\n        overflow: hidden;\n      }\n\n      .terminal-header {\n        background: #161b22;\n        padding: 12px 16px;\n        display: flex;\n        align-items: center;\n        gap: 8px;\n      }\n\n      .terminal-dot {\n        width: 12px;\n        height: 12px;\n        border-radius: 50%;\n      }\n\n      .dot-red {\n        background: #ff5f56;\n      }\n      .dot-yellow {\n        background: #ffbd2e;\n      }\n      .dot-green {\n        background: #27ca40;\n      }\n\n      .terminal-title {\n        flex: 1;\n        text-align: center;\n        color: #8b949e;\n        font-size: 0.85rem;\n        font-family: monospace;\n      }\n\n      .terminal-body {\n        padding: 24px;\n        font-family: 'Fira Code', 'Monaco', 'Consolas', monospace;\n        font-size: 0.95rem;\n        line-height: 1.8;\n      }\n\n      .line {\n        margin-bottom: 8px;\n        opacity: 0;\n        animation: fadeIn 0.3s ease forwards;\n      }\n\n      .line:nth-child(1) {\n        animation-delay: 0.2s;\n      }\n      .line:nth-child(2) {\n        animation-delay: 0.8s;\n      }\n      .line:nth-child(3) {\n        animation-delay: 1.4s;\n      }\n      .line:nth-child(4) {\n        animation-delay: 2s;\n      }\n      .line:nth-child(5) {\n        animation-delay: 2.6s;\n      }\n      .line:nth-child(6) {\n        animation-delay: 3.2s;\n      }\n\n      @keyframes fadeIn {\n        to {\n          opacity: 1;\n        }\n      }\n\n      .prompt {\n        color: #7ee787;\n      }\n\n      .command {\n        color: #c9d1d9;\n      }\n\n      .error {\n        color: #f85149;\n      }\n\n      .info {\n        color: #58a6ff;\n      }\n\n      .warning {\n        color: #d29922;\n      }\n\n      .cursor {\n        display: inline-block;\n        width: 10px;\n        height: 18px;\n        background: #7ee787;\n        margin-left: 4px;\n        animation: blink 1s step-end infinite;\n        vertical-align: text-bottom;\n      }\n\n      @keyframes blink {\n        0%,\n        100% {\n          opacity: 1;\n        }\n        50% {\n          opacity: 0;\n        }\n      }\n\n      .error-code {\n        font-size: 6rem;\n        font-weight: 900;\n        text-align: center;\n        margin: 0.5rem 0 1rem;\n        font-family: monospace;\n      }\n\n      .action-container {\n        margin-top: 2rem;\n        text-align: center;\n        opacity: 0;\n        animation: fadeIn 0.5s ease forwards;\n        animation-delay: 3.8s;\n      }\n\n      .home-button {\n        --border-radius: 8px;\n        font-family: monospace;\n      }\n\n      .search-icon {\n        animation: searching 2s ease-in-out infinite;\n        display: inline-block;\n      }\n\n      @keyframes searching {\n        0%,\n        100% {\n          transform: translateX(0);\n        }\n        25% {\n          transform: translateX(5px);\n        }\n        75% {\n          transform: translateX(-5px);\n        }\n      }\n    `,\n  ],\n  template: `\n    <div\n      class=\"terminal-container\"\n      [ngStyle]=\"{\n        background: backgroundGradient\n      }\"\n    >\n      <div\n        class=\"error-code\"\n        [ngStyle]=\"{\n          color: props.primaryColor || '#4a1d96',\n          textShadow: '0 0 40px ' + (props.primaryColor || '#4a1d96') + '4d'\n        }\"\n      >\n        404\n      </div>\n\n      <div\n        class=\"terminal\"\n        [ngStyle]=\"{ background: props.terminalBackground || '#0d1117' }\"\n      >\n        <div class=\"terminal-header\">\n          <div class=\"terminal-dot dot-red\"></div>\n          <div class=\"terminal-dot dot-yellow\"></div>\n          <div class=\"terminal-dot dot-green\"></div>\n          <span class=\"terminal-title\">bash - 404</span>\n        </div>\n\n        <div class=\"terminal-body\">\n          <div class=\"line\">\n            <span class=\"prompt\">user&#64;app</span>\n            <span class=\"command\">:~$ </span>\n            <span class=\"command\">cd /{{ requestedPath() }}</span>\n          </div>\n\n          <div class=\"line\">\n            <span class=\"error\"\n              >bash: cd: /{{ requestedPath() }}: No existe el directorio</span\n            >\n          </div>\n\n          <div class=\"line\">\n            <span class=\"prompt\">user&#64;app</span>\n            <span class=\"command\">:~$ </span>\n            <span class=\"command\">sudo find / -name \"pagina\"</span>\n          </div>\n\n          <div class=\"line\">\n            <span class=\"info\"\n              ><span class=\"search-icon\">🔍</span> Buscando...</span\n            >\n          </div>\n\n          <div class=\"line\">\n            <span class=\"warning\">find: No se encontraron resultados.</span>\n          </div>\n\n          <div class=\"line\">\n            <span class=\"prompt\">user&#64;app</span>\n            <span class=\"command\">:~$ </span>\n            <span class=\"cursor\"></span>\n          </div>\n        </div>\n      </div>\n\n      <div class=\"action-container\">\n        <ion-button\n          class=\"home-button\"\n          [routerLink]=\"props.homeRoute || '/'\"\n          expand=\"block\"\n          [ngStyle]=\"{\n            '--background': props.primaryColor || '#4a1d96',\n            '--background-hover': lightenColor(props.primaryColor || '#4a1d96')\n          }\"\n        >\n          {{ props.homeButtonText || 'cd /home' }}\n        </ion-button>\n      </div>\n    </div>\n  `,\n})\nexport class Terminal404Component implements OnInit {\n  @Input() props: Terminal404Metadata = {};\n\n  requestedPath = signal('page-not-found');\n\n  get backgroundGradient(): string {\n    const bg = this.props.backgroundColor;\n    if (bg) {\n      return `linear-gradient(135deg, ${bg.start} 0%, ${bg.middle} 50%, ${bg.end} 100%)`;\n    }\n    return 'linear-gradient(135deg, #ede9fe 0%, #ddd6fe 50%, #c4b5fd 100%)';\n  }\n\n  ngOnInit(): void {\n    if (typeof window !== 'undefined') {\n      const path = window.location.pathname.slice(1) || 'unknown';\n      this.requestedPath.set(path);\n    }\n  }\n\n  /**\n   * Lighten a hex color for hover states\n   */\n  lightenColor(hex: string): string {\n    // Simple lighten by adding to RGB values\n    const num = parseInt(hex.replace('#', ''), 16);\n    const r = Math.min(255, (num >> 16) + 20);\n    const g = Math.min(255, ((num >> 8) & 0x00ff) + 20);\n    const b = Math.min(255, (num & 0x0000ff) + 20);\n    return `#${((r << 16) | (g << 8) | b).toString(16).padStart(6, '0')}`;\n  }\n}\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvb3JnYW5pc21zL3Rlcm1pbmFsLTQwNC90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb25maWd1cmF0aW9uIGZvciB0ZXJtaW5hbCA0MDQgcGFnZSBjb21wb25lbnQuXG4gKlxuICogQHByb3BlcnR5IGJhY2tncm91bmRDb2xvciAtIEJhY2tncm91bmQgZ3JhZGllbnQgY29sb3JzIChkZWZhdWx0OiBsYXZlbmRlciBwdXJwbGUpXG4gKiBAcHJvcGVydHkgcHJpbWFyeUNvbG9yIC0gTWFpbiBicmFuZCBjb2xvciBmb3IgNDA0IHRleHQgYW5kIGJ1dHRvbiAoZGVmYXVsdDogIzRhMWQ5NilcbiAqIEBwcm9wZXJ0eSB0ZXJtaW5hbEJhY2tncm91bmQgLSBUZXJtaW5hbCB3aW5kb3cgYmFja2dyb3VuZCAoZGVmYXVsdDogIzBkMTExNylcbiAqIEBwcm9wZXJ0eSBob21lUm91dGUgLSBSb3V0ZSBmb3IgdGhlIGhvbWUgYnV0dG9uIChkZWZhdWx0OiAnLycpXG4gKiBAcHJvcGVydHkgaG9tZUJ1dHRvblRleHQgLSBUZXh0IGZvciBob21lIGJ1dHRvbiAoZGVmYXVsdDogJ2NkIC9ob21lJylcbiAqIEBwcm9wZXJ0eSBzaG93TGFuZ3VhZ2VTZWxlY3RvciAtIFNob3cgbGFuZ3VhZ2Ugc2VsZWN0b3IgaW4gaGVhZGVyIChkZWZhdWx0OiB0cnVlKVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFRlcm1pbmFsNDA0TWV0YWRhdGEge1xuICBiYWNrZ3JvdW5kQ29sb3I/OiB7XG4gICAgc3RhcnQ6IHN0cmluZztcbiAgICBtaWRkbGU6IHN0cmluZztcbiAgICBlbmQ6IHN0cmluZztcbiAgfTtcbiAgcHJpbWFyeUNvbG9yPzogc3RyaW5nO1xuICB0ZXJtaW5hbEJhY2tncm91bmQ/OiBzdHJpbmc7XG4gIGhvbWVSb3V0ZT86IHN0cmluZztcbiAgaG9tZUJ1dHRvblRleHQ/OiBzdHJpbmc7XG4gIHNob3dMYW5ndWFnZVNlbGVjdG9yPzogYm9vbGVhbjtcbn1cbiJdfQ==
|