valtech-components 2.0.612 → 2.0.614
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/molecules/features-list/features-list.component.mjs +185 -0
- package/esm2022/lib/components/molecules/features-list/types.mjs +11 -0
- package/esm2022/lib/components/molecules/footer-links/footer-links.component.mjs +3 -7
- package/esm2022/lib/config/company-footer.config.mjs +106 -0
- package/esm2022/lib/config/index.mjs +5 -0
- package/esm2022/public-api.mjs +6 -1
- package/fesm2022/valtech-components.mjs +303 -7
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/features-list/features-list.component.d.ts +66 -0
- package/lib/components/molecules/features-list/types.d.ts +36 -0
- package/lib/config/company-footer.config.d.ts +97 -0
- package/lib/config/index.d.ts +4 -0
- package/package.json +1 -1
- package/public-api.d.ts +3 -0
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, computed, inject, input } from '@angular/core';
|
|
3
|
+
import { IonIcon } from '@ionic/angular/standalone';
|
|
4
|
+
import { I18nService } from '../../../services/i18n';
|
|
5
|
+
import { FEATURES_LIST_DEFAULTS } from './types';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
/**
|
|
8
|
+
* val-features-list
|
|
9
|
+
*
|
|
10
|
+
* A component to display a list of features with icons, titles, and descriptions.
|
|
11
|
+
* Supports i18n for all text content.
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```html
|
|
15
|
+
* <val-features-list
|
|
16
|
+
* [props]="{
|
|
17
|
+
* features: [
|
|
18
|
+
* { icon: 'flash-outline', titleKey: 'feature1Title', descriptionKey: 'feature1Desc' },
|
|
19
|
+
* { icon: 'color-palette-outline', titleKey: 'feature2Title', descriptionKey: 'feature2Desc' }
|
|
20
|
+
* ],
|
|
21
|
+
* i18nNamespace: 'Features',
|
|
22
|
+
* iconColor: 'primary',
|
|
23
|
+
* direction: 'column'
|
|
24
|
+
* }"
|
|
25
|
+
* />
|
|
26
|
+
* ```
|
|
27
|
+
*
|
|
28
|
+
* @example With custom SVG icons
|
|
29
|
+
* ```html
|
|
30
|
+
* <val-features-list
|
|
31
|
+
* [props]="{
|
|
32
|
+
* features: [
|
|
33
|
+
* {
|
|
34
|
+
* icon: 'custom',
|
|
35
|
+
* svgPath: 'M13 2L3 14h9l-1 8 10-12h-9l1-8z',
|
|
36
|
+
* titleKey: 'Fast & Secure',
|
|
37
|
+
* descriptionKey: 'Modern authentication with OAuth and MFA'
|
|
38
|
+
* }
|
|
39
|
+
* ]
|
|
40
|
+
* }"
|
|
41
|
+
* />
|
|
42
|
+
* ```
|
|
43
|
+
*/
|
|
44
|
+
export class FeaturesListComponent {
|
|
45
|
+
constructor() {
|
|
46
|
+
this.i18n = inject(I18nService);
|
|
47
|
+
/** Component configuration */
|
|
48
|
+
this.props = input.required();
|
|
49
|
+
/** Merged configuration with defaults */
|
|
50
|
+
this.config = computed(() => ({
|
|
51
|
+
...FEATURES_LIST_DEFAULTS,
|
|
52
|
+
...this.props(),
|
|
53
|
+
}));
|
|
54
|
+
/** Resolved icon color (handles Ionic colors and CSS colors) */
|
|
55
|
+
this.iconColorStyle = computed(() => {
|
|
56
|
+
const color = this.config().iconColor;
|
|
57
|
+
// Check if it's an Ionic color variable
|
|
58
|
+
if (color && !color.startsWith('#') && !color.startsWith('rgb')) {
|
|
59
|
+
return `var(--ion-color-${color})`;
|
|
60
|
+
}
|
|
61
|
+
return color;
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Get translated title or return direct text
|
|
66
|
+
*/
|
|
67
|
+
getTitle(feature) {
|
|
68
|
+
const namespace = this.props().i18nNamespace;
|
|
69
|
+
if (namespace) {
|
|
70
|
+
// Track language changes
|
|
71
|
+
this.i18n.lang();
|
|
72
|
+
return this.i18n.t(feature.titleKey, namespace);
|
|
73
|
+
}
|
|
74
|
+
return feature.titleKey;
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Get translated description or return direct text
|
|
78
|
+
*/
|
|
79
|
+
getDescription(feature) {
|
|
80
|
+
const namespace = this.props().i18nNamespace;
|
|
81
|
+
if (namespace) {
|
|
82
|
+
// Track language changes
|
|
83
|
+
this.i18n.lang();
|
|
84
|
+
return this.i18n.t(feature.descriptionKey, namespace);
|
|
85
|
+
}
|
|
86
|
+
return feature.descriptionKey;
|
|
87
|
+
}
|
|
88
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeaturesListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
89
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: FeaturesListComponent, isStandalone: true, selector: "val-features-list", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: true, transformFunction: null } }, ngImport: i0, template: `
|
|
90
|
+
<div
|
|
91
|
+
class="features-list"
|
|
92
|
+
[class.features-list--column]="config().direction === 'column'"
|
|
93
|
+
[class.features-list--row]="config().direction === 'row'"
|
|
94
|
+
[class.features-list--gap-small]="config().gap === 'small'"
|
|
95
|
+
[class.features-list--gap-medium]="config().gap === 'medium'"
|
|
96
|
+
[class.features-list--gap-large]="config().gap === 'large'"
|
|
97
|
+
[class.features-list--center]="config().alignment === 'center'"
|
|
98
|
+
>
|
|
99
|
+
@for (feature of props().features; track feature.titleKey) {
|
|
100
|
+
<div class="feature">
|
|
101
|
+
<div
|
|
102
|
+
class="feature__icon"
|
|
103
|
+
[style.color]="iconColorStyle()"
|
|
104
|
+
[style.width.px]="config().iconSize"
|
|
105
|
+
[style.height.px]="config().iconSize"
|
|
106
|
+
>
|
|
107
|
+
@if (feature.icon === 'custom' && feature.svgPath) {
|
|
108
|
+
<svg
|
|
109
|
+
[attr.width]="config().iconSize"
|
|
110
|
+
[attr.height]="config().iconSize"
|
|
111
|
+
viewBox="0 0 24 24"
|
|
112
|
+
fill="none"
|
|
113
|
+
stroke="currentColor"
|
|
114
|
+
stroke-width="2"
|
|
115
|
+
stroke-linecap="round"
|
|
116
|
+
stroke-linejoin="round"
|
|
117
|
+
>
|
|
118
|
+
<path [attr.d]="feature.svgPath" />
|
|
119
|
+
</svg>
|
|
120
|
+
} @else {
|
|
121
|
+
<ion-icon
|
|
122
|
+
[name]="feature.icon"
|
|
123
|
+
[style.font-size.px]="config().iconSize"
|
|
124
|
+
></ion-icon>
|
|
125
|
+
}
|
|
126
|
+
</div>
|
|
127
|
+
<div class="feature__content">
|
|
128
|
+
<strong class="feature__title">{{ getTitle(feature) }}</strong>
|
|
129
|
+
<p class="feature__description">{{ getDescription(feature) }}</p>
|
|
130
|
+
</div>
|
|
131
|
+
</div>
|
|
132
|
+
}
|
|
133
|
+
</div>
|
|
134
|
+
`, isInline: true, styles: [".features-list{display:flex;&--column{flex-direction:column}&--row{flex-direction:row;flex-wrap:wrap}&--gap-small{gap:1rem}&--gap-medium{gap:1.5rem}&--gap-large{gap:2rem}&--center{align-items:center;text-align:center;.feature{flex-direction:column;align-items:center}.feature__content{text-align:center}}}.feature{display:flex;align-items:flex-start;gap:1rem}.feature__icon{flex-shrink:0;display:flex;align-items:center;justify-content:center}.feature__content{flex:1}.feature__title{display:block;font-size:1.1rem;font-weight:600;color:var(--ion-text-color);margin-bottom:.25rem}.feature__description{margin:0;font-size:.95rem;color:var(--ion-color-medium);line-height:1.5}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }] }); }
|
|
135
|
+
}
|
|
136
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeaturesListComponent, decorators: [{
|
|
137
|
+
type: Component,
|
|
138
|
+
args: [{ selector: 'val-features-list', standalone: true, imports: [CommonModule, IonIcon], template: `
|
|
139
|
+
<div
|
|
140
|
+
class="features-list"
|
|
141
|
+
[class.features-list--column]="config().direction === 'column'"
|
|
142
|
+
[class.features-list--row]="config().direction === 'row'"
|
|
143
|
+
[class.features-list--gap-small]="config().gap === 'small'"
|
|
144
|
+
[class.features-list--gap-medium]="config().gap === 'medium'"
|
|
145
|
+
[class.features-list--gap-large]="config().gap === 'large'"
|
|
146
|
+
[class.features-list--center]="config().alignment === 'center'"
|
|
147
|
+
>
|
|
148
|
+
@for (feature of props().features; track feature.titleKey) {
|
|
149
|
+
<div class="feature">
|
|
150
|
+
<div
|
|
151
|
+
class="feature__icon"
|
|
152
|
+
[style.color]="iconColorStyle()"
|
|
153
|
+
[style.width.px]="config().iconSize"
|
|
154
|
+
[style.height.px]="config().iconSize"
|
|
155
|
+
>
|
|
156
|
+
@if (feature.icon === 'custom' && feature.svgPath) {
|
|
157
|
+
<svg
|
|
158
|
+
[attr.width]="config().iconSize"
|
|
159
|
+
[attr.height]="config().iconSize"
|
|
160
|
+
viewBox="0 0 24 24"
|
|
161
|
+
fill="none"
|
|
162
|
+
stroke="currentColor"
|
|
163
|
+
stroke-width="2"
|
|
164
|
+
stroke-linecap="round"
|
|
165
|
+
stroke-linejoin="round"
|
|
166
|
+
>
|
|
167
|
+
<path [attr.d]="feature.svgPath" />
|
|
168
|
+
</svg>
|
|
169
|
+
} @else {
|
|
170
|
+
<ion-icon
|
|
171
|
+
[name]="feature.icon"
|
|
172
|
+
[style.font-size.px]="config().iconSize"
|
|
173
|
+
></ion-icon>
|
|
174
|
+
}
|
|
175
|
+
</div>
|
|
176
|
+
<div class="feature__content">
|
|
177
|
+
<strong class="feature__title">{{ getTitle(feature) }}</strong>
|
|
178
|
+
<p class="feature__description">{{ getDescription(feature) }}</p>
|
|
179
|
+
</div>
|
|
180
|
+
</div>
|
|
181
|
+
}
|
|
182
|
+
</div>
|
|
183
|
+
`, styles: [".features-list{display:flex;&--column{flex-direction:column}&--row{flex-direction:row;flex-wrap:wrap}&--gap-small{gap:1rem}&--gap-medium{gap:1.5rem}&--gap-large{gap:2rem}&--center{align-items:center;text-align:center;.feature{flex-direction:column;align-items:center}.feature__content{text-align:center}}}.feature{display:flex;align-items:flex-start;gap:1rem}.feature__icon{flex-shrink:0;display:flex;align-items:center;justify-content:center}.feature__content{flex:1}.feature__title{display:block;font-size:1.1rem;font-weight:600;color:var(--ion-text-color);margin-bottom:.25rem}.feature__description{margin:0;font-size:.95rem;color:var(--ion-color-medium);line-height:1.5}\n"] }]
|
|
184
|
+
}] });
|
|
185
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"features-list.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/features-list/features-list.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;AACpD,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAC;AACrD,OAAO,EAAqC,sBAAsB,EAAE,MAAM,SAAS,CAAC;;AAEpF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAoCG;AA4HH,MAAM,OAAO,qBAAqB;IA3HlC;QA4HU,SAAI,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAEnC,8BAA8B;QACrB,UAAK,GAAG,KAAK,CAAC,QAAQ,EAAwB,CAAC;QAExD,yCAAyC;QACzC,WAAM,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC;YACvB,GAAG,sBAAsB;YACzB,GAAG,IAAI,CAAC,KAAK,EAAE;SAChB,CAAC,CAAC,CAAC;QAEJ,gEAAgE;QAChE,mBAAc,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC7B,MAAM,KAAK,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,SAAS,CAAC;YACtC,wCAAwC;YACxC,IAAI,KAAK,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;gBAChE,OAAO,mBAAmB,KAAK,GAAG,CAAC;YACrC,CAAC;YACD,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,CAAC;KA2BJ;IAzBC;;OAEG;IACH,QAAQ,CAAC,OAAoB;QAC3B,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,aAAa,CAAC;QAC7C,IAAI,SAAS,EAAE,CAAC;YACd,yBAAyB;YACzB,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC;QAClD,CAAC;QACD,OAAO,OAAO,CAAC,QAAQ,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,cAAc,CAAC,OAAoB;QACjC,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,aAAa,CAAC;QAC7C,IAAI,SAAS,EAAE,CAAC;YACd,yBAAyB;YACzB,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,cAAc,EAAE,SAAS,CAAC,CAAC;QACxD,CAAC;QACD,OAAO,OAAO,CAAC,cAAc,CAAC;IAChC,CAAC;+GA9CU,qBAAqB;mGAArB,qBAAqB,gNAvHtB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6CT,6uBA9CS,YAAY,+BAAE,OAAO;;4FAwHpB,qBAAqB;kBA3HjC,SAAS;+BACE,mBAAmB,cACjB,IAAI,WACP,CAAC,YAAY,EAAE,OAAO,CAAC,YACtB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6CT","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, computed, inject, input } from '@angular/core';\nimport { IonIcon } from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport { FeatureItem, FeaturesListMetadata, FEATURES_LIST_DEFAULTS } from './types';\n\n/**\n * val-features-list\n *\n * A component to display a list of features with icons, titles, and descriptions.\n * Supports i18n for all text content.\n *\n * @example\n * ```html\n * <val-features-list\n *   [props]=\"{\n *     features: [\n *       { icon: 'flash-outline', titleKey: 'feature1Title', descriptionKey: 'feature1Desc' },\n *       { icon: 'color-palette-outline', titleKey: 'feature2Title', descriptionKey: 'feature2Desc' }\n *     ],\n *     i18nNamespace: 'Features',\n *     iconColor: 'primary',\n *     direction: 'column'\n *   }\"\n * />\n * ```\n *\n * @example With custom SVG icons\n * ```html\n * <val-features-list\n *   [props]=\"{\n *     features: [\n *       {\n *         icon: 'custom',\n *         svgPath: 'M13 2L3 14h9l-1 8 10-12h-9l1-8z',\n *         titleKey: 'Fast & Secure',\n *         descriptionKey: 'Modern authentication with OAuth and MFA'\n *       }\n *     ]\n *   }\"\n * />\n * ```\n */\n@Component({\n  selector: 'val-features-list',\n  standalone: true,\n  imports: [CommonModule, IonIcon],\n  template: `\n    <div\n      class=\"features-list\"\n      [class.features-list--column]=\"config().direction === 'column'\"\n      [class.features-list--row]=\"config().direction === 'row'\"\n      [class.features-list--gap-small]=\"config().gap === 'small'\"\n      [class.features-list--gap-medium]=\"config().gap === 'medium'\"\n      [class.features-list--gap-large]=\"config().gap === 'large'\"\n      [class.features-list--center]=\"config().alignment === 'center'\"\n    >\n      @for (feature of props().features; track feature.titleKey) {\n        <div class=\"feature\">\n          <div\n            class=\"feature__icon\"\n            [style.color]=\"iconColorStyle()\"\n            [style.width.px]=\"config().iconSize\"\n            [style.height.px]=\"config().iconSize\"\n          >\n            @if (feature.icon === 'custom' && feature.svgPath) {\n              <svg\n                [attr.width]=\"config().iconSize\"\n                [attr.height]=\"config().iconSize\"\n                viewBox=\"0 0 24 24\"\n                fill=\"none\"\n                stroke=\"currentColor\"\n                stroke-width=\"2\"\n                stroke-linecap=\"round\"\n                stroke-linejoin=\"round\"\n              >\n                <path [attr.d]=\"feature.svgPath\" />\n              </svg>\n            } @else {\n              <ion-icon\n                [name]=\"feature.icon\"\n                [style.font-size.px]=\"config().iconSize\"\n              ></ion-icon>\n            }\n          </div>\n          <div class=\"feature__content\">\n            <strong class=\"feature__title\">{{ getTitle(feature) }}</strong>\n            <p class=\"feature__description\">{{ getDescription(feature) }}</p>\n          </div>\n        </div>\n      }\n    </div>\n  `,\n  styles: [`\n    .features-list {\n      display: flex;\n\n      &--column {\n        flex-direction: column;\n      }\n\n      &--row {\n        flex-direction: row;\n        flex-wrap: wrap;\n      }\n\n      &--gap-small {\n        gap: 1rem;\n      }\n\n      &--gap-medium {\n        gap: 1.5rem;\n      }\n\n      &--gap-large {\n        gap: 2rem;\n      }\n\n      &--center {\n        align-items: center;\n        text-align: center;\n\n        .feature {\n          flex-direction: column;\n          align-items: center;\n        }\n\n        .feature__content {\n          text-align: center;\n        }\n      }\n    }\n\n    .feature {\n      display: flex;\n      align-items: flex-start;\n      gap: 1rem;\n    }\n\n    .feature__icon {\n      flex-shrink: 0;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n    }\n\n    .feature__content {\n      flex: 1;\n    }\n\n    .feature__title {\n      display: block;\n      font-size: 1.1rem;\n      font-weight: 600;\n      color: var(--ion-text-color);\n      margin-bottom: 0.25rem;\n    }\n\n    .feature__description {\n      margin: 0;\n      font-size: 0.95rem;\n      color: var(--ion-color-medium);\n      line-height: 1.5;\n    }\n  `],\n})\nexport class FeaturesListComponent {\n  private i18n = inject(I18nService);\n\n  /** Component configuration */\n  readonly props = input.required<FeaturesListMetadata>();\n\n  /** Merged configuration with defaults */\n  config = computed(() => ({\n    ...FEATURES_LIST_DEFAULTS,\n    ...this.props(),\n  }));\n\n  /** Resolved icon color (handles Ionic colors and CSS colors) */\n  iconColorStyle = computed(() => {\n    const color = this.config().iconColor;\n    // Check if it's an Ionic color variable\n    if (color && !color.startsWith('#') && !color.startsWith('rgb')) {\n      return `var(--ion-color-${color})`;\n    }\n    return color;\n  });\n\n  /**\n   * Get translated title or return direct text\n   */\n  getTitle(feature: FeatureItem): string {\n    const namespace = this.props().i18nNamespace;\n    if (namespace) {\n      // Track language changes\n      this.i18n.lang();\n      return this.i18n.t(feature.titleKey, namespace);\n    }\n    return feature.titleKey;\n  }\n\n  /**\n   * Get translated description or return direct text\n   */\n  getDescription(feature: FeatureItem): string {\n    const namespace = this.props().i18nNamespace;\n    if (namespace) {\n      // Track language changes\n      this.i18n.lang();\n      return this.i18n.t(feature.descriptionKey, namespace);\n    }\n    return feature.descriptionKey;\n  }\n}\n"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Default values for FeaturesListMetadata
|
|
3
|
+
*/
|
|
4
|
+
export const FEATURES_LIST_DEFAULTS = {
|
|
5
|
+
iconColor: 'primary',
|
|
6
|
+
iconSize: 32,
|
|
7
|
+
direction: 'column',
|
|
8
|
+
gap: 'medium',
|
|
9
|
+
alignment: 'start',
|
|
10
|
+
};
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2ZlYXR1cmVzLWxpc3QvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBa0NBOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sc0JBQXNCLEdBQXVFO0lBQ3hHLFNBQVMsRUFBRSxTQUFTO0lBQ3BCLFFBQVEsRUFBRSxFQUFFO0lBQ1osU0FBUyxFQUFFLFFBQVE7SUFDbkIsR0FBRyxFQUFFLFFBQVE7SUFDYixTQUFTLEVBQUUsT0FBTztDQUNuQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBGZWF0dXJlIGl0ZW0gY29uZmlndXJhdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIEZlYXR1cmVJdGVtIHtcbiAgLyoqIElvbmljb24gbmFtZSBvciAnY3VzdG9tJyBmb3IgU1ZHIHBhdGggKi9cbiAgaWNvbjogc3RyaW5nO1xuICAvKiogU1ZHIHBhdGggZGF0YSAob25seSB1c2VkIHdoZW4gaWNvbiBpcyAnY3VzdG9tJykgKi9cbiAgc3ZnUGF0aD86IHN0cmluZztcbiAgLyoqIFRpdGxlIGkxOG4ga2V5IG9yIGRpcmVjdCB0ZXh0ICovXG4gIHRpdGxlS2V5OiBzdHJpbmc7XG4gIC8qKiBEZXNjcmlwdGlvbiBpMThuIGtleSBvciBkaXJlY3QgdGV4dCAqL1xuICBkZXNjcmlwdGlvbktleTogc3RyaW5nO1xufVxuXG4vKipcbiAqIEZlYXR1cmVzIGxpc3QgY29tcG9uZW50IGNvbmZpZ3VyYXRpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBGZWF0dXJlc0xpc3RNZXRhZGF0YSB7XG4gIC8qKiBBcnJheSBvZiBmZWF0dXJlcyB0byBkaXNwbGF5ICovXG4gIGZlYXR1cmVzOiBGZWF0dXJlSXRlbVtdO1xuICAvKiogaTE4biBuYW1lc3BhY2UgZm9yIHRyYW5zbGF0aW9ucyAoaWYgdXNpbmcga2V5cykgKi9cbiAgaTE4bk5hbWVzcGFjZT86IHN0cmluZztcbiAgLyoqIEljb24gY29sb3IgKElvbmljIGNvbG9yIG9yIENTUyBjb2xvcikgKi9cbiAgaWNvbkNvbG9yPzogc3RyaW5nO1xuICAvKiogSWNvbiBzaXplIGluIHBpeGVscyAqL1xuICBpY29uU2l6ZT86IG51bWJlcjtcbiAgLyoqIExheW91dCBkaXJlY3Rpb24gKi9cbiAgZGlyZWN0aW9uPzogJ2NvbHVtbicgfCAncm93JztcbiAgLyoqIEdhcCBiZXR3ZWVuIGZlYXR1cmVzICovXG4gIGdhcD86ICdzbWFsbCcgfCAnbWVkaXVtJyB8ICdsYXJnZSc7XG4gIC8qKiBBbGlnbm1lbnQgb2YgZmVhdHVyZXMgKi9cbiAgYWxpZ25tZW50PzogJ3N0YXJ0JyB8ICdjZW50ZXInO1xufVxuXG4vKipcbiAqIERlZmF1bHQgdmFsdWVzIGZvciBGZWF0dXJlc0xpc3RNZXRhZGF0YVxuICovXG5leHBvcnQgY29uc3QgRkVBVFVSRVNfTElTVF9ERUZBVUxUUzogUmVxdWlyZWQ8T21pdDxGZWF0dXJlc0xpc3RNZXRhZGF0YSwgJ2ZlYXR1cmVzJyB8ICdpMThuTmFtZXNwYWNlJz4+ID0ge1xuICBpY29uQ29sb3I6ICdwcmltYXJ5JyxcbiAgaWNvblNpemU6IDMyLFxuICBkaXJlY3Rpb246ICdjb2x1bW4nLFxuICBnYXA6ICdtZWRpdW0nLFxuICBhbGlnbm1lbnQ6ICdzdGFydCcsXG59O1xuIl19
|
|
@@ -96,7 +96,6 @@ export class FooterLinksComponent {
|
|
|
96
96
|
<val-image
|
|
97
97
|
*ngIf="props.logo"
|
|
98
98
|
class="logo-clickable footer-logo footer-logo--light"
|
|
99
|
-
style="display: block; margin-top: 1rem;"
|
|
100
99
|
[props]="{
|
|
101
100
|
width: 14,
|
|
102
101
|
mode: 'box',
|
|
@@ -114,7 +113,6 @@ export class FooterLinksComponent {
|
|
|
114
113
|
<val-image
|
|
115
114
|
*ngIf="props.logo && props.logoDark"
|
|
116
115
|
class="logo-clickable footer-logo footer-logo--dark"
|
|
117
|
-
style="display: block; margin-top: 1rem;"
|
|
118
116
|
[props]="{
|
|
119
117
|
width: 14,
|
|
120
118
|
mode: 'box',
|
|
@@ -198,7 +196,7 @@ export class FooterLinksComponent {
|
|
|
198
196
|
</ion-col>
|
|
199
197
|
</ion-row>
|
|
200
198
|
</div>
|
|
201
|
-
`, isInline: true, styles: [".footer-links .mobile-layout{display:block;text-align:center;padding:8px 0}.footer-links .desktop-layout{display:none;align-items:center;justify-content:space-between;margin-top:16px;padding:8px 0}.footer-links .social-links-container{display:flex;align-items:center;justify-content:center;gap:12px;margin-top:8px;flex-wrap:wrap}.footer-links .social-button{--color: #666;--color-hover: #333;transition:all .2s ease;min-height:40px;min-width:40px;margin:0}.footer-links .social-button:hover{--color: #333;transform:translateY(-1px)}.footer-links .social-button ion-icon{font-size:20px}.footer-links .logo-clickable{cursor:pointer}.footer-links .footer-logo--light{display:block}.footer-links .footer-logo--dark{display:none}@media (prefers-color-scheme: dark){.footer-links .footer-logo--light{display:none}.footer-links .footer-logo--dark{display:block}}.footer-links :host-context(.dark) .footer-logo--light{display:none}.footer-links :host-context(.dark) .footer-logo--dark{display:block}@media (min-width: 768px){.footer-links .mobile-layout{display:none!important}.footer-links .desktop-layout{display:flex!important}.footer-links .social-links-container{margin-top:0;justify-content:flex-end}}@media (max-width: 767px){.footer-links .mobile-layout{display:block!important}.footer-links .desktop-layout{display:none!important}.footer-links .social-links-container{justify-content:center;margin-bottom:8px}}\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: "component", type: IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: IonRow, selector: "ion-row" }, { 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: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: DividerComponent, selector: "val-divider", inputs: ["props"] }, { kind: "component", type: ImageComponent, selector: "val-image", inputs: ["props"] }, { kind: "component", type: LinksCakeComponent, selector: "val-links-cake", inputs: ["props"] }, { kind: "component", type: LanguageSelectorComponent, selector: "val-language-selector", inputs: ["props"], outputs: ["languageChange"] }] }); }
|
|
199
|
+
`, isInline: true, styles: [".footer-links .mobile-layout{display:block;text-align:center;padding:8px 0}.footer-links .desktop-layout{display:none;align-items:center;justify-content:space-between;margin-top:16px;padding:8px 0}.footer-links .social-links-container{display:flex;align-items:center;justify-content:center;gap:12px;margin-top:8px;flex-wrap:wrap}.footer-links .social-button{--color: #666;--color-hover: #333;transition:all .2s ease;min-height:40px;min-width:40px;margin:0}.footer-links .social-button:hover{--color: #333;transform:translateY(-1px)}.footer-links .social-button ion-icon{font-size:20px}.footer-links .logo-clickable{cursor:pointer}.footer-links .footer-logo{margin-top:1rem}.footer-links .footer-logo--light{display:block}.footer-links .footer-logo--dark{display:none}@media (prefers-color-scheme: dark){.footer-links .footer-logo--light{display:none}.footer-links .footer-logo--dark{display:block}}.footer-links :host-context(.dark) .footer-logo--light{display:none}.footer-links :host-context(.dark) .footer-logo--dark{display:block}@media (min-width: 768px){.footer-links .mobile-layout{display:none!important}.footer-links .desktop-layout{display:flex!important}.footer-links .social-links-container{margin-top:0;justify-content:flex-end}}@media (max-width: 767px){.footer-links .mobile-layout{display:block!important}.footer-links .desktop-layout{display:none!important}.footer-links .social-links-container{justify-content:center;margin-bottom:8px}}\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: "component", type: IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: IonRow, selector: "ion-row" }, { 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: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: DividerComponent, selector: "val-divider", inputs: ["props"] }, { kind: "component", type: ImageComponent, selector: "val-image", inputs: ["props"] }, { kind: "component", type: LinksCakeComponent, selector: "val-links-cake", inputs: ["props"] }, { kind: "component", type: LanguageSelectorComponent, selector: "val-language-selector", inputs: ["props"], outputs: ["languageChange"] }] }); }
|
|
202
200
|
}
|
|
203
201
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FooterLinksComponent, decorators: [{
|
|
204
202
|
type: Component,
|
|
@@ -240,7 +238,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
240
238
|
<val-image
|
|
241
239
|
*ngIf="props.logo"
|
|
242
240
|
class="logo-clickable footer-logo footer-logo--light"
|
|
243
|
-
style="display: block; margin-top: 1rem;"
|
|
244
241
|
[props]="{
|
|
245
242
|
width: 14,
|
|
246
243
|
mode: 'box',
|
|
@@ -258,7 +255,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
258
255
|
<val-image
|
|
259
256
|
*ngIf="props.logo && props.logoDark"
|
|
260
257
|
class="logo-clickable footer-logo footer-logo--dark"
|
|
261
|
-
style="display: block; margin-top: 1rem;"
|
|
262
258
|
[props]="{
|
|
263
259
|
width: 14,
|
|
264
260
|
mode: 'box',
|
|
@@ -342,10 +338,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
342
338
|
</ion-col>
|
|
343
339
|
</ion-row>
|
|
344
340
|
</div>
|
|
345
|
-
`, styles: [".footer-links .mobile-layout{display:block;text-align:center;padding:8px 0}.footer-links .desktop-layout{display:none;align-items:center;justify-content:space-between;margin-top:16px;padding:8px 0}.footer-links .social-links-container{display:flex;align-items:center;justify-content:center;gap:12px;margin-top:8px;flex-wrap:wrap}.footer-links .social-button{--color: #666;--color-hover: #333;transition:all .2s ease;min-height:40px;min-width:40px;margin:0}.footer-links .social-button:hover{--color: #333;transform:translateY(-1px)}.footer-links .social-button ion-icon{font-size:20px}.footer-links .logo-clickable{cursor:pointer}.footer-links .footer-logo--light{display:block}.footer-links .footer-logo--dark{display:none}@media (prefers-color-scheme: dark){.footer-links .footer-logo--light{display:none}.footer-links .footer-logo--dark{display:block}}.footer-links :host-context(.dark) .footer-logo--light{display:none}.footer-links :host-context(.dark) .footer-logo--dark{display:block}@media (min-width: 768px){.footer-links .mobile-layout{display:none!important}.footer-links .desktop-layout{display:flex!important}.footer-links .social-links-container{margin-top:0;justify-content:flex-end}}@media (max-width: 767px){.footer-links .mobile-layout{display:block!important}.footer-links .desktop-layout{display:none!important}.footer-links .social-links-container{justify-content:center;margin-bottom:8px}}\n"] }]
|
|
341
|
+
`, styles: [".footer-links .mobile-layout{display:block;text-align:center;padding:8px 0}.footer-links .desktop-layout{display:none;align-items:center;justify-content:space-between;margin-top:16px;padding:8px 0}.footer-links .social-links-container{display:flex;align-items:center;justify-content:center;gap:12px;margin-top:8px;flex-wrap:wrap}.footer-links .social-button{--color: #666;--color-hover: #333;transition:all .2s ease;min-height:40px;min-width:40px;margin:0}.footer-links .social-button:hover{--color: #333;transform:translateY(-1px)}.footer-links .social-button ion-icon{font-size:20px}.footer-links .logo-clickable{cursor:pointer}.footer-links .footer-logo{margin-top:1rem}.footer-links .footer-logo--light{display:block}.footer-links .footer-logo--dark{display:none}@media (prefers-color-scheme: dark){.footer-links .footer-logo--light{display:none}.footer-links .footer-logo--dark{display:block}}.footer-links :host-context(.dark) .footer-logo--light{display:none}.footer-links :host-context(.dark) .footer-logo--dark{display:block}@media (min-width: 768px){.footer-links .mobile-layout{display:none!important}.footer-links .desktop-layout{display:flex!important}.footer-links .social-links-container{margin-top:0;justify-content:flex-end}}@media (max-width: 767px){.footer-links .mobile-layout{display:block!important}.footer-links .desktop-layout{display:none!important}.footer-links .social-links-container{justify-content:center;margin-bottom:8px}}\n"] }]
|
|
346
342
|
}], propDecorators: { props: [{
|
|
347
343
|
type: Input
|
|
348
344
|
}], logoClick: [{
|
|
349
345
|
type: Output
|
|
350
346
|
}] } });
|
|
351
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"footer-links.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/footer-links/footer-links.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,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,2BAA2B,CAAC;AAC/E,OAAO,EAAE,iBAAiB,EAAE,MAAM,sCAAsC,CAAC;AACzE,OAAO,EAAE,gBAAgB,EAAE,MAAM,uCAAuC,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,MAAM,mCAAmC,CAAC;AACnE,OAAO,EAAE,yBAAyB,EAAE,MAAM,kDAAkD,CAAC;AAE7F,OAAO,EAAE,kBAAkB,EAAE,MAAM,oCAAoC,CAAC;;;AAGxE;;;;;;;;;;;;;;;;;;;;;GAqBG;AAoJH,MAAM,OAAO,oBAAoB;IAnJjC;QAoJU,QAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAOxC;;WAEG;QACO,cAAS,GAAG,IAAI,YAAY,EAAQ,CAAC;KAgChD;IA9BC;;OAEG;IACH,IAAI,sBAAsB;QACxB,IAAI,IAAI,CAAC,KAAK,CAAC,gBAAgB,IAAI,OAAO,IAAI,CAAC,KAAK,CAAC,gBAAgB,KAAK,QAAQ,EAAE,CAAC;YACnF,OAAO,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC;QACrC,CAAC;QACD,OAAO;YACL,SAAS,EAAE,KAAK;YAChB,WAAW,EAAE;gBACX,SAAS,EAAE,SAAS;gBACpB,GAAG,EAAE,UAAU;gBACf,QAAQ,EAAE,QAAQ;aACnB;YACD,SAAS,EAAE,IAAI;YACf,KAAK,EAAE,SAAS;YAChB,IAAI,EAAE,SAAS;YACf,IAAI,EAAE,SAAS;SAChB,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,WAAW;QACT,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACzB,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC;QACD,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;IACxB,CAAC;+GA1CU,oBAAoB;mGAApB,oBAAoB,6IArIrB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkIT,88CA5IC,YAAY,gQACZ,MAAM,kTACN,MAAM,oDACN,SAAS,oPACT,OAAO,2JACP,gBAAgB,2EAChB,cAAc,yEACd,kBAAkB,8EAClB,yBAAyB;;4FAuIhB,oBAAoB;kBAnJhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP;wBACP,YAAY;wBACZ,MAAM;wBACN,MAAM;wBACN,SAAS;wBACT,OAAO;wBACP,gBAAgB;wBAChB,cAAc;wBACd,kBAAkB;wBAClB,yBAAyB;qBAC1B,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkIT;8BASQ,KAAK;sBAAb,KAAK;gBAKI,SAAS;sBAAlB,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { IonButton, IonCol, IonIcon, IonRow } from '@ionic/angular/standalone';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { DividerComponent } from '../../atoms/divider/divider.component';\nimport { ImageComponent } from '../../atoms/image/image.component';\nimport { LanguageSelectorComponent } from '../language-selector/language-selector.component';\nimport { LanguageSelectorMetadata } from '../language-selector/types';\nimport { LinksCakeComponent } from '../links-cake/links-cake.component';\nimport { FooterLinksMetadata } from './types';\n\n/**\n * val-footer-links\n *\n * A responsive footer component with two columns of links, social media icons,\n * logo, and optional language selector.\n *\n * @example\n * <val-footer-links\n *   [props]=\"{\n *     leftLinks: { title: 'Company', links: [...] },\n *     rightLinks: { title: 'Resources', links: [...] },\n *     logo: 'logo.svg',\n *     logoRoute: '/',\n *     socialLinks: [\n *       { icon: 'logo-facebook', url: 'https://facebook.com', name: 'Facebook' }\n *     ]\n *   }\"\n * ></val-footer-links>\n *\n * @input props - Footer configuration\n * @output logoClick - Emits when logo is clicked\n */\n@Component({\n  selector: 'val-footer-links',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonCol,\n    IonRow,\n    IonButton,\n    IonIcon,\n    DividerComponent,\n    ImageComponent,\n    LinksCakeComponent,\n    LanguageSelectorComponent,\n  ],\n  template: `\n    <div class=\"footer-links\">\n      <ion-row>\n        <ion-col size=\"12\">\n          <val-divider [props]=\"props.divider || { size: 'small', color: 'medium', fill: 'solid' }\" />\n        </ion-col>\n        <ion-col size-xs=\"12\" size-md=\"6\">\n          <val-links-cake [props]=\"props.leftLinks\" />\n        </ion-col>\n        <ion-col size-xs=\"12\" size-md=\"6\">\n          <val-links-cake [props]=\"props.rightLinks\" />\n        </ion-col>\n      </ion-row>\n\n      <ion-row *ngIf=\"props.languageSelector !== false\" style=\"margin-top: 1rem;\">\n        <ion-col size=\"12\">\n          <val-language-selector [props]=\"languageSelectorConfig\" />\n        </ion-col>\n      </ion-row>\n\n      <ion-row style=\"margin-top: 1rem;\">\n        <ion-col size=\"12\" *ngIf=\"props.logo || props.socialLinks?.length\">\n          <!-- Mobile layout: centered -->\n          <div class=\"mobile-layout\">\n            <!-- Light mode logo -->\n            <val-image\n              *ngIf=\"props.logo\"\n              class=\"logo-clickable footer-logo footer-logo--light\"\n              style=\"display: block; margin-top: 1rem;\"\n              [props]=\"{\n                width: 14,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logo,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <!-- Dark mode logo -->\n            <val-image\n              *ngIf=\"props.logo && props.logoDark\"\n              class=\"logo-clickable footer-logo footer-logo--dark\"\n              style=\"display: block; margin-top: 1rem;\"\n              [props]=\"{\n                width: 14,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logoDark,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <div class=\"social-links-container\" *ngIf=\"props.socialLinks?.length\" style=\"margin-top: 1rem\">\n              <ion-button\n                *ngFor=\"let social of props.socialLinks\"\n                fill=\"clear\"\n                size=\"small\"\n                [href]=\"social.url\"\n                target=\"_blank\"\n                rel=\"noopener\"\n                class=\"social-button\"\n                [attr.aria-label]=\"social.name\"\n              >\n                <ion-icon [name]=\"social.icon\" slot=\"icon-only\" [color]=\"props.socialIconColor || 'dark'\"></ion-icon>\n              </ion-button>\n            </div>\n          </div>\n\n          <!-- Desktop layout: space-between -->\n          <div class=\"desktop-layout\">\n            <!-- Light mode logo -->\n            <val-image\n              *ngIf=\"props.logo\"\n              class=\"logo-clickable footer-logo footer-logo--light\"\n              [props]=\"{\n                width: 16,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logo,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <!-- Dark mode logo -->\n            <val-image\n              *ngIf=\"props.logo && props.logoDark\"\n              class=\"logo-clickable footer-logo footer-logo--dark\"\n              [props]=\"{\n                width: 16,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logoDark,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <div class=\"social-links-container\" *ngIf=\"props.socialLinks?.length\">\n              <ion-button\n                *ngFor=\"let social of props.socialLinks\"\n                fill=\"clear\"\n                size=\"small\"\n                [href]=\"social.url\"\n                target=\"_blank\"\n                rel=\"noopener\"\n                class=\"social-button\"\n                [attr.aria-label]=\"social.name\"\n              >\n                <ion-icon [name]=\"social.icon\" slot=\"icon-only\" [color]=\"props.socialIconColor || 'dark'\"></ion-icon>\n              </ion-button>\n            </div>\n          </div>\n        </ion-col>\n      </ion-row>\n    </div>\n  `,\n  styleUrls: ['./footer-links.component.scss'],\n})\nexport class FooterLinksComponent {\n  private nav = inject(NavigationService);\n\n  /**\n   * Footer configuration.\n   */\n  @Input() props!: FooterLinksMetadata;\n\n  /**\n   * Emits when the logo is clicked.\n   */\n  @Output() logoClick = new EventEmitter<void>();\n\n  /**\n   * Default language selector configuration.\n   */\n  get languageSelectorConfig(): LanguageSelectorMetadata {\n    if (this.props.languageSelector && typeof this.props.languageSelector === 'object') {\n      return this.props.languageSelector;\n    }\n    return {\n      showLabel: false,\n      labelConfig: {\n        className: '_global',\n        key: 'language',\n        fallback: 'Idioma',\n      },\n      showFlags: true,\n      color: 'primary',\n      size: 'default',\n      fill: 'outline',\n    };\n  }\n\n  /**\n   * Handles logo click - navigates if route provided, otherwise emits event.\n   */\n  onLogoClick(): void {\n    if (this.props.logoRoute) {\n      this.nav.navigateByUrl(this.props.logoRoute);\n    }\n    this.logoClick.emit();\n  }\n}\n"]}
|
|
347
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"footer-links.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/footer-links/footer-links.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,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,2BAA2B,CAAC;AAC/E,OAAO,EAAE,iBAAiB,EAAE,MAAM,sCAAsC,CAAC;AACzE,OAAO,EAAE,gBAAgB,EAAE,MAAM,uCAAuC,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,MAAM,mCAAmC,CAAC;AACnE,OAAO,EAAE,yBAAyB,EAAE,MAAM,kDAAkD,CAAC;AAE7F,OAAO,EAAE,kBAAkB,EAAE,MAAM,oCAAoC,CAAC;;;AAGxE;;;;;;;;;;;;;;;;;;;;;GAqBG;AAkJH,MAAM,OAAO,oBAAoB;IAjJjC;QAkJU,QAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAOxC;;WAEG;QACO,cAAS,GAAG,IAAI,YAAY,EAAQ,CAAC;KAgChD;IA9BC;;OAEG;IACH,IAAI,sBAAsB;QACxB,IAAI,IAAI,CAAC,KAAK,CAAC,gBAAgB,IAAI,OAAO,IAAI,CAAC,KAAK,CAAC,gBAAgB,KAAK,QAAQ,EAAE,CAAC;YACnF,OAAO,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC;QACrC,CAAC;QACD,OAAO;YACL,SAAS,EAAE,KAAK;YAChB,WAAW,EAAE;gBACX,SAAS,EAAE,SAAS;gBACpB,GAAG,EAAE,UAAU;gBACf,QAAQ,EAAE,QAAQ;aACnB;YACD,SAAS,EAAE,IAAI;YACf,KAAK,EAAE,SAAS;YAChB,IAAI,EAAE,SAAS;YACf,IAAI,EAAE,SAAS;SAChB,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,WAAW;QACT,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACzB,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC;QACD,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;IACxB,CAAC;+GA1CU,oBAAoB;mGAApB,oBAAoB,6IAnIrB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgIT,y/CA1IC,YAAY,gQACZ,MAAM,kTACN,MAAM,oDACN,SAAS,oPACT,OAAO,2JACP,gBAAgB,2EAChB,cAAc,yEACd,kBAAkB,8EAClB,yBAAyB;;4FAqIhB,oBAAoB;kBAjJhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP;wBACP,YAAY;wBACZ,MAAM;wBACN,MAAM;wBACN,SAAS;wBACT,OAAO;wBACP,gBAAgB;wBAChB,cAAc;wBACd,kBAAkB;wBAClB,yBAAyB;qBAC1B,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgIT;8BASQ,KAAK;sBAAb,KAAK;gBAKI,SAAS;sBAAlB,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { IonButton, IonCol, IonIcon, IonRow } from '@ionic/angular/standalone';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { DividerComponent } from '../../atoms/divider/divider.component';\nimport { ImageComponent } from '../../atoms/image/image.component';\nimport { LanguageSelectorComponent } from '../language-selector/language-selector.component';\nimport { LanguageSelectorMetadata } from '../language-selector/types';\nimport { LinksCakeComponent } from '../links-cake/links-cake.component';\nimport { FooterLinksMetadata } from './types';\n\n/**\n * val-footer-links\n *\n * A responsive footer component with two columns of links, social media icons,\n * logo, and optional language selector.\n *\n * @example\n * <val-footer-links\n *   [props]=\"{\n *     leftLinks: { title: 'Company', links: [...] },\n *     rightLinks: { title: 'Resources', links: [...] },\n *     logo: 'logo.svg',\n *     logoRoute: '/',\n *     socialLinks: [\n *       { icon: 'logo-facebook', url: 'https://facebook.com', name: 'Facebook' }\n *     ]\n *   }\"\n * ></val-footer-links>\n *\n * @input props - Footer configuration\n * @output logoClick - Emits when logo is clicked\n */\n@Component({\n  selector: 'val-footer-links',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonCol,\n    IonRow,\n    IonButton,\n    IonIcon,\n    DividerComponent,\n    ImageComponent,\n    LinksCakeComponent,\n    LanguageSelectorComponent,\n  ],\n  template: `\n    <div class=\"footer-links\">\n      <ion-row>\n        <ion-col size=\"12\">\n          <val-divider [props]=\"props.divider || { size: 'small', color: 'medium', fill: 'solid' }\" />\n        </ion-col>\n        <ion-col size-xs=\"12\" size-md=\"6\">\n          <val-links-cake [props]=\"props.leftLinks\" />\n        </ion-col>\n        <ion-col size-xs=\"12\" size-md=\"6\">\n          <val-links-cake [props]=\"props.rightLinks\" />\n        </ion-col>\n      </ion-row>\n\n      <ion-row *ngIf=\"props.languageSelector !== false\" style=\"margin-top: 1rem;\">\n        <ion-col size=\"12\">\n          <val-language-selector [props]=\"languageSelectorConfig\" />\n        </ion-col>\n      </ion-row>\n\n      <ion-row style=\"margin-top: 1rem;\">\n        <ion-col size=\"12\" *ngIf=\"props.logo || props.socialLinks?.length\">\n          <!-- Mobile layout: centered -->\n          <div class=\"mobile-layout\">\n            <!-- Light mode logo -->\n            <val-image\n              *ngIf=\"props.logo\"\n              class=\"logo-clickable footer-logo footer-logo--light\"\n              [props]=\"{\n                width: 14,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logo,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <!-- Dark mode logo -->\n            <val-image\n              *ngIf=\"props.logo && props.logoDark\"\n              class=\"logo-clickable footer-logo footer-logo--dark\"\n              [props]=\"{\n                width: 14,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logoDark,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <div class=\"social-links-container\" *ngIf=\"props.socialLinks?.length\" style=\"margin-top: 1rem\">\n              <ion-button\n                *ngFor=\"let social of props.socialLinks\"\n                fill=\"clear\"\n                size=\"small\"\n                [href]=\"social.url\"\n                target=\"_blank\"\n                rel=\"noopener\"\n                class=\"social-button\"\n                [attr.aria-label]=\"social.name\"\n              >\n                <ion-icon [name]=\"social.icon\" slot=\"icon-only\" [color]=\"props.socialIconColor || 'dark'\"></ion-icon>\n              </ion-button>\n            </div>\n          </div>\n\n          <!-- Desktop layout: space-between -->\n          <div class=\"desktop-layout\">\n            <!-- Light mode logo -->\n            <val-image\n              *ngIf=\"props.logo\"\n              class=\"logo-clickable footer-logo footer-logo--light\"\n              [props]=\"{\n                width: 16,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logo,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <!-- Dark mode logo -->\n            <val-image\n              *ngIf=\"props.logo && props.logoDark\"\n              class=\"logo-clickable footer-logo footer-logo--dark\"\n              [props]=\"{\n                width: 16,\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                src: props.logoDark,\n                alt: props.logoAlt || 'Logo',\n                limited: false,\n                dark: true\n              }\"\n              (click)=\"onLogoClick()\"\n            />\n            <div class=\"social-links-container\" *ngIf=\"props.socialLinks?.length\">\n              <ion-button\n                *ngFor=\"let social of props.socialLinks\"\n                fill=\"clear\"\n                size=\"small\"\n                [href]=\"social.url\"\n                target=\"_blank\"\n                rel=\"noopener\"\n                class=\"social-button\"\n                [attr.aria-label]=\"social.name\"\n              >\n                <ion-icon [name]=\"social.icon\" slot=\"icon-only\" [color]=\"props.socialIconColor || 'dark'\"></ion-icon>\n              </ion-button>\n            </div>\n          </div>\n        </ion-col>\n      </ion-row>\n    </div>\n  `,\n  styleUrls: ['./footer-links.component.scss'],\n})\nexport class FooterLinksComponent {\n  private nav = inject(NavigationService);\n\n  /**\n   * Footer configuration.\n   */\n  @Input() props!: FooterLinksMetadata;\n\n  /**\n   * Emits when the logo is clicked.\n   */\n  @Output() logoClick = new EventEmitter<void>();\n\n  /**\n   * Default language selector configuration.\n   */\n  get languageSelectorConfig(): LanguageSelectorMetadata {\n    if (this.props.languageSelector && typeof this.props.languageSelector === 'object') {\n      return this.props.languageSelector;\n    }\n    return {\n      showLabel: false,\n      labelConfig: {\n        className: '_global',\n        key: 'language',\n        fallback: 'Idioma',\n      },\n      showFlags: true,\n      color: 'primary',\n      size: 'default',\n      fill: 'outline',\n    };\n  }\n\n  /**\n   * Handles logo click - navigates if route provided, otherwise emits event.\n   */\n  onLogoClick(): void {\n    if (this.props.logoRoute) {\n      this.nav.navigateByUrl(this.props.logoRoute);\n    }\n    this.logoClick.emit();\n  }\n}\n"]}
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared Company Footer Configuration
|
|
3
|
+
*
|
|
4
|
+
* This configuration is shared across all Valtech products:
|
|
5
|
+
* - ui-docs
|
|
6
|
+
* - showcase
|
|
7
|
+
* - myvaltech (company website)
|
|
8
|
+
*
|
|
9
|
+
* When links or social profiles change, update here and publish a new version.
|
|
10
|
+
*/
|
|
11
|
+
/**
|
|
12
|
+
* Social media links for Valtech
|
|
13
|
+
*/
|
|
14
|
+
export const VALTECH_SOCIAL_LINKS = [
|
|
15
|
+
{ icon: 'logo-facebook', url: 'https://m.facebook.com/profile.php?id=61557610734470', name: 'Facebook' },
|
|
16
|
+
{ icon: 'logo-instagram', url: 'https://www.instagram.com/valtechltda/', name: 'Instagram' },
|
|
17
|
+
{ icon: 'logo-linkedin', url: 'https://www.linkedin.com/company/valtechltda/', name: 'LinkedIn' },
|
|
18
|
+
{ icon: 'logo-twitter', url: 'https://twitter.com/valtechltda', name: 'Twitter' },
|
|
19
|
+
{ icon: 'logo-youtube', url: 'https://www.youtube.com/channel/UCuF4FGdTiUXxANx1HS4Wi5Q', name: 'YouTube' },
|
|
20
|
+
{ icon: 'logo-tiktok', url: 'https://www.tiktok.com/@valtechltda', name: 'TikTok' },
|
|
21
|
+
];
|
|
22
|
+
/**
|
|
23
|
+
* Footer logo configuration
|
|
24
|
+
* Uses CSS variables for theme-aware logo switching
|
|
25
|
+
*/
|
|
26
|
+
export const VALTECH_FOOTER_LOGO = {
|
|
27
|
+
logo: 'assets/images/main-light.svg',
|
|
28
|
+
logoDark: 'assets/images/main-dark.svg',
|
|
29
|
+
logoAlt: 'Valtech Logo',
|
|
30
|
+
logoRoute: '/',
|
|
31
|
+
};
|
|
32
|
+
/**
|
|
33
|
+
* Company links organized by section
|
|
34
|
+
*/
|
|
35
|
+
export const VALTECH_COMPANY_LINKS = {
|
|
36
|
+
legal: [
|
|
37
|
+
{ key: 'aboutUs', url: '/about', external: false },
|
|
38
|
+
{ key: 'privacyPolicy', url: '/legal/privacy', external: false },
|
|
39
|
+
{ key: 'termsConditions', url: '/legal/terms', external: false },
|
|
40
|
+
],
|
|
41
|
+
support: [
|
|
42
|
+
{ key: 'contactSupport', url: '/contact', external: false },
|
|
43
|
+
{ key: 'faq', url: '/help/faq', external: false },
|
|
44
|
+
{ key: 'feedback', url: '/feedback', external: false },
|
|
45
|
+
],
|
|
46
|
+
};
|
|
47
|
+
/**
|
|
48
|
+
* i18n keys for the footer (Layout namespace)
|
|
49
|
+
* These should be registered in each app's i18n config
|
|
50
|
+
*/
|
|
51
|
+
export const VALTECH_FOOTER_I18N = {
|
|
52
|
+
es: {
|
|
53
|
+
// Footer titles
|
|
54
|
+
footerLeftTitle: 'Sigamos en contacto',
|
|
55
|
+
footerRightTitle: 'Soporte',
|
|
56
|
+
// Legal links
|
|
57
|
+
aboutUs: 'Nosotros',
|
|
58
|
+
privacyPolicy: 'Política de privacidad',
|
|
59
|
+
termsConditions: 'Términos y condiciones',
|
|
60
|
+
// Support links
|
|
61
|
+
contactSupport: 'Contactar a soporte',
|
|
62
|
+
faq: 'Preguntas frecuentes',
|
|
63
|
+
feedback: 'Feedback',
|
|
64
|
+
},
|
|
65
|
+
en: {
|
|
66
|
+
// Footer titles
|
|
67
|
+
footerLeftTitle: 'Stay in touch',
|
|
68
|
+
footerRightTitle: 'Support',
|
|
69
|
+
// Legal links
|
|
70
|
+
aboutUs: 'About Us',
|
|
71
|
+
privacyPolicy: 'Privacy Policy',
|
|
72
|
+
termsConditions: 'Terms & Conditions',
|
|
73
|
+
// Support links
|
|
74
|
+
contactSupport: 'Contact Support',
|
|
75
|
+
faq: 'FAQ',
|
|
76
|
+
feedback: 'Feedback',
|
|
77
|
+
},
|
|
78
|
+
};
|
|
79
|
+
/**
|
|
80
|
+
* Default language selector configuration for footer
|
|
81
|
+
*/
|
|
82
|
+
export const VALTECH_LANGUAGE_SELECTOR = {
|
|
83
|
+
mode: 'default',
|
|
84
|
+
showFlags: true,
|
|
85
|
+
showLabel: false,
|
|
86
|
+
fill: 'outline',
|
|
87
|
+
size: 'default',
|
|
88
|
+
};
|
|
89
|
+
/**
|
|
90
|
+
* Helper to build footer links from company links config
|
|
91
|
+
* @param links - Array of company links
|
|
92
|
+
* @param t - Translation function (key: string) => string
|
|
93
|
+
* @returns Array of link objects ready for FooterLinksMetadata
|
|
94
|
+
*/
|
|
95
|
+
export function buildFooterLinks(links, t) {
|
|
96
|
+
return links.map(link => ({
|
|
97
|
+
url: link.url,
|
|
98
|
+
text: t(link.key),
|
|
99
|
+
color: 'dark',
|
|
100
|
+
token: '',
|
|
101
|
+
download: false,
|
|
102
|
+
hoverable: true,
|
|
103
|
+
...(link.external ? { target: '_blank' } : {}),
|
|
104
|
+
}));
|
|
105
|
+
}
|
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"company-footer.config.js","sourceRoot":"","sources":["../../../../../src/lib/config/company-footer.config.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH;;GAEG;AACH,MAAM,CAAC,MAAM,oBAAoB,GAAG;IAClC,EAAE,IAAI,EAAE,eAAe,EAAE,GAAG,EAAE,sDAAsD,EAAE,IAAI,EAAE,UAAU,EAAE;IACxG,EAAE,IAAI,EAAE,gBAAgB,EAAE,GAAG,EAAE,wCAAwC,EAAE,IAAI,EAAE,WAAW,EAAE;IAC5F,EAAE,IAAI,EAAE,eAAe,EAAE,GAAG,EAAE,+CAA+C,EAAE,IAAI,EAAE,UAAU,EAAE;IACjG,EAAE,IAAI,EAAE,cAAc,EAAE,GAAG,EAAE,iCAAiC,EAAE,IAAI,EAAE,SAAS,EAAE;IACjF,EAAE,IAAI,EAAE,cAAc,EAAE,GAAG,EAAE,0DAA0D,EAAE,IAAI,EAAE,SAAS,EAAE;IAC1G,EAAE,IAAI,EAAE,aAAa,EAAE,GAAG,EAAE,qCAAqC,EAAE,IAAI,EAAE,QAAQ,EAAE;CACpF,CAAC;AAEF;;;GAGG;AACH,MAAM,CAAC,MAAM,mBAAmB,GAAG;IACjC,IAAI,EAAE,8BAA8B;IACpC,QAAQ,EAAE,6BAA6B;IACvC,OAAO,EAAE,cAAc;IACvB,SAAS,EAAE,GAAG;CACf,CAAC;AAcF;;GAEG;AACH,MAAM,CAAC,MAAM,qBAAqB,GAAG;IACnC,KAAK,EAAE;QACL,EAAE,GAAG,EAAE,SAAS,EAAE,GAAG,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE;QAClD,EAAE,GAAG,EAAE,eAAe,EAAE,GAAG,EAAE,gBAAgB,EAAE,QAAQ,EAAE,KAAK,EAAE;QAChE,EAAE,GAAG,EAAE,iBAAiB,EAAE,GAAG,EAAE,cAAc,EAAE,QAAQ,EAAE,KAAK,EAAE;KAChD;IAClB,OAAO,EAAE;QACP,EAAE,GAAG,EAAE,gBAAgB,EAAE,GAAG,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE;QAC3D,EAAE,GAAG,EAAE,KAAK,EAAE,GAAG,EAAE,WAAW,EAAE,QAAQ,EAAE,KAAK,EAAE;QACjD,EAAE,GAAG,EAAE,UAAU,EAAE,GAAG,EAAE,WAAW,EAAE,QAAQ,EAAE,KAAK,EAAE;KACtC;CACnB,CAAC;AAEF;;;GAGG;AACH,MAAM,CAAC,MAAM,mBAAmB,GAAG;IACjC,EAAE,EAAE;QACF,gBAAgB;QAChB,eAAe,EAAE,qBAAqB;QACtC,gBAAgB,EAAE,SAAS;QAC3B,cAAc;QACd,OAAO,EAAE,UAAU;QACnB,aAAa,EAAE,wBAAwB;QACvC,eAAe,EAAE,wBAAwB;QACzC,gBAAgB;QAChB,cAAc,EAAE,qBAAqB;QACrC,GAAG,EAAE,sBAAsB;QAC3B,QAAQ,EAAE,UAAU;KACrB;IACD,EAAE,EAAE;QACF,gBAAgB;QAChB,eAAe,EAAE,eAAe;QAChC,gBAAgB,EAAE,SAAS;QAC3B,cAAc;QACd,OAAO,EAAE,UAAU;QACnB,aAAa,EAAE,gBAAgB;QAC/B,eAAe,EAAE,oBAAoB;QACrC,gBAAgB;QAChB,cAAc,EAAE,iBAAiB;QACjC,GAAG,EAAE,KAAK;QACV,QAAQ,EAAE,UAAU;KACrB;CACF,CAAC;AAEF;;GAEG;AACH,MAAM,CAAC,MAAM,yBAAyB,GAAG;IACvC,IAAI,EAAE,SAAkB;IACxB,SAAS,EAAE,IAAI;IACf,SAAS,EAAE,KAAK;IAChB,IAAI,EAAE,SAAkB;IACxB,IAAI,EAAE,SAAkB;CACzB,CAAC;AAEF;;;;;GAKG;AACH,MAAM,UAAU,gBAAgB,CAC9B,KAAoB,EACpB,CAA0B;IAU1B,OAAO,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACxB,GAAG,EAAE,IAAI,CAAC,GAAG;QACb,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;QACjB,KAAK,EAAE,MAAM;QACb,KAAK,EAAE,EAAE;QACT,QAAQ,EAAE,KAAK;QACf,SAAS,EAAE,IAAI;QACf,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;KAC/C,CAAC,CAAC,CAAC;AACN,CAAC","sourcesContent":["/**\n * Shared Company Footer Configuration\n *\n * This configuration is shared across all Valtech products:\n * - ui-docs\n * - showcase\n * - myvaltech (company website)\n *\n * When links or social profiles change, update here and publish a new version.\n */\n\n/**\n * Social media links for Valtech\n */\nexport const VALTECH_SOCIAL_LINKS = [\n  { icon: 'logo-facebook', url: 'https://m.facebook.com/profile.php?id=61557610734470', name: 'Facebook' },\n  { icon: 'logo-instagram', url: 'https://www.instagram.com/valtechltda/', name: 'Instagram' },\n  { icon: 'logo-linkedin', url: 'https://www.linkedin.com/company/valtechltda/', name: 'LinkedIn' },\n  { icon: 'logo-twitter', url: 'https://twitter.com/valtechltda', name: 'Twitter' },\n  { icon: 'logo-youtube', url: 'https://www.youtube.com/channel/UCuF4FGdTiUXxANx1HS4Wi5Q', name: 'YouTube' },\n  { icon: 'logo-tiktok', url: 'https://www.tiktok.com/@valtechltda', name: 'TikTok' },\n];\n\n/**\n * Footer logo configuration\n * Uses CSS variables for theme-aware logo switching\n */\nexport const VALTECH_FOOTER_LOGO = {\n  logo: 'assets/images/main-light.svg',\n  logoDark: 'assets/images/main-dark.svg',\n  logoAlt: 'Valtech Logo',\n  logoRoute: '/',\n};\n\n/**\n * Company link definition\n */\nexport interface CompanyLink {\n  /** i18n key for the link text */\n  key: string;\n  /** URL path or external URL */\n  url: string;\n  /** Whether the link opens in a new tab */\n  external?: boolean;\n}\n\n/**\n * Company links organized by section\n */\nexport const VALTECH_COMPANY_LINKS = {\n  legal: [\n    { key: 'aboutUs', url: '/about', external: false },\n    { key: 'privacyPolicy', url: '/legal/privacy', external: false },\n    { key: 'termsConditions', url: '/legal/terms', external: false },\n  ] as CompanyLink[],\n  support: [\n    { key: 'contactSupport', url: '/contact', external: false },\n    { key: 'faq', url: '/help/faq', external: false },\n    { key: 'feedback', url: '/feedback', external: false },\n  ] as CompanyLink[],\n};\n\n/**\n * i18n keys for the footer (Layout namespace)\n * These should be registered in each app's i18n config\n */\nexport const VALTECH_FOOTER_I18N = {\n  es: {\n    // Footer titles\n    footerLeftTitle: 'Sigamos en contacto',\n    footerRightTitle: 'Soporte',\n    // Legal links\n    aboutUs: 'Nosotros',\n    privacyPolicy: 'Política de privacidad',\n    termsConditions: 'Términos y condiciones',\n    // Support links\n    contactSupport: 'Contactar a soporte',\n    faq: 'Preguntas frecuentes',\n    feedback: 'Feedback',\n  },\n  en: {\n    // Footer titles\n    footerLeftTitle: 'Stay in touch',\n    footerRightTitle: 'Support',\n    // Legal links\n    aboutUs: 'About Us',\n    privacyPolicy: 'Privacy Policy',\n    termsConditions: 'Terms & Conditions',\n    // Support links\n    contactSupport: 'Contact Support',\n    faq: 'FAQ',\n    feedback: 'Feedback',\n  },\n};\n\n/**\n * Default language selector configuration for footer\n */\nexport const VALTECH_LANGUAGE_SELECTOR = {\n  mode: 'default' as const,\n  showFlags: true,\n  showLabel: false,\n  fill: 'outline' as const,\n  size: 'default' as const,\n};\n\n/**\n * Helper to build footer links from company links config\n * @param links - Array of company links\n * @param t - Translation function (key: string) => string\n * @returns Array of link objects ready for FooterLinksMetadata\n */\nexport function buildFooterLinks(\n  links: CompanyLink[],\n  t: (key: string) => string\n): Array<{\n  url: string;\n  text: string;\n  color: string;\n  token: string;\n  download: boolean;\n  hoverable: boolean;\n  target?: string;\n}> {\n  return links.map(link => ({\n    url: link.url,\n    text: t(link.key),\n    color: 'dark',\n    token: '',\n    download: false,\n    hoverable: true,\n    ...(link.external ? { target: '_blank' } : {}),\n  }));\n}\n"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared configuration exports
|
|
3
|
+
*/
|
|
4
|
+
export * from './company-footer.config';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbmZpZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUNILGNBQWMseUJBQXlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNoYXJlZCBjb25maWd1cmF0aW9uIGV4cG9ydHNcbiAqL1xuZXhwb3J0ICogZnJvbSAnLi9jb21wYW55LWZvb3Rlci5jb25maWcnO1xuIl19
|