valtech-components 2.0.422 → 2.0.424
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/README.md +23 -236
- package/esm2022/lib/components/templates/page-content/page-content.component.mjs +12 -8
- package/esm2022/lib/components/templates/page-wrapper/page-wrapper.component.mjs +12 -8
- package/esm2022/lib/components/types.mjs +1 -1
- package/fesm2022/valtech-components.mjs +22 -14
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/templates/page-content/page-content.component.d.ts +33 -2
- package/lib/components/templates/page-wrapper/page-wrapper.component.d.ts +33 -2
- package/lib/components/types.d.ts +1 -1
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -106,7 +106,7 @@ Import the desired standalone components in your Angular modules or components:
|
|
|
106
106
|
|
|
107
107
|
```typescript
|
|
108
108
|
import { inject } from '@angular/core';
|
|
109
|
-
import { TextComponent, ButtonComponent
|
|
109
|
+
import { TextComponent, ButtonComponent } from 'valtech-components';
|
|
110
110
|
|
|
111
111
|
@Component({
|
|
112
112
|
selector: 'your-component',
|
|
@@ -117,33 +117,19 @@ import { TextComponent, ButtonComponent, LangService } from 'valtech-components'
|
|
|
117
117
|
],
|
|
118
118
|
template: `
|
|
119
119
|
<!-- Static text -->
|
|
120
|
-
<val-text
|
|
120
|
+
<val-text
|
|
121
121
|
[props]="{
|
|
122
|
-
content: 'This is a body text.',
|
|
123
|
-
color: 'primary',
|
|
124
|
-
bold: false,
|
|
125
|
-
size: 'small'
|
|
122
|
+
content: 'This is a body text.',
|
|
123
|
+
color: 'primary',
|
|
124
|
+
bold: false,
|
|
125
|
+
size: 'small'
|
|
126
126
|
}"
|
|
127
127
|
/>
|
|
128
|
-
|
|
129
|
-
<!--
|
|
130
|
-
<val-text
|
|
131
|
-
[props]="{
|
|
132
|
-
contentKey: 'save',
|
|
133
|
-
contentClass: '_global',
|
|
134
|
-
contentFallback: 'Save',
|
|
135
|
-
color: 'primary',
|
|
136
|
-
bold: true,
|
|
137
|
-
size: 'medium'
|
|
138
|
-
}"
|
|
139
|
-
/>
|
|
140
|
-
|
|
141
|
-
<!-- Button with reactive text -->
|
|
128
|
+
|
|
129
|
+
<!-- Button -->
|
|
142
130
|
<val-button
|
|
143
131
|
[props]="{
|
|
144
|
-
|
|
145
|
-
contentClass: '_global',
|
|
146
|
-
contentFallback: 'Cancel',
|
|
132
|
+
text: 'Cancel',
|
|
147
133
|
color: 'secondary',
|
|
148
134
|
type: 'button'
|
|
149
135
|
}"
|
|
@@ -152,238 +138,41 @@ import { TextComponent, ButtonComponent, LangService } from 'valtech-components'
|
|
|
152
138
|
`
|
|
153
139
|
})
|
|
154
140
|
export class YourComponent {
|
|
155
|
-
langService = inject(LangService);
|
|
156
|
-
|
|
157
141
|
handleCancel() {
|
|
158
142
|
console.log('Cancel clicked');
|
|
159
143
|
}
|
|
160
144
|
}
|
|
161
145
|
```
|
|
162
146
|
|
|
163
|
-
## 🌍
|
|
164
|
-
|
|
165
|
-
The library includes a powerful reactive internationalization system with **LangService** that automatically updates content when the language changes. It supports both **global content** (reusable across components) and **component-specific content**.
|
|
147
|
+
## 🌍 Locale Management
|
|
166
148
|
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
- **Global Content**: Reusable texts like buttons ("Save", "Cancel"), states ("Loading", "Error"), and confirmations
|
|
170
|
-
- **Component-Specific Content**: Custom texts for individual components
|
|
171
|
-
- **Reactive Updates**: Text automatically updates when language changes
|
|
172
|
-
- **Interpolation Support**: Dynamic values in text strings with `{{variable}}` syntax
|
|
173
|
-
- **Type-Safe**: Full TypeScript support with autocompletion
|
|
174
|
-
- **Simple API**: Direct LangService injection, no complex setup
|
|
175
|
-
- **Automatic Fallbacks**: Intelligent language fallback with console warnings
|
|
176
|
-
- **Unified System**: Single service for all content needs
|
|
177
|
-
|
|
178
|
-
### Available Global Content
|
|
179
|
-
|
|
180
|
-
- **Buttons**: `ok`, `cancel`, `save`, `delete`, `edit`, `close`, `back`, `next`, `previous`
|
|
181
|
-
- **Actions**: `add`, `remove`, `search`, `filter`, `sort`, `refresh`
|
|
182
|
-
- **States**: `loading`, `error`, `success`, `warning`, `info`, `noData`
|
|
183
|
-
- **Forms**: `searchPlaceholder`
|
|
184
|
-
- **Confirmations**: `deleteConfirmation`, `unsavedChanges`
|
|
185
|
-
- **General**: `language`, `areYouSure`, `copied`
|
|
186
|
-
|
|
187
|
-
### Basic Setup
|
|
149
|
+
The library includes a simple locale management service using localStorage:
|
|
188
150
|
|
|
189
151
|
```typescript
|
|
190
|
-
// main.ts
|
|
191
|
-
import { TextContent, ValtechConfig, ValtechConfigService } from 'valtech-components';
|
|
192
|
-
|
|
193
|
-
const appContent = new TextContent({
|
|
194
|
-
es: {
|
|
195
|
-
welcome: 'Bienvenido',
|
|
196
|
-
description: 'Esta es la descripción'
|
|
197
|
-
},
|
|
198
|
-
en: {
|
|
199
|
-
welcome: 'Welcome',
|
|
200
|
-
description: 'This is the description'
|
|
201
|
-
}
|
|
202
|
-
});
|
|
203
|
-
|
|
204
|
-
const valtechConfig: ValtechConfig = {
|
|
205
|
-
content: { AppComponent: appContent }
|
|
206
|
-
};
|
|
207
|
-
|
|
208
|
-
bootstrapApplication(AppComponent, {
|
|
209
|
-
providers: [
|
|
210
|
-
{ provide: ValtechConfigService, useValue: valtechConfig },
|
|
211
|
-
// ... other providers
|
|
212
|
-
]
|
|
213
|
-
});
|
|
214
|
-
```
|
|
215
|
-
|
|
216
|
-
### Using LangService (Current API)
|
|
217
|
-
|
|
218
|
-
```typescript
|
|
219
|
-
// component.ts
|
|
220
152
|
import { inject } from '@angular/core';
|
|
221
|
-
import {
|
|
153
|
+
import { LocaleService, LanguageSelectorComponent } from 'valtech-components';
|
|
222
154
|
|
|
223
155
|
@Component({
|
|
224
156
|
template: `
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
color: 'primary',
|
|
230
|
-
size: 'large',
|
|
231
|
-
bold: true
|
|
232
|
-
}"></val-text>
|
|
233
|
-
|
|
234
|
-
<!-- Component-specific content using val-text -->
|
|
235
|
-
<val-text [props]="{
|
|
236
|
-
contentKey: 'welcome',
|
|
237
|
-
contentClass: 'AppComponent',
|
|
238
|
-
contentFallback: 'Welcome',
|
|
239
|
-
color: 'secondary',
|
|
240
|
-
size: 'medium'
|
|
241
|
-
}"></val-text>
|
|
242
|
-
|
|
243
|
-
<!-- Direct observable binding -->
|
|
244
|
-
<h2>{{ title$ | async }}</h2>
|
|
245
|
-
|
|
246
|
-
<!-- With interpolation -->
|
|
247
|
-
<p>{{ greeting$ | async }}</p>
|
|
157
|
+
<val-language-selector
|
|
158
|
+
[props]="{ showFlags: true, showLabel: true }"
|
|
159
|
+
(languageChange)="onLanguageChange($event)"
|
|
160
|
+
></val-language-selector>
|
|
248
161
|
`
|
|
249
162
|
})
|
|
250
163
|
export class MyComponent {
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
// Global content
|
|
254
|
-
saveButton$ = this.langService.getContent('_global', 'save', 'Save');
|
|
255
|
-
cancelButton$ = this.langService.getContent('_global', 'cancel', 'Cancel');
|
|
256
|
-
|
|
257
|
-
// Component-specific content
|
|
258
|
-
title$ = this.langService.getContent('AppComponent', 'welcome', 'Welcome');
|
|
259
|
-
|
|
260
|
-
// Content with interpolation
|
|
261
|
-
greeting$ = this.langService.getContentWithInterpolation(
|
|
262
|
-
'AppComponent',
|
|
263
|
-
'personalizedGreeting',
|
|
264
|
-
{ name: 'John', count: 5 },
|
|
265
|
-
'Hello!'
|
|
266
|
-
);
|
|
267
|
-
|
|
268
|
-
// Switch language
|
|
269
|
-
changeLanguage() {
|
|
270
|
-
const currentLang = this.langService.currentLang;
|
|
271
|
-
this.langService.setLang(currentLang === 'es' ? 'en' : 'es');
|
|
272
|
-
}
|
|
273
|
-
|
|
274
|
-
constructor() {}
|
|
275
|
-
}
|
|
276
|
-
```
|
|
164
|
+
localeService = inject(LocaleService);
|
|
277
165
|
|
|
278
|
-
|
|
166
|
+
// Get current locale
|
|
167
|
+
currentLocale = this.localeService.getLocale(); // 'es', 'en', etc.
|
|
279
168
|
|
|
280
|
-
|
|
281
|
-
//
|
|
282
|
-
|
|
283
|
-
langService = inject(LangService);
|
|
284
|
-
|
|
285
|
-
// Method 1: Direct service usage
|
|
286
|
-
greeting$ = this.langService.getContentWithInterpolation(
|
|
287
|
-
'UserComponent',
|
|
288
|
-
'personalizedGreeting',
|
|
289
|
-
{ name: 'John', count: 5 },
|
|
290
|
-
'Hello!'
|
|
291
|
-
);
|
|
292
|
-
// Results in: "Hello John, you have 5 messages"
|
|
293
|
-
|
|
294
|
-
// Method 2: Using val-text component
|
|
295
|
-
// In template:
|
|
296
|
-
// <val-text [props]="{
|
|
297
|
-
// contentKey: 'personalizedGreeting',
|
|
298
|
-
// contentClass: 'UserComponent',
|
|
299
|
-
// contentInterpolation: { name: 'John', count: 5 },
|
|
300
|
-
// contentFallback: 'Hello!'
|
|
301
|
-
// }"></val-text>
|
|
302
|
-
}
|
|
303
|
-
```
|
|
304
|
-
|
|
305
|
-
### Helper Functions
|
|
306
|
-
|
|
307
|
-
```typescript
|
|
308
|
-
// Using utility helpers from simple-content
|
|
309
|
-
import {
|
|
310
|
-
fromContentWithInterpolation,
|
|
311
|
-
interpolateStaticContent,
|
|
312
|
-
shouldUseReactiveContent
|
|
313
|
-
} from 'valtech-components';
|
|
314
|
-
|
|
315
|
-
export class AdvancedComponent {
|
|
316
|
-
langService = inject(LangService);
|
|
317
|
-
|
|
318
|
-
// Using helper for reactive content with interpolation
|
|
319
|
-
message$ = fromContentWithInterpolation(this.langService, {
|
|
320
|
-
contentClass: 'MyComponent',
|
|
321
|
-
contentKey: 'message',
|
|
322
|
-
contentInterpolation: { count: 10 }
|
|
323
|
-
});
|
|
324
|
-
|
|
325
|
-
// Static content with interpolation
|
|
326
|
-
processStaticText() {
|
|
327
|
-
const text = interpolateStaticContent(
|
|
328
|
-
'You have {{count}} items',
|
|
329
|
-
{ count: 5 }
|
|
330
|
-
);
|
|
331
|
-
// Results in: "You have 5 items"
|
|
169
|
+
onLanguageChange(locale: string) {
|
|
170
|
+
// Changes locale and reloads the page
|
|
171
|
+
this.localeService.setLocale(locale);
|
|
332
172
|
}
|
|
333
173
|
}
|
|
334
174
|
```
|
|
335
175
|
|
|
336
|
-
### Migration Guide
|
|
337
|
-
|
|
338
|
-
If you're migrating from the old `ContentService` API, here's a quick reference:
|
|
339
|
-
|
|
340
|
-
```typescript
|
|
341
|
-
// OLD API (ContentService - removed)
|
|
342
|
-
content = inject(ContentService);
|
|
343
|
-
text$ = this.content.fromContent({ key: 'save' });
|
|
344
|
-
|
|
345
|
-
// NEW API (LangService - current)
|
|
346
|
-
langService = inject(LangService);
|
|
347
|
-
text$ = this.langService.getContent('_global', 'save', 'Save');
|
|
348
|
-
|
|
349
|
-
// OLD API (with interpolation)
|
|
350
|
-
text$ = this.content.fromContentWithInterpolation({
|
|
351
|
-
key: 'greeting',
|
|
352
|
-
interpolation: { name: 'John' }
|
|
353
|
-
});
|
|
354
|
-
|
|
355
|
-
// NEW API (with interpolation)
|
|
356
|
-
text$ = this.langService.getContentWithInterpolation(
|
|
357
|
-
'_global',
|
|
358
|
-
'greeting',
|
|
359
|
-
{ name: 'John' },
|
|
360
|
-
'Hello!'
|
|
361
|
-
);
|
|
362
|
-
```
|
|
363
|
-
|
|
364
|
-
### Available Utility Functions
|
|
365
|
-
|
|
366
|
-
```typescript
|
|
367
|
-
import {
|
|
368
|
-
shouldUseReactiveContent,
|
|
369
|
-
extractContentConfig,
|
|
370
|
-
interpolateStaticContent,
|
|
371
|
-
fromContentWithInterpolation
|
|
372
|
-
} from 'valtech-components';
|
|
373
|
-
|
|
374
|
-
// Check if metadata should use reactive content
|
|
375
|
-
const useReactive = shouldUseReactiveContent(metadata);
|
|
376
|
-
|
|
377
|
-
// Extract configuration from metadata
|
|
378
|
-
const config = extractContentConfig(metadata);
|
|
379
|
-
|
|
380
|
-
// Interpolate static strings
|
|
381
|
-
const result = interpolateStaticContent('Hello {{name}}!', { name: 'World' });
|
|
382
|
-
|
|
383
|
-
// Helper for reactive content with interpolation
|
|
384
|
-
const content$ = fromContentWithInterpolation(langService, metadata);
|
|
385
|
-
```
|
|
386
|
-
|
|
387
176
|
## 🔗 Automatic Link Processing
|
|
388
177
|
|
|
389
178
|
The library now includes automatic link detection and processing capabilities. Convert URLs and internal routes in text content into clickable links automatically:
|
|
@@ -453,15 +242,13 @@ processText(text: string) {
|
|
|
453
242
|
- `src/lib/components/templates/` – Page templates and layouts
|
|
454
243
|
- `src/lib/components/styles/` – Shared SCSS variables, mixins, and utilities
|
|
455
244
|
- `src/lib/services/` – Core services:
|
|
456
|
-
- `
|
|
245
|
+
- `locale.service.ts` – Simple locale management with localStorage
|
|
457
246
|
- `theme.service.ts` – Theme management
|
|
458
247
|
- `navigation.service.ts` – Routing utilities
|
|
459
248
|
- `download.service.ts` – File download helpers
|
|
460
249
|
- `icons.service.ts` – Icon management
|
|
461
250
|
- `toast.service.ts` – Toast notifications
|
|
462
251
|
- `src/lib/shared/` – Shared utilities:
|
|
463
|
-
- `utils/simple-content.ts` – Content system helpers and interfaces
|
|
464
|
-
- `utils/content.ts` – Legacy content utilities (for backward compatibility)
|
|
465
252
|
- `pipes/process-links.pipe.ts` – Automatic link processing
|
|
466
253
|
- `constants/` – Shared constants
|
|
467
254
|
|
|
@@ -45,12 +45,10 @@ export class PageContentComponent {
|
|
|
45
45
|
* Emits when a header action is clicked.
|
|
46
46
|
*/
|
|
47
47
|
this.onHeaderClick = new EventEmitter();
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
get headerProps() {
|
|
53
|
-
return (this.props.header || {
|
|
48
|
+
/**
|
|
49
|
+
* Default header configuration (cached to avoid infinite change detection).
|
|
50
|
+
*/
|
|
51
|
+
this.defaultHeader = {
|
|
54
52
|
bordered: true,
|
|
55
53
|
translucent: true,
|
|
56
54
|
toolbar: {
|
|
@@ -79,7 +77,13 @@ export class PageContentComponent {
|
|
|
79
77
|
},
|
|
80
78
|
],
|
|
81
79
|
},
|
|
82
|
-
}
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Gets header props, using cached default if not provided.
|
|
84
|
+
*/
|
|
85
|
+
get headerProps() {
|
|
86
|
+
return this.props.header || this.defaultHeader;
|
|
83
87
|
}
|
|
84
88
|
/**
|
|
85
89
|
* Gets the background color based on theme.
|
|
@@ -153,4 +157,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
153
157
|
}], onHeaderClick: [{
|
|
154
158
|
type: Output
|
|
155
159
|
}] } });
|
|
156
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
160
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-content/page-content.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { IonContent } from '@ionic/angular/standalone';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageContentMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-content\n *\n * A page content template with corporate header, main content area,\n * and footer slots. Supports dark mode and customizable backgrounds.\n *\n * @example\n * <val-page-content\n *   [props]=\"{\n *     header: { ... },\n *     background: '--main-background',\n *     homeRoute: '/'\n *   }\"\n *   (onHeaderClick)=\"handleHeaderAction($event)\"\n * >\n *   <div content>\n *     <!-- Main page content -->\n *   </div>\n *   <div footer>\n *     <val-company-footer [props]=\"footerProps\"></val-company-footer>\n *   </div>\n * </val-page-content>\n *\n * @input props - Page content configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-content',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <ng-content select=\"[content]\"></ng-content>\n        </main>\n        <ng-content select=\"[footer]\"></ng-content>\n      </ion-content>\n      <ng-content select=\"[extra-footer]\"></ng-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageContentComponent {\n  private theme = inject(ThemeService);\n  private nav = inject(NavigationService);\n\n  /**\n   * Page content configuration.\n   */\n  @Input() props: PageContentMetadata = {};\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      actions: [\n        {\n          token: 'header-logo',\n          description: '',\n          position: 'left' as const,\n          type: 'IMAGE' as const,\n          image: {\n            width: 10,\n            src: '--main-logo',\n            alt: 'header logo',\n            mode: 'box' as const,\n            shaded: false,\n            bordered: false,\n            size: 'small' as const,\n            limited: false,\n            flex: true,\n          },\n        },\n      ],\n    },\n  };\n\n  /**\n   * Gets header props, using cached default if not provided.\n   */\n  get headerProps() {\n    return this.props.header || this.defaultHeader;\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    // Navigate to home route if configured and logo was clicked\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
|
|
@@ -46,12 +46,10 @@ export class PageWrapperComponent {
|
|
|
46
46
|
* Emits when a header action is clicked.
|
|
47
47
|
*/
|
|
48
48
|
this.onHeaderClick = new EventEmitter();
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
get headerProps() {
|
|
54
|
-
return (this.props.header || {
|
|
49
|
+
/**
|
|
50
|
+
* Default header configuration (cached to avoid infinite change detection).
|
|
51
|
+
*/
|
|
52
|
+
this.defaultHeader = {
|
|
55
53
|
bordered: true,
|
|
56
54
|
translucent: true,
|
|
57
55
|
toolbar: {
|
|
@@ -80,7 +78,13 @@ export class PageWrapperComponent {
|
|
|
80
78
|
},
|
|
81
79
|
],
|
|
82
80
|
},
|
|
83
|
-
}
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Gets header props, using cached default if not provided.
|
|
85
|
+
*/
|
|
86
|
+
get headerProps() {
|
|
87
|
+
return this.props.header || this.defaultHeader;
|
|
84
88
|
}
|
|
85
89
|
ngOnInit() {
|
|
86
90
|
if (this.props.scrollToTopOnNavigate !== false) {
|
|
@@ -192,4 +196,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
192
196
|
}], onHeaderClick: [{
|
|
193
197
|
type: Output
|
|
194
198
|
}] } });
|
|
195
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-wrapper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-wrapper/page-wrapper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { NavigationEnd, Router, RouterOutlet } from '@angular/router';\nimport { IonContent, ViewWillEnter, ViewWillLeave } from '@ionic/angular/standalone';\nimport { filter, Subscription } from 'rxjs';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { CompanyFooterComponent } from '../../organisms/company-footer/company-footer.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageWrapperMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-wrapper\n *\n * A complete page wrapper template with header, router outlet, footer,\n * and automatic scroll-to-top on navigation.\n *\n * @example\n * <val-page-wrapper\n *   [props]=\"{\n *     homeRoute: '/',\n *     scrollToTopOnNavigate: true,\n *     footer: {\n *       links: { ... },\n *       rights: { companyName: 'Acme' }\n *     }\n *   }\"\n * ></val-page-wrapper>\n *\n * @input props - Page wrapper configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-wrapper',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, CompanyFooterComponent, RouterOutlet, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [id]=\"props.contentId || 'page-wrapper'\"\n        class=\"ion-padding\"\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <router-outlet></router-outlet>\n        </main>\n        @if (props.footer) {\n          <val-company-footer [props]=\"props.footer\" />\n        }\n      </ion-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageWrapperComponent implements ViewWillEnter, ViewWillLeave, OnInit, OnDestroy {\n  private theme = inject(ThemeService);\n  private nav = inject(NavigationService);\n  private router = inject(Router);\n  private routerSubscription?: Subscription;\n\n  /**\n   * Page wrapper configuration.\n   */\n  @Input() props: PageWrapperMetadata = {\n    scrollToTopOnNavigate: true,\n  };\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration.\n   */\n  get headerProps() {\n    return (\n      this.props.header || {\n        bordered: true,\n        translucent: true,\n        toolbar: {\n          withBack: false,\n          withActions: true,\n          textColor: 'dark',\n          withMenu: true,\n          title: '',\n          actions: [\n            {\n              token: 'header-logo',\n              description: '',\n              position: 'left',\n              type: 'IMAGE',\n              image: {\n                width: 10,\n                src: '--main-logo',\n                alt: 'header logo',\n                mode: 'box',\n                shaded: false,\n                bordered: false,\n                size: 'small',\n                limited: false,\n                flex: true,\n              },\n            },\n          ],\n        },\n      }\n    );\n  }\n\n  ngOnInit(): void {\n    if (this.props.scrollToTopOnNavigate !== false) {\n      this.routerSubscription = this.router.events\n        .pipe(filter((event) => event instanceof NavigationEnd))\n        .subscribe(() => {\n          this.scrollToTop();\n        });\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.routerSubscription) {\n      this.routerSubscription.unsubscribe();\n    }\n  }\n\n  ionViewWillLeave(): void {\n    // Optional: can add logic on view leave\n  }\n\n  ionViewWillEnter(): void {\n    this.scrollToTop();\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Scrolls the content to the top.\n   */\n  private scrollToTop(): void {\n    const contentId = this.props.contentId || 'page-wrapper';\n    const ionContent = document.querySelector(`#${contentId}`) as any;\n    if (ionContent && ionContent.scrollToTop) {\n      ionContent.scrollToTop(300);\n    } else {\n      window.scrollTo({ top: 0, behavior: 'smooth' });\n    }\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
|
|
199
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-wrapper.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-wrapper/page-wrapper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { NavigationEnd, Router, RouterOutlet } from '@angular/router';\nimport { IonContent, ViewWillEnter, ViewWillLeave } from '@ionic/angular/standalone';\nimport { filter, Subscription } from 'rxjs';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { CompanyFooterComponent } from '../../organisms/company-footer/company-footer.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageWrapperMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-wrapper\n *\n * A complete page wrapper template with header, router outlet, footer,\n * and automatic scroll-to-top on navigation.\n *\n * @example\n * <val-page-wrapper\n *   [props]=\"{\n *     homeRoute: '/',\n *     scrollToTopOnNavigate: true,\n *     footer: {\n *       links: { ... },\n *       rights: { companyName: 'Acme' }\n *     }\n *   }\"\n * ></val-page-wrapper>\n *\n * @input props - Page wrapper configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-wrapper',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, CompanyFooterComponent, RouterOutlet, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [id]=\"props.contentId || 'page-wrapper'\"\n        class=\"ion-padding\"\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <main>\n          <router-outlet></router-outlet>\n        </main>\n        @if (props.footer) {\n          <val-company-footer [props]=\"props.footer\" />\n        }\n      </ion-content>\n    </div>\n  `,\n  styles: `\n    main {\n      min-height: 60vh;\n    }\n  `,\n})\nexport class PageWrapperComponent implements ViewWillEnter, ViewWillLeave, OnInit, OnDestroy {\n  private theme = inject(ThemeService);\n  private nav = inject(NavigationService);\n  private router = inject(Router);\n  private routerSubscription?: Subscription;\n\n  /**\n   * Page wrapper configuration.\n   */\n  @Input() props: PageWrapperMetadata = {\n    scrollToTopOnNavigate: true,\n  };\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      actions: [\n        {\n          token: 'header-logo',\n          description: '',\n          position: 'left' as const,\n          type: 'IMAGE' as const,\n          image: {\n            width: 10,\n            src: '--main-logo',\n            alt: 'header logo',\n            mode: 'box' as const,\n            shaded: false,\n            bordered: false,\n            size: 'small' as const,\n            limited: false,\n            flex: true,\n          },\n        },\n      ],\n    },\n  };\n\n  /**\n   * Gets header props, using cached default if not provided.\n   */\n  get headerProps() {\n    return this.props.header || this.defaultHeader;\n  }\n\n  ngOnInit(): void {\n    if (this.props.scrollToTopOnNavigate !== false) {\n      this.routerSubscription = this.router.events\n        .pipe(filter((event) => event instanceof NavigationEnd))\n        .subscribe(() => {\n          this.scrollToTop();\n        });\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.routerSubscription) {\n      this.routerSubscription.unsubscribe();\n    }\n  }\n\n  ionViewWillLeave(): void {\n    // Optional: can add logic on view leave\n  }\n\n  ionViewWillEnter(): void {\n    this.scrollToTop();\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Scrolls the content to the top.\n   */\n  private scrollToTop(): void {\n    const contentId = this.props.contentId || 'page-wrapper';\n    const ionContent = document.querySelector(`#${contentId}`) as any;\n    if (ionContent && ionContent.scrollToTop) {\n      ionContent.scrollToTop(300);\n    } else {\n      window.scrollTo({ top: 0, behavior: 'smooth' });\n    }\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
|
|
@@ -55,4 +55,4 @@ export var ToolbarActionType;
|
|
|
55
55
|
ToolbarActionType["IMAGE"] = "IMAGE";
|
|
56
56
|
ToolbarActionType["BUTTON"] = "BUTTON";
|
|
57
57
|
})(ToolbarActionType || (ToolbarActionType = {}));
|
|
58
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/components/types.ts"],"names":[],"mappings":"AAQA,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,QAAQ,GAAmB,UAAU,CAAC;AAC5C,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,KAAK,GAAmB,OAAO,CAAC;AAEtC;;GAEG;AACH,MAAM,CAAC,MAAM,eAAe,GAAG,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;AAErE;;GAEG;AACH,MAAM,CAAN,IAAY,UAMX;AAND,WAAY,UAAU;IACpB,iEAAe,CAAA;IACf,mEAAgB,CAAA;IAChB,iEAAe,CAAA;IACf,+DAAc,CAAA;IACd,uEAAkB,CAAA;AACpB,CAAC,EANW,UAAU,KAAV,UAAU,QAMrB;AAcD;;GAEG;AACH,MAAM,CAAN,IAAY,SAuBX;AAvBD,WAAY,SAAS;IACnB,yCAAI,CAAA;IACJ,iDAAQ,CAAA;IACR,2CAAK,CAAA;IACL,iDAAQ,CAAA;IACR,+CAAO,CAAA;IACP,6CAAM,CAAA;IACN,6DAAc,CAAA;IACd,iDAAQ,CAAA;IACR,yCAAI,CAAA;IACJ,qDAAU,CAAA;IACV,0CAAI,CAAA;IACJ,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,8CAAM,CAAA;IACN,4DAAa,CAAA;IACb,0DAAY,CAAA;IACZ,wEAAmB,CAAA;IACnB,0CAAI,CAAA;IACJ,8CAAM,CAAA;IACN,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,kDAAQ,CAAA;AACV,CAAC,EAvBW,SAAS,KAAT,SAAS,QAuBpB;AA6HD;;GAEG;AACH,MAAM,CAAN,IAAY,iBAKX;AALD,WAAY,iBAAiB;IAC3B,sCAAiB,CAAA;IACjB,kCAAa,CAAA;IACb,oCAAe,CAAA;IACf,sCAAiB,CAAA;AACnB,CAAC,EALW,iBAAiB,KAAjB,iBAAiB,QAK5B","sourcesContent":["import { FormControl, ValidatorFn } from '@angular/forms';\nimport { Color } from '@ionic/core';\nimport { ImageMetadata } from './atoms/image/types';\n\n/**\n * Possible states for an interactive component.\n */\nexport type ComponentState = 'ENABLED' | 'DISABLED' | 'WORKING' | 'ERROR';\nconst ENABLED: ComponentState = 'ENABLED';\nconst DISABLED: ComponentState = 'DISABLED';\nconst WORKING: ComponentState = 'WORKING';\nconst ERROR: ComponentState = 'ERROR';\n\n/**\n * Object containing all possible component states.\n */\nexport const ComponentStates = { ENABLED, DISABLED, WORKING, ERROR };\n\n/**\n * Types of actions that a button or link can perform.\n */\nexport enum ActionType {\n  BROWSER_NEW_TAB, // Open in a new browser tab\n  BROWSER_DOWNLOAD, // Download via browser\n  NATIVE_DOWNLOAD, // Download using native capabilities\n  APP_NAVIGATION, // Internal app navigation\n  BROWSER_NAVIGATION, // Navigation in the browser\n}\n\n/**\n * Represents an executable action for a button or link.\n */\nexport type Action = {\n  /** Action description */\n  description: string;\n  /** Action type */\n  type: ActionType;\n  /** Action source or destination */\n  source: string;\n};\n\n/**\n * Supported input types for forms.\n */\nexport enum InputType {\n  TEXT,\n  TEXTAREA,\n  EMAIL,\n  PASSWORD,\n  COMMENT,\n  NUMBER,\n  NUMBER_FROM_TO,\n  PIN_CODE,\n  DATE,\n  DATE_RANGE,\n  HOUR,\n  CHECK,\n  RADIO,\n  SELECT,\n  SEARCH_SELECT,\n  MULTI_SELECT,\n  MULTI_SELECT_SIMPLE,\n  FILE,\n  TOGGLE,\n  RANGE,\n  PHONE,\n  CURRENCY,\n}\n\n/**\n * Option for select, radio, etc. inputs.\n */\nexport type InputOption = {\n  /** Unique option identifier */\n  id: string;\n  /** Display name */\n  name: string;\n  /** Whether the option is selected by default */\n  selected?: boolean;\n  /** Display order */\n  order: number;\n};\n\n/**\n * Metadata for a form field.\n */\nexport type InputMetadata = {\n  /** Associated form control */\n  control: FormControl;\n  /** From control (only for NUMBER_FROM_TO type) */\n  fromControl?: FormControl;\n  /** To control (only for NUMBER_FROM_TO type) */\n  toControl?: FormControl;\n  /** Unique token for the input */\n  token: string;\n  /** Display label */\n  label: string;\n  /** Field name */\n  name: string;\n  /** Help text */\n  hint: string;\n  /** Input placeholder */\n  placeholder: string;\n  /** Input type */\n  type: InputType;\n  /** Display order */\n  order: number;\n  /** Associated validators */\n  validators: ValidatorFn[];\n  /** Options (for select, radio, etc.) */\n  options?: InputOption[];\n  /** Allowed range (for number, date, etc.) */\n  range?: {\n    min: number;\n    max: number;\n  };\n  /** Custom error messages */\n  errors: {\n    [key: string]: string;\n  };\n  /** Initial value for the field */\n  value?: string;\n  /** Default value configuration - string for custom defaults, true for auto defaults */\n  withDefault?: string | boolean;\n  /** Field state */\n  state: ComponentState;\n  /** Label for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromLabel?: string;\n  /** Label for \"to\" field (only for NUMBER_FROM_TO type) */\n  toLabel?: string;\n  /** Placeholder for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromPlaceholder?: string;\n  /** Placeholder for \"to\" field (only for NUMBER_FROM_TO type) */\n  toPlaceholder?: string;\n\n  // i18n support properties\n  /** Key for reactive content from LangService */\n  contentKey?: string;\n  /** Component class name for content lookup */\n  contentClass?: string;\n  /** Fallback text if content key is not found */\n  contentFallback?: string;\n\n  // select-input specific i18n properties\n  /** Custom header text for select modal */\n  modalHeader?: string;\n  /** Custom cancel button text for select modal */\n  cancelText?: string;\n  /** Custom OK button text for select modal */\n  okText?: string;\n\n  // check-input specific properties\n  /** Position of label for checkbox ('start' | 'end') */\n  labelPlacement?: 'start' | 'end';\n};\n\n/**\n * A section in a form, grouping multiple fields.\n */\nexport type FormSection = {\n  /** Section name */\n  name: string;\n  /** Display order */\n  order: number;\n  /** Fields included in the section */\n  fields: InputMetadata[];\n};\n\n/**\n * Data sent when submitting a form.\n */\nexport type FormSubmit = {\n  /** List of fields and their values */\n  fields: { key: string; value: string }[];\n  /** Optional token for the operation */\n  token?: string;\n};\n\n/**\n * Metadata for a complete form.\n */\nexport type FormMetadata = {\n  /** Form name */\n  name: string;\n  /** Form sections */\n  sections: FormSection[];\n  /** Action buttons configuration */\n  actions: ButtonMetadata;\n  /** Global form state */\n  state: ComponentState;\n};\n\n/**\n * Possible action types for a toolbar.\n */\nexport enum ToolbarActionType {\n  AVATAR = 'AVATAR',\n  ICON = 'ICON',\n  IMAGE = 'IMAGE',\n  BUTTON = 'BUTTON',\n}\n\n/**\n * Toolbar action definition.\n */\nexport type ToolbarAction = {\n  /** Action type */\n  type: 'AVATAR' | 'ICON' | 'IMAGE' | 'BUTTON';\n  /** Optional token identifier */\n  token?: string;\n  /** Toolbar position */\n  position: 'left' | 'right' | 'center';\n  /** Optional description */\n  description?: string;\n  /** Associated image (if any) */\n  image?: ImageMetadata;\n};\n\n/**\n * Metadata for an icon.\n */\nexport interface IconMetada {\n  /** Icon name */\n  name: string;\n  /** Icon slot position */\n  slot: 'start' | 'end';\n}\n\n/**\n * Button configuration object.\n * Supports both static and reactive content.\n * @type {ButtonMetadata}\n * @property text - Static button label (takes precedence over textConfig).\n * @property textConfig - Reactive content configuration for button text.\n * @property color - The button color (Ionic color string).\n * @property icon - Icon to display (optional).\n * @property state - Button state (enabled, disabled, working, etc.).\n * @property expand, fill, size, shape, href, target, download, handler, etc. - See ButtonMetadata for all options.\n */\nexport interface ButtonMetadata {\n  /** Associated action type */\n  actionType?: ActionType;\n  /** Button expansion */\n  expand?: 'full' | 'block';\n  /** Associated link */\n  link?: string;\n  /** Associated href link */\n  href?: string;\n  /** Link target */\n  target?: '_blank' | '_self' | '_parent' | '_top';\n  /** Download file name */\n  download?: string;\n  /** Button color */\n  color: Color;\n  /** Button state */\n  state: ComponentState;\n  /** Static display text (takes precedence over reactive content) */\n  text?: string;\n  /** Reactive content configuration for button text */\n  contentKey?: string;\n  /** Component class name for content lookup (required with contentKey) */\n  contentClass?: string;\n  /** Fallback text if contentKey is not found */\n  contentFallback?: string;\n  /** Values to interpolate into the content string */\n  contentInterpolation?: Record<string, string | number>;\n  /** Associated icon */\n  icon?: IconMetada;\n  /** Button shape */\n  shape?: 'round';\n  /** Button size */\n  size?: 'small' | 'default' | 'large';\n  /** Button fill */\n  fill?: 'clear' | 'outline' | 'solid' | 'default';\n  /** Button type */\n  type: 'button' | 'submit' | 'reset';\n  /** Optional token identifier */\n  token?: string;\n  /** Optional reference */\n  ref?: any;\n  /** Action handler */\n  handler?: (value: any) => any | Promise<any>;\n}\n\n/**\n * Configuration for reactive content in val-button component.\n * Use this interface when you only need to specify content-related properties.\n * This follows the same pattern as TextContentConfig for consistency.\n */\nexport interface ButtonContentConfig {\n  contentKey: string;\n  contentClass: string;\n  contentFallback?: string;\n  contentInterpolation?: Record<string, string | number>;\n}\n"]}
|
|
58
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/components/types.ts"],"names":[],"mappings":"AAQA,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,QAAQ,GAAmB,UAAU,CAAC;AAC5C,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,KAAK,GAAmB,OAAO,CAAC;AAEtC;;GAEG;AACH,MAAM,CAAC,MAAM,eAAe,GAAG,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;AAErE;;GAEG;AACH,MAAM,CAAN,IAAY,UAMX;AAND,WAAY,UAAU;IACpB,iEAAe,CAAA;IACf,mEAAgB,CAAA;IAChB,iEAAe,CAAA;IACf,+DAAc,CAAA;IACd,uEAAkB,CAAA;AACpB,CAAC,EANW,UAAU,KAAV,UAAU,QAMrB;AAcD;;GAEG;AACH,MAAM,CAAN,IAAY,SAuBX;AAvBD,WAAY,SAAS;IACnB,yCAAI,CAAA;IACJ,iDAAQ,CAAA;IACR,2CAAK,CAAA;IACL,iDAAQ,CAAA;IACR,+CAAO,CAAA;IACP,6CAAM,CAAA;IACN,6DAAc,CAAA;IACd,iDAAQ,CAAA;IACR,yCAAI,CAAA;IACJ,qDAAU,CAAA;IACV,0CAAI,CAAA;IACJ,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,8CAAM,CAAA;IACN,4DAAa,CAAA;IACb,0DAAY,CAAA;IACZ,wEAAmB,CAAA;IACnB,0CAAI,CAAA;IACJ,8CAAM,CAAA;IACN,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,kDAAQ,CAAA;AACV,CAAC,EAvBW,SAAS,KAAT,SAAS,QAuBpB;AA6HD;;GAEG;AACH,MAAM,CAAN,IAAY,iBAKX;AALD,WAAY,iBAAiB;IAC3B,sCAAiB,CAAA;IACjB,kCAAa,CAAA;IACb,oCAAe,CAAA;IACf,sCAAiB,CAAA;AACnB,CAAC,EALW,iBAAiB,KAAjB,iBAAiB,QAK5B","sourcesContent":["import { FormControl, ValidatorFn } from '@angular/forms';\nimport { Color } from '@ionic/core';\nimport { ImageMetadata } from './atoms/image/types';\n\n/**\n * Possible states for an interactive component.\n */\nexport type ComponentState = 'ENABLED' | 'DISABLED' | 'WORKING' | 'ERROR';\nconst ENABLED: ComponentState = 'ENABLED';\nconst DISABLED: ComponentState = 'DISABLED';\nconst WORKING: ComponentState = 'WORKING';\nconst ERROR: ComponentState = 'ERROR';\n\n/**\n * Object containing all possible component states.\n */\nexport const ComponentStates = { ENABLED, DISABLED, WORKING, ERROR };\n\n/**\n * Types of actions that a button or link can perform.\n */\nexport enum ActionType {\n  BROWSER_NEW_TAB, // Open in a new browser tab\n  BROWSER_DOWNLOAD, // Download via browser\n  NATIVE_DOWNLOAD, // Download using native capabilities\n  APP_NAVIGATION, // Internal app navigation\n  BROWSER_NAVIGATION, // Navigation in the browser\n}\n\n/**\n * Represents an executable action for a button or link.\n */\nexport type Action = {\n  /** Action description */\n  description: string;\n  /** Action type */\n  type: ActionType;\n  /** Action source or destination */\n  source: string;\n};\n\n/**\n * Supported input types for forms.\n */\nexport enum InputType {\n  TEXT,\n  TEXTAREA,\n  EMAIL,\n  PASSWORD,\n  COMMENT,\n  NUMBER,\n  NUMBER_FROM_TO,\n  PIN_CODE,\n  DATE,\n  DATE_RANGE,\n  HOUR,\n  CHECK,\n  RADIO,\n  SELECT,\n  SEARCH_SELECT,\n  MULTI_SELECT,\n  MULTI_SELECT_SIMPLE,\n  FILE,\n  TOGGLE,\n  RANGE,\n  PHONE,\n  CURRENCY,\n}\n\n/**\n * Option for select, radio, etc. inputs.\n */\nexport type InputOption = {\n  /** Unique option identifier */\n  id: string;\n  /** Display name */\n  name: string;\n  /** Whether the option is selected by default */\n  selected?: boolean;\n  /** Display order */\n  order: number;\n};\n\n/**\n * Metadata for a form field.\n */\nexport type InputMetadata = {\n  /** Associated form control */\n  control: FormControl;\n  /** From control (only for NUMBER_FROM_TO type) */\n  fromControl?: FormControl;\n  /** To control (only for NUMBER_FROM_TO type) */\n  toControl?: FormControl;\n  /** Unique token for the input */\n  token: string;\n  /** Display label */\n  label: string;\n  /** Field name */\n  name: string;\n  /** Help text */\n  hint: string;\n  /** Input placeholder */\n  placeholder: string;\n  /** Input type */\n  type: InputType;\n  /** Display order */\n  order: number;\n  /** Associated validators */\n  validators: ValidatorFn[];\n  /** Options (for select, radio, etc.) */\n  options?: InputOption[];\n  /** Allowed range (for number, date, etc.) */\n  range?: {\n    min: number;\n    max: number;\n  };\n  /** Custom error messages */\n  errors: {\n    [key: string]: string;\n  };\n  /** Initial value for the field */\n  value?: string;\n  /** Default value configuration - string for custom defaults, true for auto defaults */\n  withDefault?: string | boolean;\n  /** Field state */\n  state: ComponentState;\n  /** Label for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromLabel?: string;\n  /** Label for \"to\" field (only for NUMBER_FROM_TO type) */\n  toLabel?: string;\n  /** Placeholder for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromPlaceholder?: string;\n  /** Placeholder for \"to\" field (only for NUMBER_FROM_TO type) */\n  toPlaceholder?: string;\n\n  // i18n support properties\n  /** Key for content lookup */\n  contentKey?: string;\n  /** Component class name for content lookup */\n  contentClass?: string;\n  /** Fallback text if content key is not found */\n  contentFallback?: string;\n\n  // select-input specific i18n properties\n  /** Custom header text for select modal */\n  modalHeader?: string;\n  /** Custom cancel button text for select modal */\n  cancelText?: string;\n  /** Custom OK button text for select modal */\n  okText?: string;\n\n  // check-input specific properties\n  /** Position of label for checkbox ('start' | 'end') */\n  labelPlacement?: 'start' | 'end';\n};\n\n/**\n * A section in a form, grouping multiple fields.\n */\nexport type FormSection = {\n  /** Section name */\n  name: string;\n  /** Display order */\n  order: number;\n  /** Fields included in the section */\n  fields: InputMetadata[];\n};\n\n/**\n * Data sent when submitting a form.\n */\nexport type FormSubmit = {\n  /** List of fields and their values */\n  fields: { key: string; value: string }[];\n  /** Optional token for the operation */\n  token?: string;\n};\n\n/**\n * Metadata for a complete form.\n */\nexport type FormMetadata = {\n  /** Form name */\n  name: string;\n  /** Form sections */\n  sections: FormSection[];\n  /** Action buttons configuration */\n  actions: ButtonMetadata;\n  /** Global form state */\n  state: ComponentState;\n};\n\n/**\n * Possible action types for a toolbar.\n */\nexport enum ToolbarActionType {\n  AVATAR = 'AVATAR',\n  ICON = 'ICON',\n  IMAGE = 'IMAGE',\n  BUTTON = 'BUTTON',\n}\n\n/**\n * Toolbar action definition.\n */\nexport type ToolbarAction = {\n  /** Action type */\n  type: 'AVATAR' | 'ICON' | 'IMAGE' | 'BUTTON';\n  /** Optional token identifier */\n  token?: string;\n  /** Toolbar position */\n  position: 'left' | 'right' | 'center';\n  /** Optional description */\n  description?: string;\n  /** Associated image (if any) */\n  image?: ImageMetadata;\n};\n\n/**\n * Metadata for an icon.\n */\nexport interface IconMetada {\n  /** Icon name */\n  name: string;\n  /** Icon slot position */\n  slot: 'start' | 'end';\n}\n\n/**\n * Button configuration object.\n * Supports both static and reactive content.\n * @type {ButtonMetadata}\n * @property text - Static button label (takes precedence over textConfig).\n * @property textConfig - Reactive content configuration for button text.\n * @property color - The button color (Ionic color string).\n * @property icon - Icon to display (optional).\n * @property state - Button state (enabled, disabled, working, etc.).\n * @property expand, fill, size, shape, href, target, download, handler, etc. - See ButtonMetadata for all options.\n */\nexport interface ButtonMetadata {\n  /** Associated action type */\n  actionType?: ActionType;\n  /** Button expansion */\n  expand?: 'full' | 'block';\n  /** Associated link */\n  link?: string;\n  /** Associated href link */\n  href?: string;\n  /** Link target */\n  target?: '_blank' | '_self' | '_parent' | '_top';\n  /** Download file name */\n  download?: string;\n  /** Button color */\n  color: Color;\n  /** Button state */\n  state: ComponentState;\n  /** Static display text (takes precedence over reactive content) */\n  text?: string;\n  /** Reactive content configuration for button text */\n  contentKey?: string;\n  /** Component class name for content lookup (required with contentKey) */\n  contentClass?: string;\n  /** Fallback text if contentKey is not found */\n  contentFallback?: string;\n  /** Values to interpolate into the content string */\n  contentInterpolation?: Record<string, string | number>;\n  /** Associated icon */\n  icon?: IconMetada;\n  /** Button shape */\n  shape?: 'round';\n  /** Button size */\n  size?: 'small' | 'default' | 'large';\n  /** Button fill */\n  fill?: 'clear' | 'outline' | 'solid' | 'default';\n  /** Button type */\n  type: 'button' | 'submit' | 'reset';\n  /** Optional token identifier */\n  token?: string;\n  /** Optional reference */\n  ref?: any;\n  /** Action handler */\n  handler?: (value: any) => any | Promise<any>;\n}\n\n/**\n * Configuration for reactive content in val-button component.\n * Use this interface when you only need to specify content-related properties.\n * This follows the same pattern as TextContentConfig for consistency.\n */\nexport interface ButtonContentConfig {\n  contentKey: string;\n  contentClass: string;\n  contentFallback?: string;\n  contentInterpolation?: Record<string, string | number>;\n}\n"]}
|
|
@@ -19761,12 +19761,10 @@ class PageContentComponent {
|
|
|
19761
19761
|
* Emits when a header action is clicked.
|
|
19762
19762
|
*/
|
|
19763
19763
|
this.onHeaderClick = new EventEmitter();
|
|
19764
|
-
|
|
19765
|
-
|
|
19766
|
-
|
|
19767
|
-
|
|
19768
|
-
get headerProps() {
|
|
19769
|
-
return (this.props.header || {
|
|
19764
|
+
/**
|
|
19765
|
+
* Default header configuration (cached to avoid infinite change detection).
|
|
19766
|
+
*/
|
|
19767
|
+
this.defaultHeader = {
|
|
19770
19768
|
bordered: true,
|
|
19771
19769
|
translucent: true,
|
|
19772
19770
|
toolbar: {
|
|
@@ -19795,7 +19793,13 @@ class PageContentComponent {
|
|
|
19795
19793
|
},
|
|
19796
19794
|
],
|
|
19797
19795
|
},
|
|
19798
|
-
}
|
|
19796
|
+
};
|
|
19797
|
+
}
|
|
19798
|
+
/**
|
|
19799
|
+
* Gets header props, using cached default if not provided.
|
|
19800
|
+
*/
|
|
19801
|
+
get headerProps() {
|
|
19802
|
+
return this.props.header || this.defaultHeader;
|
|
19799
19803
|
}
|
|
19800
19804
|
/**
|
|
19801
19805
|
* Gets the background color based on theme.
|
|
@@ -19906,12 +19910,10 @@ class PageWrapperComponent {
|
|
|
19906
19910
|
* Emits when a header action is clicked.
|
|
19907
19911
|
*/
|
|
19908
19912
|
this.onHeaderClick = new EventEmitter();
|
|
19909
|
-
|
|
19910
|
-
|
|
19911
|
-
|
|
19912
|
-
|
|
19913
|
-
get headerProps() {
|
|
19914
|
-
return (this.props.header || {
|
|
19913
|
+
/**
|
|
19914
|
+
* Default header configuration (cached to avoid infinite change detection).
|
|
19915
|
+
*/
|
|
19916
|
+
this.defaultHeader = {
|
|
19915
19917
|
bordered: true,
|
|
19916
19918
|
translucent: true,
|
|
19917
19919
|
toolbar: {
|
|
@@ -19940,7 +19942,13 @@ class PageWrapperComponent {
|
|
|
19940
19942
|
},
|
|
19941
19943
|
],
|
|
19942
19944
|
},
|
|
19943
|
-
}
|
|
19945
|
+
};
|
|
19946
|
+
}
|
|
19947
|
+
/**
|
|
19948
|
+
* Gets header props, using cached default if not provided.
|
|
19949
|
+
*/
|
|
19950
|
+
get headerProps() {
|
|
19951
|
+
return this.props.header || this.defaultHeader;
|
|
19944
19952
|
}
|
|
19945
19953
|
ngOnInit() {
|
|
19946
19954
|
if (this.props.scrollToTopOnNavigate !== false) {
|