@gnggln/ng-ui-system 1.0.0-alpha.0
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/gnggln-ng-ui-system.mjs +5 -0
- package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
- package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
- package/esm2022/lib/components/accordion/index.mjs +2 -0
- package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
- package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
- package/esm2022/lib/components/base-layout/index.mjs +14 -0
- package/esm2022/lib/components/button/button-area.component.mjs +196 -0
- package/esm2022/lib/components/button/button.component.mjs +164 -0
- package/esm2022/lib/components/button/button.types.mjs +6 -0
- package/esm2022/lib/components/button/index.mjs +16 -0
- package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
- package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
- package/esm2022/lib/components/crud-table/index.mjs +16 -0
- package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
- package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
- package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
- package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
- package/esm2022/lib/components/form-builder/index.mjs +19 -0
- package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
- package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
- package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
- package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
- package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
- package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
- package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
- package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
- package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
- package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
- package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
- package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
- package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
- package/esm2022/lib/components/layout-builder/index.mjs +18 -0
- package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
- package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
- package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
- package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
- package/esm2022/lib/components/modal/index.mjs +4 -0
- package/esm2022/lib/components/modal/modal.component.mjs +139 -0
- package/esm2022/lib/components/modal/modal.service.mjs +194 -0
- package/esm2022/lib/components/modal/modal.types.mjs +6 -0
- package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
- package/esm2022/lib/components/page-header/index.mjs +20 -0
- package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
- package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
- package/esm2022/lib/components/table/index.mjs +2 -0
- package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
- package/esm2022/lib/components/table/table.types.mjs +6 -0
- package/esm2022/lib/core/types/index.mjs +6 -0
- package/esm2022/lib/core/utils/index.mjs +53 -0
- package/esm2022/lib/sources/location-data.opt.json +8942 -0
- package/esm2022/lib/sources/nazioni.opt.json +215 -0
- package/esm2022/public-api.mjs +34 -0
- package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
- package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/accordion/accordion.component.d.ts +118 -0
- package/lib/components/accordion/accordion.types.d.ts +62 -0
- package/lib/components/accordion/index.d.ts +2 -0
- package/lib/components/base-layout/base-layout.component.d.ts +83 -0
- package/lib/components/base-layout/base-layout.types.d.ts +26 -0
- package/lib/components/base-layout/index.d.ts +13 -0
- package/lib/components/button/button-area.component.d.ts +88 -0
- package/lib/components/button/button.component.d.ts +55 -0
- package/lib/components/button/button.types.d.ts +70 -0
- package/lib/components/button/index.d.ts +15 -0
- package/lib/components/crud-table/crud-table.component.d.ts +143 -0
- package/lib/components/crud-table/crud-table.types.d.ts +207 -0
- package/lib/components/crud-table/index.d.ts +15 -0
- package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
- package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
- package/lib/components/form-builder/form-builder.component.d.ts +183 -0
- package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
- package/lib/components/form-builder/index.d.ts +13 -0
- package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
- package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
- package/lib/components/form-builder/services/location.service.d.ts +83 -0
- package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
- package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
- package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
- package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
- package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
- package/lib/components/form-builder/types/condition.types.d.ts +51 -0
- package/lib/components/form-builder/types/field.types.d.ts +288 -0
- package/lib/components/form-builder/types/index.d.ts +5 -0
- package/lib/components/form-builder/types/schema.types.d.ts +227 -0
- package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
- package/lib/components/form-builder/types/validation.types.d.ts +174 -0
- package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
- package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
- package/lib/components/form-builder-editor/index.d.ts +15 -0
- package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
- package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
- package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
- package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
- package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
- package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
- package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
- package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
- package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
- package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
- package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
- package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
- package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
- package/lib/components/layout-builder/index.d.ts +16 -0
- package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
- package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
- package/lib/components/layout-builder/layout.service.d.ts +100 -0
- package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
- package/lib/components/modal/index.d.ts +4 -0
- package/lib/components/modal/modal.component.d.ts +44 -0
- package/lib/components/modal/modal.service.d.ts +93 -0
- package/lib/components/modal/modal.types.d.ts +110 -0
- package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
- package/lib/components/page-header/index.d.ts +16 -0
- package/lib/components/page-header/page-header.component.d.ts +59 -0
- package/lib/components/page-header/page-header.types.d.ts +96 -0
- package/lib/components/table/index.d.ts +2 -0
- package/lib/components/table/paginated-table.component.d.ts +85 -0
- package/lib/components/table/table.types.d.ts +81 -0
- package/lib/core/types/index.d.ts +57 -0
- package/lib/core/utils/index.d.ts +29 -0
- package/package.json +44 -0
- package/public-api.d.ts +22 -0
|
@@ -0,0 +1,218 @@
|
|
|
1
|
+
import { Component, DestroyRef, inject, Input, ChangeDetectionStrategy, ViewEncapsulation, } from '@angular/core';
|
|
2
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
3
|
+
import { Location } from '@angular/common';
|
|
4
|
+
import { ActivatedRoute, NavigationEnd, Router } from '@angular/router';
|
|
5
|
+
import { filter, debounceTime } from 'rxjs/operators';
|
|
6
|
+
import { UiPageHeaderComponent } from '../page-header/page-header.component';
|
|
7
|
+
import { UiBreadcrumbService } from '../page-header/breadcrumb.service';
|
|
8
|
+
import { UiButtonAreaComponent } from '../button/button-area.component';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
/**
|
|
11
|
+
* Full-page layout component that combines a page header (breadcrumbs + title),
|
|
12
|
+
* a main content area via `<ng-content>`, and a sticky footer action bar
|
|
13
|
+
* powered by `UiButtonAreaComponent`.
|
|
14
|
+
*
|
|
15
|
+
* Optionally renders an automatic "back" button in the footer, derived from
|
|
16
|
+
* the breadcrumb trail (navigates to the parent breadcrumb).
|
|
17
|
+
*
|
|
18
|
+
* @selector ui-base-layout
|
|
19
|
+
*
|
|
20
|
+
* @example
|
|
21
|
+
* ```html
|
|
22
|
+
* <ui-base-layout [actions]="pageActions" footerAlign="end">
|
|
23
|
+
* <p>Your page content here</p>
|
|
24
|
+
* </ui-base-layout>
|
|
25
|
+
* ```
|
|
26
|
+
*
|
|
27
|
+
* @example
|
|
28
|
+
* ```html
|
|
29
|
+
* <!-- With back button and custom title -->
|
|
30
|
+
* <ui-base-layout
|
|
31
|
+
* title="Dettaglio utente"
|
|
32
|
+
* [showBackButton]="true"
|
|
33
|
+
* [actions]="[
|
|
34
|
+
* { id: 'save', label: 'Salva', variant: 'primary', icon: 'save', action: save },
|
|
35
|
+
* ]"
|
|
36
|
+
* >
|
|
37
|
+
* <app-user-detail />
|
|
38
|
+
* </ui-base-layout>
|
|
39
|
+
* ```
|
|
40
|
+
*/
|
|
41
|
+
export class UiBaseLayoutComponent {
|
|
42
|
+
constructor() {
|
|
43
|
+
/** Whether to display the page header (breadcrumbs + title). */
|
|
44
|
+
this.showHeader = true;
|
|
45
|
+
/** Override the auto-detected page title. Passed through to `ui-page-header`. */
|
|
46
|
+
this.title = '';
|
|
47
|
+
/** Route path for the Home breadcrumb link. */
|
|
48
|
+
this.homeRoute = '/';
|
|
49
|
+
/** Whether to display the Home link in the breadcrumb trail. */
|
|
50
|
+
this.showHome = true;
|
|
51
|
+
/** When `true`, updates `document.title` on navigation. */
|
|
52
|
+
this.updateDocumentTitle = false;
|
|
53
|
+
/** Suffix appended to `document.title`. */
|
|
54
|
+
this.titleSuffix = '';
|
|
55
|
+
/** Whether to show an automatic "back" button based on breadcrumbs. */
|
|
56
|
+
this.showBackButton = true;
|
|
57
|
+
/** Configuration overrides for the automatic back button. */
|
|
58
|
+
this.backButtonConfig = {};
|
|
59
|
+
/** Array of button descriptors for the footer action area. */
|
|
60
|
+
this.actions = [];
|
|
61
|
+
/** Horizontal alignment of the footer button area. */
|
|
62
|
+
this.footerAlign = 'start';
|
|
63
|
+
/** Gap size between footer buttons. */
|
|
64
|
+
this.footerGap = 'sm';
|
|
65
|
+
this.router = inject(Router);
|
|
66
|
+
this.location = inject(Location);
|
|
67
|
+
this.activatedRoute = inject(ActivatedRoute);
|
|
68
|
+
this.breadcrumbService = inject(UiBreadcrumbService);
|
|
69
|
+
this.destroyRef = inject(DestroyRef);
|
|
70
|
+
/** @internal Current breadcrumbs, updated on each navigation. */
|
|
71
|
+
this.breadcrumbs = [];
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Merged action list (back button + consumer actions).
|
|
75
|
+
* Computed on every change-detection check so it stays in sync
|
|
76
|
+
* with both navigation state (breadcrumbs) and input changes.
|
|
77
|
+
*/
|
|
78
|
+
get mergedActions() {
|
|
79
|
+
const result = [];
|
|
80
|
+
if (this.showBackButton) {
|
|
81
|
+
const backTarget = this.getBackTarget();
|
|
82
|
+
result.push({
|
|
83
|
+
id: '__ui-back',
|
|
84
|
+
label: this.backButtonConfig.label ?? 'Indietro',
|
|
85
|
+
icon: this.backButtonConfig.icon ?? 'arrow-left',
|
|
86
|
+
iconPosition: 'leading',
|
|
87
|
+
variant: this.backButtonConfig.variant ?? 'outline',
|
|
88
|
+
hidden: !backTarget,
|
|
89
|
+
action: () => this.navigateBack(),
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
result.push(...this.actions);
|
|
93
|
+
return result;
|
|
94
|
+
}
|
|
95
|
+
/** Whether the footer should render (at least one non-hidden action exists). */
|
|
96
|
+
get hasVisibleActions() {
|
|
97
|
+
return this.mergedActions.some((a) => !a.hidden);
|
|
98
|
+
}
|
|
99
|
+
ngOnInit() {
|
|
100
|
+
this.refreshBreadcrumbs();
|
|
101
|
+
this.router.events
|
|
102
|
+
.pipe(filter((event) => event instanceof NavigationEnd), debounceTime(50), takeUntilDestroyed(this.destroyRef))
|
|
103
|
+
.subscribe(() => this.refreshBreadcrumbs());
|
|
104
|
+
}
|
|
105
|
+
/** @internal Refreshes breadcrumb state from the current route. */
|
|
106
|
+
refreshBreadcrumbs() {
|
|
107
|
+
this.breadcrumbs = this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);
|
|
108
|
+
}
|
|
109
|
+
/** @internal Determines the back navigation target from breadcrumbs. */
|
|
110
|
+
getBackTarget() {
|
|
111
|
+
if (this.breadcrumbs.length < 2)
|
|
112
|
+
return null;
|
|
113
|
+
return this.breadcrumbs[this.breadcrumbs.length - 2];
|
|
114
|
+
}
|
|
115
|
+
/** @internal Navigates to the parent breadcrumb or falls back to browser history. */
|
|
116
|
+
navigateBack() {
|
|
117
|
+
const target = this.getBackTarget();
|
|
118
|
+
if (target?.url) {
|
|
119
|
+
this.router.navigateByUrl(target.url);
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
this.location.back();
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBaseLayoutComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
126
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiBaseLayoutComponent, isStandalone: true, selector: "ui-base-layout", inputs: { showHeader: "showHeader", title: "title", homeRoute: "homeRoute", showHome: "showHome", updateDocumentTitle: "updateDocumentTitle", titleSuffix: "titleSuffix", showBackButton: "showBackButton", backButtonConfig: "backButtonConfig", actions: "actions", footerAlign: "footerAlign", footerGap: "footerGap" }, host: { classAttribute: "ui-base-layout-host" }, ngImport: i0, template: `
|
|
127
|
+
<div class="ui-base-layout">
|
|
128
|
+
@if (showHeader) {
|
|
129
|
+
<ui-page-header
|
|
130
|
+
[title]="title"
|
|
131
|
+
[homeRoute]="homeRoute"
|
|
132
|
+
[showHome]="showHome"
|
|
133
|
+
[updateDocumentTitle]="updateDocumentTitle"
|
|
134
|
+
[titleSuffix]="titleSuffix"
|
|
135
|
+
/>
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
<main class="ui-base-layout__content">
|
|
139
|
+
<ng-content />
|
|
140
|
+
</main>
|
|
141
|
+
|
|
142
|
+
@if (hasVisibleActions) {
|
|
143
|
+
<aside
|
|
144
|
+
class="ui-base-layout__footer"
|
|
145
|
+
aria-label="Page actions"
|
|
146
|
+
>
|
|
147
|
+
<ui-button-area
|
|
148
|
+
[buttons]="mergedActions"
|
|
149
|
+
[align]="footerAlign"
|
|
150
|
+
[gap]="footerGap"
|
|
151
|
+
[stackOnMobile]="true"
|
|
152
|
+
ariaLabel="Page actions"
|
|
153
|
+
/>
|
|
154
|
+
</aside>
|
|
155
|
+
}
|
|
156
|
+
</div>
|
|
157
|
+
`, isInline: true, styles: [".ui-base-layout-host{display:block;min-height:100%}.ui-base-layout{display:flex;flex-direction:column;min-height:100%;font-family:var(--ui-font-family);padding:var(--ui-spacing-6)}.ui-base-layout__content{flex:1 1 auto}.ui-base-layout__footer{position:sticky;bottom:0;z-index:var(--ui-z-sticky, 100);padding:var(--ui-spacing-4) 0;margin-top:var(--ui-spacing-6);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border)}@media (min-width: 640px){.ui-base-layout__footer{padding:var(--ui-spacing-3) 0}}\n"], dependencies: [{ kind: "component", type: UiPageHeaderComponent, selector: "ui-page-header", inputs: ["title", "homeRoute", "showHome", "updateDocumentTitle", "titleSuffix"] }, { kind: "component", type: UiButtonAreaComponent, selector: "ui-button-area", inputs: ["buttons", "align", "ariaLabel", "gap", "stackOnMobile", "disableWhileLoading", "loadingIds"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
158
|
+
}
|
|
159
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiBaseLayoutComponent, decorators: [{
|
|
160
|
+
type: Component,
|
|
161
|
+
args: [{ selector: 'ui-base-layout', standalone: true, imports: [UiPageHeaderComponent, UiButtonAreaComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
|
|
162
|
+
class: 'ui-base-layout-host',
|
|
163
|
+
}, template: `
|
|
164
|
+
<div class="ui-base-layout">
|
|
165
|
+
@if (showHeader) {
|
|
166
|
+
<ui-page-header
|
|
167
|
+
[title]="title"
|
|
168
|
+
[homeRoute]="homeRoute"
|
|
169
|
+
[showHome]="showHome"
|
|
170
|
+
[updateDocumentTitle]="updateDocumentTitle"
|
|
171
|
+
[titleSuffix]="titleSuffix"
|
|
172
|
+
/>
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
<main class="ui-base-layout__content">
|
|
176
|
+
<ng-content />
|
|
177
|
+
</main>
|
|
178
|
+
|
|
179
|
+
@if (hasVisibleActions) {
|
|
180
|
+
<aside
|
|
181
|
+
class="ui-base-layout__footer"
|
|
182
|
+
aria-label="Page actions"
|
|
183
|
+
>
|
|
184
|
+
<ui-button-area
|
|
185
|
+
[buttons]="mergedActions"
|
|
186
|
+
[align]="footerAlign"
|
|
187
|
+
[gap]="footerGap"
|
|
188
|
+
[stackOnMobile]="true"
|
|
189
|
+
ariaLabel="Page actions"
|
|
190
|
+
/>
|
|
191
|
+
</aside>
|
|
192
|
+
}
|
|
193
|
+
</div>
|
|
194
|
+
`, styles: [".ui-base-layout-host{display:block;min-height:100%}.ui-base-layout{display:flex;flex-direction:column;min-height:100%;font-family:var(--ui-font-family);padding:var(--ui-spacing-6)}.ui-base-layout__content{flex:1 1 auto}.ui-base-layout__footer{position:sticky;bottom:0;z-index:var(--ui-z-sticky, 100);padding:var(--ui-spacing-4) 0;margin-top:var(--ui-spacing-6);background:var(--ui-color-surface);border-top:1px solid var(--ui-color-border)}@media (min-width: 640px){.ui-base-layout__footer{padding:var(--ui-spacing-3) 0}}\n"] }]
|
|
195
|
+
}], propDecorators: { showHeader: [{
|
|
196
|
+
type: Input
|
|
197
|
+
}], title: [{
|
|
198
|
+
type: Input
|
|
199
|
+
}], homeRoute: [{
|
|
200
|
+
type: Input
|
|
201
|
+
}], showHome: [{
|
|
202
|
+
type: Input
|
|
203
|
+
}], updateDocumentTitle: [{
|
|
204
|
+
type: Input
|
|
205
|
+
}], titleSuffix: [{
|
|
206
|
+
type: Input
|
|
207
|
+
}], showBackButton: [{
|
|
208
|
+
type: Input
|
|
209
|
+
}], backButtonConfig: [{
|
|
210
|
+
type: Input
|
|
211
|
+
}], actions: [{
|
|
212
|
+
type: Input
|
|
213
|
+
}], footerAlign: [{
|
|
214
|
+
type: Input
|
|
215
|
+
}], footerGap: [{
|
|
216
|
+
type: Input
|
|
217
|
+
}] } });
|
|
218
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-layout.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/base-layout/base-layout.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  DestroyRef,\r\n  inject,\r\n  Input,\r\n  OnInit,\r\n  ChangeDetectionStrategy,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { Location } from '@angular/common';\r\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\r\nimport { filter, debounceTime } from 'rxjs/operators';\r\nimport { UiPageHeaderComponent } from '../page-header/page-header.component';\r\nimport { UiBreadcrumbService } from '../page-header/breadcrumb.service';\r\nimport { UiBreadcrumbItem } from '../page-header/page-header.types';\r\nimport { UiButtonAreaComponent } from '../button/button-area.component';\r\nimport { UiButtonDescriptor, UiButtonAreaAlign } from '../button/button.types';\r\nimport { UiBackButtonConfig } from './base-layout.types';\r\n\r\n/**\r\n * Full-page layout component that combines a page header (breadcrumbs + title),\r\n * a main content area via `<ng-content>`, and a sticky footer action bar\r\n * powered by `UiButtonAreaComponent`.\r\n *\r\n * Optionally renders an automatic \"back\" button in the footer, derived from\r\n * the breadcrumb trail (navigates to the parent breadcrumb).\r\n *\r\n * @selector ui-base-layout\r\n *\r\n * @example\r\n * ```html\r\n * <ui-base-layout [actions]=\"pageActions\" footerAlign=\"end\">\r\n *   <p>Your page content here</p>\r\n * </ui-base-layout>\r\n * ```\r\n *\r\n * @example\r\n * ```html\r\n * <!-- With back button and custom title -->\r\n * <ui-base-layout\r\n *   title=\"Dettaglio utente\"\r\n *   [showBackButton]=\"true\"\r\n *   [actions]=\"[\r\n *     { id: 'save', label: 'Salva', variant: 'primary', icon: 'save', action: save },\r\n *   ]\"\r\n * >\r\n *   <app-user-detail />\r\n * </ui-base-layout>\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-base-layout',\r\n  standalone: true,\r\n  imports: [UiPageHeaderComponent, UiButtonAreaComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-base-layout-host',\r\n  },\r\n  template: `\r\n    <div class=\"ui-base-layout\">\r\n      @if (showHeader) {\r\n        <ui-page-header\r\n          [title]=\"title\"\r\n          [homeRoute]=\"homeRoute\"\r\n          [showHome]=\"showHome\"\r\n          [updateDocumentTitle]=\"updateDocumentTitle\"\r\n          [titleSuffix]=\"titleSuffix\"\r\n        />\r\n      }\r\n\r\n      <main class=\"ui-base-layout__content\">\r\n        <ng-content />\r\n      </main>\r\n\r\n      @if (hasVisibleActions) {\r\n        <aside\r\n          class=\"ui-base-layout__footer\"\r\n          aria-label=\"Page actions\"\r\n        >\r\n          <ui-button-area\r\n            [buttons]=\"mergedActions\"\r\n            [align]=\"footerAlign\"\r\n            [gap]=\"footerGap\"\r\n            [stackOnMobile]=\"true\"\r\n            ariaLabel=\"Page actions\"\r\n          />\r\n        </aside>\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './base-layout.component.scss',\r\n})\r\nexport class UiBaseLayoutComponent implements OnInit {\r\n  /** Whether to display the page header (breadcrumbs + title). */\r\n  @Input() showHeader = true;\r\n\r\n  /** Override the auto-detected page title. Passed through to `ui-page-header`. */\r\n  @Input() title = '';\r\n\r\n  /** Route path for the Home breadcrumb link. */\r\n  @Input() homeRoute = '/';\r\n\r\n  /** Whether to display the Home link in the breadcrumb trail. */\r\n  @Input() showHome = true;\r\n\r\n  /** When `true`, updates `document.title` on navigation. */\r\n  @Input() updateDocumentTitle = false;\r\n\r\n  /** Suffix appended to `document.title`. */\r\n  @Input() titleSuffix = '';\r\n\r\n  /** Whether to show an automatic \"back\" button based on breadcrumbs. */\r\n  @Input() showBackButton = true;\r\n\r\n  /** Configuration overrides for the automatic back button. */\r\n  @Input() backButtonConfig: UiBackButtonConfig = {};\r\n\r\n  /** Array of button descriptors for the footer action area. */\r\n  @Input() actions: UiButtonDescriptor[] = [];\r\n\r\n  /** Horizontal alignment of the footer button area. */\r\n  @Input() footerAlign: UiButtonAreaAlign = 'start';\r\n\r\n  /** Gap size between footer buttons. */\r\n  @Input() footerGap: 'xs' | 'sm' | 'md' | 'lg' | 'xl' = 'sm';\r\n\r\n  private readonly router = inject(Router);\r\n  private readonly location = inject(Location);\r\n  private readonly activatedRoute = inject(ActivatedRoute);\r\n  private readonly breadcrumbService = inject(UiBreadcrumbService);\r\n  private readonly destroyRef = inject(DestroyRef);\r\n\r\n  /** @internal Current breadcrumbs, updated on each navigation. */\r\n  private breadcrumbs: UiBreadcrumbItem[] = [];\r\n\r\n  /**\r\n   * Merged action list (back button + consumer actions).\r\n   * Computed on every change-detection check so it stays in sync\r\n   * with both navigation state (breadcrumbs) and input changes.\r\n   */\r\n  get mergedActions(): UiButtonDescriptor[] {\r\n    const result: UiButtonDescriptor[] = [];\r\n\r\n    if (this.showBackButton) {\r\n      const backTarget = this.getBackTarget();\r\n      result.push({\r\n        id: '__ui-back',\r\n        label: this.backButtonConfig.label ?? 'Indietro',\r\n        icon: (this.backButtonConfig.icon as any) ?? 'arrow-left',\r\n        iconPosition: 'leading',\r\n        variant: (this.backButtonConfig.variant as any) ?? 'outline',\r\n        hidden: !backTarget,\r\n        action: () => this.navigateBack(),\r\n      });\r\n    }\r\n\r\n    result.push(...this.actions);\r\n    return result;\r\n  }\r\n\r\n  /** Whether the footer should render (at least one non-hidden action exists). */\r\n  get hasVisibleActions(): boolean {\r\n    return this.mergedActions.some((a) => !a.hidden);\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.refreshBreadcrumbs();\r\n\r\n    this.router.events\r\n      .pipe(\r\n        filter((event) => event instanceof NavigationEnd),\r\n        debounceTime(50),\r\n        takeUntilDestroyed(this.destroyRef),\r\n      )\r\n      .subscribe(() => this.refreshBreadcrumbs());\r\n  }\r\n\r\n  /** @internal Refreshes breadcrumb state from the current route. */\r\n  private refreshBreadcrumbs(): void {\r\n    this.breadcrumbs = this.breadcrumbService.getBreadcrumbsForRoute(this.activatedRoute);\r\n  }\r\n\r\n  /** @internal Determines the back navigation target from breadcrumbs. */\r\n  private getBackTarget(): UiBreadcrumbItem | null {\r\n    if (this.breadcrumbs.length < 2) return null;\r\n    return this.breadcrumbs[this.breadcrumbs.length - 2];\r\n  }\r\n\r\n  /** @internal Navigates to the parent breadcrumb or falls back to browser history. */\r\n  private navigateBack(): void {\r\n    const target = this.getBackTarget();\r\n    if (target?.url) {\r\n      this.router.navigateByUrl(target.url);\r\n    } else {\r\n      this.location.back();\r\n    }\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/base-layout
|
|
3
|
+
* Types and interfaces for UiBaseLayout component.
|
|
4
|
+
*/
|
|
5
|
+
export {};
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1sYXlvdXQudHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2Jhc2UtbGF5b3V0LnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBAbW9kdWxlIG5nLXVpLXN5c3RlbS9iYXNlLWxheW91dFxyXG4gKiBUeXBlcyBhbmQgaW50ZXJmYWNlcyBmb3IgVWlCYXNlTGF5b3V0IGNvbXBvbmVudC5cclxuICovXHJcblxyXG5pbXBvcnQgeyBVaUJ1dHRvbkRlc2NyaXB0b3IsIFVpQnV0dG9uQXJlYUFsaWduIH0gZnJvbSAnLi4vYnV0dG9uL2J1dHRvbi50eXBlcyc7XHJcblxyXG4vLyBSZS1leHBvcnQgZm9yIGNvbnN1bWVyIGNvbnZlbmllbmNlXHJcbmV4cG9ydCB7IFVpQnV0dG9uRGVzY3JpcHRvciwgVWlCdXR0b25BcmVhQWxpZ24gfTtcclxuXHJcbi8qKlxyXG4gKiBDb25maWd1cmF0aW9uIGZvciB0aGUgYXV0b21hdGljIFwiYmFja1wiIGJ1dHRvbiByZW5kZXJlZCBpbiB0aGUgZm9vdGVyIGFjdGlvbiBhcmVhLlxyXG4gKlxyXG4gKiBAdXNhZ2VOb3Rlc1xyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIGNvbnN0IGJhY2tDb25maWc6IFVpQmFja0J1dHRvbkNvbmZpZyA9IHtcclxuICogICBsYWJlbDogJ0luZGlldHJvJyxcclxuICogICBpY29uOiAnYXJyb3ctbGVmdCcsXHJcbiAqICAgdmFyaWFudDogJ291dGxpbmUnLFxyXG4gKiB9O1xyXG4gKiBgYGBcclxuICovXHJcbmV4cG9ydCBpbnRlcmZhY2UgVWlCYWNrQnV0dG9uQ29uZmlnIHtcclxuICAvKiogTGFiZWwgdGV4dCBmb3IgdGhlIGJhY2sgYnV0dG9uLiBAZGVmYXVsdCAnSW5kaWV0cm8nICovXHJcbiAgbGFiZWw/OiBzdHJpbmc7XHJcbiAgLyoqIEx1Y2lkZSBpY29uIG5hbWUuIEBkZWZhdWx0ICdhcnJvdy1sZWZ0JyAqL1xyXG4gIGljb24/OiBzdHJpbmc7XHJcbiAgLyoqIFZpc3VhbCB2YXJpYW50LiBAZGVmYXVsdCAnb3V0bGluZScgKi9cclxuICB2YXJpYW50Pzogc3RyaW5nO1xyXG59XHJcbiJdfQ==
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ng-ui-system — Base Layout entry point.
|
|
3
|
+
*
|
|
4
|
+
* @example
|
|
5
|
+
* ```typescript
|
|
6
|
+
* import {
|
|
7
|
+
* UiBaseLayoutComponent,
|
|
8
|
+
* UiBackButtonConfig,
|
|
9
|
+
* } from 'ng-ui-system';
|
|
10
|
+
* ```
|
|
11
|
+
*/
|
|
12
|
+
// Components
|
|
13
|
+
export { UiBaseLayoutComponent } from './base-layout.component';
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtbGF5b3V0L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7O0dBVUc7QUFFSCxhQUFhO0FBQ2IsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0seUJBQXlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogbmctdWktc3lzdGVtIOKAlCBCYXNlIExheW91dCBlbnRyeSBwb2ludC5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogYGBgdHlwZXNjcmlwdFxyXG4gKiBpbXBvcnQge1xyXG4gKiAgIFVpQmFzZUxheW91dENvbXBvbmVudCxcclxuICogICBVaUJhY2tCdXR0b25Db25maWcsXHJcbiAqIH0gZnJvbSAnbmctdWktc3lzdGVtJztcclxuICogYGBgXHJcbiAqL1xyXG5cclxuLy8gQ29tcG9uZW50c1xyXG5leHBvcnQgeyBVaUJhc2VMYXlvdXRDb21wb25lbnQgfSBmcm9tICcuL2Jhc2UtbGF5b3V0LmNvbXBvbmVudCc7XHJcblxyXG4vLyBUeXBlc1xyXG5leHBvcnQgeyBVaUJhY2tCdXR0b25Db25maWcgfSBmcm9tICcuL2Jhc2UtbGF5b3V0LnR5cGVzJztcclxuIl19
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
import { Component, Input, ChangeDetectionStrategy, ViewEncapsulation, inject } from '@angular/core';
|
|
2
|
+
import { Router } from '@angular/router';
|
|
3
|
+
import { UiButtonComponent } from './button.component';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
/**
|
|
6
|
+
* Standalone button-group component driven by a `UiButtonDescriptor[]` configuration.
|
|
7
|
+
*
|
|
8
|
+
* Renders a group of `UiButton` instances from declarative descriptors.
|
|
9
|
+
* Supports per-button loading/disabled states, dynamic visibility,
|
|
10
|
+
* alignment control, and optional navigation via Angular Router.
|
|
11
|
+
*
|
|
12
|
+
* @selector ui-button-area
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* ```html
|
|
16
|
+
* <ui-button-area
|
|
17
|
+
* [buttons]="actions"
|
|
18
|
+
* align="end"
|
|
19
|
+
* gap="sm"
|
|
20
|
+
* [loadingIds]="currentlyLoading"
|
|
21
|
+
* />
|
|
22
|
+
* ```
|
|
23
|
+
*
|
|
24
|
+
* @example
|
|
25
|
+
* ```typescript
|
|
26
|
+
* // Dynamic configuration
|
|
27
|
+
* actions: UiButtonDescriptor[] = [
|
|
28
|
+
* { id: 'export', label: 'Export Excel', icon: 'file-spreadsheet', variant: 'primary',
|
|
29
|
+
* action: () => this.exportExcel() },
|
|
30
|
+
* { id: 'delete', label: 'Delete', icon: 'trash-2', variant: 'warn',
|
|
31
|
+
* action: () => this.deleteSelected(), hidden: !this.hasSelection },
|
|
32
|
+
* ];
|
|
33
|
+
*
|
|
34
|
+
* // Set loading state from outside:
|
|
35
|
+
* currentlyLoading: string | string[] | null = 'export';
|
|
36
|
+
* ```
|
|
37
|
+
*/
|
|
38
|
+
export class UiButtonAreaComponent {
|
|
39
|
+
constructor() {
|
|
40
|
+
/**
|
|
41
|
+
* Array of button descriptors to render.
|
|
42
|
+
* Buttons with `hidden: true` are filtered out.
|
|
43
|
+
*/
|
|
44
|
+
this.buttons = [];
|
|
45
|
+
/** Horizontal alignment of the button group. */
|
|
46
|
+
this.align = 'end';
|
|
47
|
+
/** Accessible group label for screen readers. */
|
|
48
|
+
this.ariaLabel = 'Actions';
|
|
49
|
+
/** Gap between buttons (maps to design token spacing). */
|
|
50
|
+
this.gap = 'sm';
|
|
51
|
+
/** Stack buttons vertically on mobile viewports (<600px). */
|
|
52
|
+
this.stackOnMobile = true;
|
|
53
|
+
/**
|
|
54
|
+
* When `true`, all buttons are disabled while any button is in a loading state.
|
|
55
|
+
* Useful to prevent double-actions during async operations.
|
|
56
|
+
*/
|
|
57
|
+
this.disableWhileLoading = false;
|
|
58
|
+
/** @internal */
|
|
59
|
+
this._loadingIds = new Set();
|
|
60
|
+
/** @internal Optional router for href navigation. */
|
|
61
|
+
this.router = inject(Router, { optional: true });
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Set one or more buttons to a loading state by their `id`.
|
|
65
|
+
* Accepts a single string, an array of strings, or `null` to clear.
|
|
66
|
+
*
|
|
67
|
+
* @example
|
|
68
|
+
* ```html
|
|
69
|
+
* <!-- Single -->
|
|
70
|
+
* <ui-button-area [loadingIds]="'save'" />
|
|
71
|
+
*
|
|
72
|
+
* <!-- Multiple -->
|
|
73
|
+
* <ui-button-area [loadingIds]="['save', 'export']" />
|
|
74
|
+
* ```
|
|
75
|
+
*/
|
|
76
|
+
set loadingIds(value) {
|
|
77
|
+
if (value === null || value === undefined) {
|
|
78
|
+
this._loadingIds = new Set();
|
|
79
|
+
}
|
|
80
|
+
else if (Array.isArray(value)) {
|
|
81
|
+
this._loadingIds = new Set(value);
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
this._loadingIds = new Set([value]);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
/** Buttons filtered to visible (non-hidden) entries. */
|
|
88
|
+
get visibleButtons() {
|
|
89
|
+
return this.buttons.filter((b) => !b.hidden);
|
|
90
|
+
}
|
|
91
|
+
/** Assembled CSS class string for the button area container. */
|
|
92
|
+
get areaClasses() {
|
|
93
|
+
return [
|
|
94
|
+
'ui-button-area',
|
|
95
|
+
`ui-button-area--align-${this.align}`,
|
|
96
|
+
`ui-button-area--gap-${this.gap}`,
|
|
97
|
+
this.stackOnMobile ? 'ui-button-area--stack-mobile' : '',
|
|
98
|
+
]
|
|
99
|
+
.filter(Boolean)
|
|
100
|
+
.join(' ');
|
|
101
|
+
}
|
|
102
|
+
/** @internal Track function for @for loop. */
|
|
103
|
+
trackButton(index, button) {
|
|
104
|
+
return button.id ?? `idx-${index}`;
|
|
105
|
+
}
|
|
106
|
+
/** Whether a specific button is in a loading state. */
|
|
107
|
+
isButtonLoading(button) {
|
|
108
|
+
return !!button.loading || (!!button.id && this._loadingIds.has(button.id));
|
|
109
|
+
}
|
|
110
|
+
/** Whether a specific button is disabled (includes loading logic). */
|
|
111
|
+
isButtonDisabled(button) {
|
|
112
|
+
if (button.disabled)
|
|
113
|
+
return true;
|
|
114
|
+
if (this.isButtonLoading(button))
|
|
115
|
+
return true;
|
|
116
|
+
if (this.disableWhileLoading && this._loadingIds.size > 0)
|
|
117
|
+
return true;
|
|
118
|
+
return false;
|
|
119
|
+
}
|
|
120
|
+
/** @internal Handles button click: calls action callback and/or navigates. */
|
|
121
|
+
handleButtonClick(button, event) {
|
|
122
|
+
if (button.action) {
|
|
123
|
+
button.action(event);
|
|
124
|
+
}
|
|
125
|
+
if (button.href && this.router) {
|
|
126
|
+
this.router.navigateByUrl(button.href);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiButtonAreaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
130
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiButtonAreaComponent, isStandalone: true, selector: "ui-button-area", inputs: { buttons: "buttons", align: "align", ariaLabel: "ariaLabel", gap: "gap", stackOnMobile: "stackOnMobile", disableWhileLoading: "disableWhileLoading", loadingIds: "loadingIds" }, host: { classAttribute: "ui-button-area-host" }, ngImport: i0, template: `
|
|
131
|
+
<div
|
|
132
|
+
[class]="areaClasses"
|
|
133
|
+
role="group"
|
|
134
|
+
[attr.aria-label]="ariaLabel"
|
|
135
|
+
>
|
|
136
|
+
@for (button of visibleButtons; track trackButton($index, button)) {
|
|
137
|
+
<ui-button
|
|
138
|
+
[attr.data-button-id]="button.id || null"
|
|
139
|
+
[label]="button.label"
|
|
140
|
+
[tooltip]="button.tooltip ?? ''"
|
|
141
|
+
[icon]="button.icon"
|
|
142
|
+
[iconPosition]="button.iconPosition ?? 'trailing'"
|
|
143
|
+
[variant]="button.variant ?? 'primary'"
|
|
144
|
+
[size]="button.size ?? 'md'"
|
|
145
|
+
[loading]="isButtonLoading(button)"
|
|
146
|
+
[disabled]="isButtonDisabled(button)"
|
|
147
|
+
[customClass]="button.customClass"
|
|
148
|
+
(click)="handleButtonClick(button, $event)"
|
|
149
|
+
/>
|
|
150
|
+
}
|
|
151
|
+
</div>
|
|
152
|
+
`, isInline: true, styles: [".ui-button-area-host{display:block}@media (max-width: 767.98px){.ui-button-area-host{width:100%}}.ui-button-area{display:flex;flex-wrap:wrap;align-items:center}.ui-button-area--gap-xs{gap:var(--ui-spacing-1)}.ui-button-area--gap-sm{gap:var(--ui-spacing-2)}.ui-button-area--gap-md{gap:var(--ui-spacing-3)}.ui-button-area--gap-lg{gap:var(--ui-spacing-4)}.ui-button-area--gap-xl{gap:var(--ui-spacing-5)}.ui-button-area--align-start{justify-content:flex-start}.ui-button-area--align-center{justify-content:center}.ui-button-area--align-end{justify-content:flex-end}.ui-button-area--align-between{justify-content:space-between}@media (max-width: 767.98px){.ui-button-area--stack-mobile{flex-direction:column;width:100%}.ui-button-area--stack-mobile .ui-button-host{display:flex;width:100%}.ui-button-area--stack-mobile .ui-button{width:100%}}\n"], dependencies: [{ kind: "component", type: UiButtonComponent, selector: "ui-button", inputs: ["label", "tooltip", "variant", "size", "icon", "iconPosition", "loading", "disabled", "fullWidth", "type", "ariaLabel", "customClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
|
|
153
|
+
}
|
|
154
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiButtonAreaComponent, decorators: [{
|
|
155
|
+
type: Component,
|
|
156
|
+
args: [{ selector: 'ui-button-area', standalone: true, imports: [UiButtonComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
|
|
157
|
+
class: 'ui-button-area-host',
|
|
158
|
+
}, template: `
|
|
159
|
+
<div
|
|
160
|
+
[class]="areaClasses"
|
|
161
|
+
role="group"
|
|
162
|
+
[attr.aria-label]="ariaLabel"
|
|
163
|
+
>
|
|
164
|
+
@for (button of visibleButtons; track trackButton($index, button)) {
|
|
165
|
+
<ui-button
|
|
166
|
+
[attr.data-button-id]="button.id || null"
|
|
167
|
+
[label]="button.label"
|
|
168
|
+
[tooltip]="button.tooltip ?? ''"
|
|
169
|
+
[icon]="button.icon"
|
|
170
|
+
[iconPosition]="button.iconPosition ?? 'trailing'"
|
|
171
|
+
[variant]="button.variant ?? 'primary'"
|
|
172
|
+
[size]="button.size ?? 'md'"
|
|
173
|
+
[loading]="isButtonLoading(button)"
|
|
174
|
+
[disabled]="isButtonDisabled(button)"
|
|
175
|
+
[customClass]="button.customClass"
|
|
176
|
+
(click)="handleButtonClick(button, $event)"
|
|
177
|
+
/>
|
|
178
|
+
}
|
|
179
|
+
</div>
|
|
180
|
+
`, styles: [".ui-button-area-host{display:block}@media (max-width: 767.98px){.ui-button-area-host{width:100%}}.ui-button-area{display:flex;flex-wrap:wrap;align-items:center}.ui-button-area--gap-xs{gap:var(--ui-spacing-1)}.ui-button-area--gap-sm{gap:var(--ui-spacing-2)}.ui-button-area--gap-md{gap:var(--ui-spacing-3)}.ui-button-area--gap-lg{gap:var(--ui-spacing-4)}.ui-button-area--gap-xl{gap:var(--ui-spacing-5)}.ui-button-area--align-start{justify-content:flex-start}.ui-button-area--align-center{justify-content:center}.ui-button-area--align-end{justify-content:flex-end}.ui-button-area--align-between{justify-content:space-between}@media (max-width: 767.98px){.ui-button-area--stack-mobile{flex-direction:column;width:100%}.ui-button-area--stack-mobile .ui-button-host{display:flex;width:100%}.ui-button-area--stack-mobile .ui-button{width:100%}}\n"] }]
|
|
181
|
+
}], propDecorators: { buttons: [{
|
|
182
|
+
type: Input
|
|
183
|
+
}], align: [{
|
|
184
|
+
type: Input
|
|
185
|
+
}], ariaLabel: [{
|
|
186
|
+
type: Input
|
|
187
|
+
}], gap: [{
|
|
188
|
+
type: Input
|
|
189
|
+
}], stackOnMobile: [{
|
|
190
|
+
type: Input
|
|
191
|
+
}], disableWhileLoading: [{
|
|
192
|
+
type: Input
|
|
193
|
+
}], loadingIds: [{
|
|
194
|
+
type: Input
|
|
195
|
+
}] } });
|
|
196
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button-area.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/button/button-area.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ChangeDetectionStrategy, ViewEncapsulation, inject } from '@angular/core';\r\nimport { Router } from '@angular/router';\r\nimport { UiSize } from '../../core/types';\r\nimport { UiButtonComponent } from './button.component';\r\nimport { UiButtonDescriptor, UiButtonAreaAlign } from './button.types';\r\n\r\n/**\r\n * Standalone button-group component driven by a `UiButtonDescriptor[]` configuration.\r\n *\r\n * Renders a group of `UiButton` instances from declarative descriptors.\r\n * Supports per-button loading/disabled states, dynamic visibility,\r\n * alignment control, and optional navigation via Angular Router.\r\n *\r\n * @selector ui-button-area\r\n *\r\n * @example\r\n * ```html\r\n * <ui-button-area\r\n *   [buttons]=\"actions\"\r\n *   align=\"end\"\r\n *   gap=\"sm\"\r\n *   [loadingIds]=\"currentlyLoading\"\r\n * />\r\n * ```\r\n *\r\n * @example\r\n * ```typescript\r\n * // Dynamic configuration\r\n * actions: UiButtonDescriptor[] = [\r\n *   { id: 'export', label: 'Export Excel', icon: 'file-spreadsheet', variant: 'primary',\r\n *     action: () => this.exportExcel() },\r\n *   { id: 'delete', label: 'Delete', icon: 'trash-2', variant: 'warn',\r\n *     action: () => this.deleteSelected(), hidden: !this.hasSelection },\r\n * ];\r\n *\r\n * // Set loading state from outside:\r\n * currentlyLoading: string | string[] | null = 'export';\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-button-area',\r\n  standalone: true,\r\n  imports: [UiButtonComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-button-area-host',\r\n  },\r\n  template: `\r\n    <div\r\n      [class]=\"areaClasses\"\r\n      role=\"group\"\r\n      [attr.aria-label]=\"ariaLabel\"\r\n    >\r\n      @for (button of visibleButtons; track trackButton($index, button)) {\r\n        <ui-button\r\n          [attr.data-button-id]=\"button.id || null\"\r\n          [label]=\"button.label\"\r\n          [tooltip]=\"button.tooltip ?? ''\"\r\n          [icon]=\"button.icon\"\r\n          [iconPosition]=\"button.iconPosition ?? 'trailing'\"\r\n          [variant]=\"button.variant ?? 'primary'\"\r\n          [size]=\"button.size ?? 'md'\"\r\n          [loading]=\"isButtonLoading(button)\"\r\n          [disabled]=\"isButtonDisabled(button)\"\r\n          [customClass]=\"button.customClass\"\r\n          (click)=\"handleButtonClick(button, $event)\"\r\n        />\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './button-area.component.scss',\r\n})\r\nexport class UiButtonAreaComponent {\r\n  /**\r\n   * Array of button descriptors to render.\r\n   * Buttons with `hidden: true` are filtered out.\r\n   */\r\n  @Input() buttons: UiButtonDescriptor[] = [];\r\n\r\n  /** Horizontal alignment of the button group. */\r\n  @Input() align: UiButtonAreaAlign = 'end';\r\n\r\n  /** Accessible group label for screen readers. */\r\n  @Input() ariaLabel = 'Actions';\r\n\r\n  /** Gap between buttons (maps to design token spacing). */\r\n  @Input() gap: UiSize = 'sm';\r\n\r\n  /** Stack buttons vertically on mobile viewports (<600px). */\r\n  @Input() stackOnMobile = true;\r\n\r\n  /**\r\n   * When `true`, all buttons are disabled while any button is in a loading state.\r\n   * Useful to prevent double-actions during async operations.\r\n   */\r\n  @Input() disableWhileLoading = false;\r\n\r\n  /**\r\n   * Set one or more buttons to a loading state by their `id`.\r\n   * Accepts a single string, an array of strings, or `null` to clear.\r\n   *\r\n   * @example\r\n   * ```html\r\n   * <!-- Single -->\r\n   * <ui-button-area [loadingIds]=\"'save'\" />\r\n   *\r\n   * <!-- Multiple -->\r\n   * <ui-button-area [loadingIds]=\"['save', 'export']\" />\r\n   * ```\r\n   */\r\n  @Input()\r\n  set loadingIds(value: string | string[] | null | undefined) {\r\n    if (value === null || value === undefined) {\r\n      this._loadingIds = new Set();\r\n    } else if (Array.isArray(value)) {\r\n      this._loadingIds = new Set(value);\r\n    } else {\r\n      this._loadingIds = new Set([value]);\r\n    }\r\n  }\r\n\r\n  /** @internal */\r\n  private _loadingIds = new Set<string>();\r\n\r\n  /** @internal Optional router for href navigation. */\r\n  private router = inject(Router, { optional: true });\r\n\r\n  /** Buttons filtered to visible (non-hidden) entries. */\r\n  get visibleButtons(): UiButtonDescriptor[] {\r\n    return this.buttons.filter((b) => !b.hidden);\r\n  }\r\n\r\n  /** Assembled CSS class string for the button area container. */\r\n  get areaClasses(): string {\r\n    return [\r\n      'ui-button-area',\r\n      `ui-button-area--align-${this.align}`,\r\n      `ui-button-area--gap-${this.gap}`,\r\n      this.stackOnMobile ? 'ui-button-area--stack-mobile' : '',\r\n    ]\r\n      .filter(Boolean)\r\n      .join(' ');\r\n  }\r\n\r\n  /** @internal Track function for @for loop. */\r\n  trackButton(index: number, button: UiButtonDescriptor): string {\r\n    return button.id ?? `idx-${index}`;\r\n  }\r\n\r\n  /** Whether a specific button is in a loading state. */\r\n  isButtonLoading(button: UiButtonDescriptor): boolean {\r\n    return !!button.loading || (!!button.id && this._loadingIds.has(button.id));\r\n  }\r\n\r\n  /** Whether a specific button is disabled (includes loading logic). */\r\n  isButtonDisabled(button: UiButtonDescriptor): boolean {\r\n    if (button.disabled) return true;\r\n    if (this.isButtonLoading(button)) return true;\r\n    if (this.disableWhileLoading && this._loadingIds.size > 0) return true;\r\n    return false;\r\n  }\r\n\r\n  /** @internal Handles button click: calls action callback and/or navigates. */\r\n  handleButtonClick(button: UiButtonDescriptor, event: MouseEvent): void {\r\n    if (button.action) {\r\n      button.action(event);\r\n    }\r\n    if (button.href && this.router) {\r\n      this.router.navigateByUrl(button.href);\r\n    }\r\n  }\r\n}\r\n"]}
|