@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,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module ng-ui-system/layout-builder
|
|
3
|
+
* Types and interfaces for the UiLayoutBuilder shell component.
|
|
4
|
+
*
|
|
5
|
+
* The layout builder uses a schema-driven approach (similar to UiFormBuilder)
|
|
6
|
+
* to materialize a full responsive application shell from a single descriptor.
|
|
7
|
+
*/
|
|
8
|
+
export {};
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"layout-builder.types.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/layout-builder/layout-builder.types.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG","sourcesContent":["/**\r\n * @module ng-ui-system/layout-builder\r\n * Types and interfaces for the UiLayoutBuilder shell component.\r\n *\r\n * The layout builder uses a schema-driven approach (similar to UiFormBuilder)\r\n * to materialize a full responsive application shell from a single descriptor.\r\n */\r\n\r\nimport { UiIconName, UiVariant } from '../../core/types/index';\r\n\r\n// ─── Layout Mode & Content Type ──────────────────────────────────────\r\n\r\n/** Layout mode controlling the overall navigation strategy. */\r\nexport type UiLayoutMode = 'sidebar' | 'topbar';\r\n\r\n/** Content area sizing strategy. */\r\nexport type UiContentType = 'fluid' | 'boxed' | 'fullscreen';\r\n\r\n// ─── Breadcrumb ──────────────────────────────────────────────────────\r\n\r\n/** Single breadcrumb entry derived from the navigation schema. */\r\nexport interface UiLayoutBreadcrumb {\r\n  /** Display label. */\r\n  label: string;\r\n  /** Route URL (omitted for the current/last crumb). */\r\n  url?: string;\r\n  /** Whether this is the last (current) breadcrumb in the trail. */\r\n  isLast: boolean;\r\n}\r\n\r\n// ─── Navigation ──────────────────────────────────────────────────────\r\n\r\n/** Discriminator for navigation item behaviour. */\r\nexport type UiNavItemType = 'route' | 'external' | 'divider';\r\n\r\n/**\r\n * Single navigation item descriptor.\r\n *\r\n * @usageNotes\r\n * ### Internal route\r\n * ```typescript\r\n * { id: 'users', label: 'Utenti', icon: 'users', route: '/users' }\r\n * ```\r\n *\r\n * ### External link\r\n * ```typescript\r\n * { id: 'docs', label: 'Docs', icon: 'external-link', type: 'external',\r\n *   href: 'https://docs.example.com', target: '_blank' }\r\n * ```\r\n *\r\n * ### Expandable with children\r\n * ```typescript\r\n * {\r\n *   id: 'form-builder', label: 'Form Builder', icon: 'file-text',\r\n *   route: '/form-builder',\r\n *   children: [\r\n *     { label: 'Panoramica', items: [\r\n *       { id: 'fb-all', label: 'Tutti i campi', route: '/form-builder/all-fields' }\r\n *     ]}\r\n *   ]\r\n * }\r\n * ```\r\n */\r\nexport interface UiNavItem {\r\n  /** Unique item identifier. */\r\n  id: string;\r\n  /** Display label. */\r\n  label: string;\r\n  /** Lucide icon name. */\r\n  icon?: UiIconName;\r\n  /** Item behaviour type. @default 'route' */\r\n  type?: UiNavItemType;\r\n  /** Angular router path (for `type: 'route'`). */\r\n  route?: string;\r\n  /** Whether to use exact matching for active state. @default false */\r\n  routeActiveExact?: boolean;\r\n  /** External URL (for `type: 'external'`). */\r\n  href?: string;\r\n  /** Link target for external links. @default '_blank' */\r\n  target?: '_blank' | '_self';\r\n  /** Nested child groups for expandable navigation. */\r\n  children?: UiNavGroup[];\r\n  /** Whether this item is hidden. @default false */\r\n  hidden?: boolean;\r\n  /** Badge text or count displayed next to the label. */\r\n  badge?: string | number;\r\n  /** Whether to include this item in the mobile bottom navigation bar. @default false */\r\n  bottomNav?: boolean;\r\n  /** Sort order in the bottom navigation bar (lower = first). */\r\n  bottomNavOrder?: number;\r\n}\r\n\r\n/**\r\n * Labelled group of navigation items (nested under a parent `UiNavItem`).\r\n *\r\n * @usageNotes\r\n * ```typescript\r\n * const group: UiNavGroup = {\r\n *   label: 'Validazioni',\r\n *   items: [\r\n *     { id: 'v-std', label: 'Standard', route: '/form-builder/validations-standard' },\r\n *     { id: 'v-cross', label: 'Cross-field', route: '/form-builder/validations-crossfield' },\r\n *   ]\r\n * };\r\n * ```\r\n */\r\nexport interface UiNavGroup {\r\n  /** Group heading displayed above the items. */\r\n  label: string;\r\n  /** Items belonging to this group. */\r\n  items: UiNavItem[];\r\n}\r\n\r\n/**\r\n * Top-level grouping used to partition `UiNavItem[]` into labelled sections\r\n * in the sidebar (e.g. \"Overview\", \"Components\").\r\n */\r\nexport interface UiNavSection {\r\n  /** Unique section identifier. */\r\n  id: string;\r\n  /** Section heading displayed in the sidebar. */\r\n  label: string;\r\n  /** IDs of `UiNavItem`s belonging to this section (order preserved). */\r\n  itemIds: string[];\r\n}\r\n\r\n/**\r\n * Auto-discovery configuration for generating navigation items from Angular routes.\r\n *\r\n * When enabled, the layout builder scans the route tree and creates\r\n * `UiNavItem` entries from routes that declare `data.id` and `data.title`.\r\n */\r\nexport interface UiNavAutoDiscoveryConfig {\r\n  /** Whether auto-discovery is enabled. */\r\n  enabled: boolean;\r\n  /** Route paths to exclude from discovery (exact match). */\r\n  excludePaths?: string[];\r\n  /** How to merge discovered items with manually defined items. @default 'append' */\r\n  mergeStrategy?: 'prepend' | 'append' | 'replace';\r\n}\r\n\r\n/** Navigation configuration block within `UiLayoutSchema`. */\r\nexport interface UiLayoutNavigationConfig {\r\n  /** Flat list of all navigation items. */\r\n  items: UiNavItem[];\r\n  /** Optional grouping into labelled sidebar sections. */\r\n  sections?: UiNavSection[];\r\n  /** Whether the desktop sidebar can collapse to icon-only mode. @default true */\r\n  collapsible?: boolean;\r\n  /** Mobile navigation behaviour. @default 'both' */\r\n  mobileMode?: 'drawer' | 'bottom-nav' | 'both';\r\n  /** Auto-discovery settings for route-driven navigation. */\r\n  autoDiscovery?: UiNavAutoDiscoveryConfig;\r\n}\r\n\r\n// ─── Header ──────────────────────────────────────────────────────────\r\n\r\n/** Logo configuration for the layout header. */\r\nexport interface UiLayoutLogo {\r\n  /** Short text rendered inside a styled badge (e.g. \"UI\"). Mutually exclusive with `src`. */\r\n  text?: string;\r\n  /** Image URL for the logo. Mutually exclusive with `text`. */\r\n  src?: string;\r\n  /** Alt text for the logo image. */\r\n  alt?: string;\r\n}\r\n\r\n/** Header / top-bar configuration. */\r\nexport interface UiLayoutHeaderConfig {\r\n  /** Logo displayed in the sidebar header and mobile top-bar. */\r\n  logo?: UiLayoutLogo;\r\n  /** Application title. */\r\n  title: string;\r\n  /** Optional subtitle or tagline. */\r\n  subtitle?: string;\r\n}\r\n\r\n// ─── Footer ──────────────────────────────────────────────────────────\r\n\r\n/** Single link rendered in the layout footer. */\r\nexport interface UiFooterLink {\r\n  /** Display label. */\r\n  label: string;\r\n  /** External URL. */\r\n  href?: string;\r\n  /** Angular router path (alternative to `href`). */\r\n  route?: string;\r\n  /** Lucide icon name. */\r\n  icon?: UiIconName;\r\n  /** Link target for external links. @default '_self' */\r\n  target?: '_blank' | '_self';\r\n}\r\n\r\n/** Footer configuration. */\r\nexport interface UiLayoutFooterConfig {\r\n  /** Links rendered in the footer. */\r\n  links?: UiFooterLink[];\r\n  /** Static text (e.g. copyright, version). */\r\n  text?: string;\r\n}\r\n\r\n// ─── FAB (Floating Action Button / Speed Dial) ───────────────────────\r\n\r\n/**\r\n * Single action within the speed-dial FAB.\r\n *\r\n * @usageNotes\r\n * ```typescript\r\n * const action: UiFabAction = {\r\n *   id: 'new-user',\r\n *   label: 'Nuovo utente',\r\n *   icon: 'user',\r\n *   variant: 'primary',\r\n *   action: () => router.navigate(['/users/new']),\r\n * };\r\n * ```\r\n */\r\nexport interface UiFabAction {\r\n  /** Unique action identifier. */\r\n  id: string;\r\n  /** Tooltip / label text. */\r\n  label: string;\r\n  /** Lucide icon name. */\r\n  icon: UiIconName;\r\n  /** Callback invoked on click. */\r\n  action?: () => void;\r\n  /** Visual variant for the button. @default 'primary' */\r\n  variant?: UiVariant;\r\n  /** Accessible label (falls back to `label` if omitted). */\r\n  ariaLabel?: string;\r\n}\r\n\r\n/** Speed-dial FAB configuration. */\r\nexport interface UiLayoutFabConfig {\r\n  /** Primary FAB action (always visible). */\r\n  mainAction: UiFabAction;\r\n  /** Secondary actions revealed on expand. */\r\n  secondaryActions?: UiFabAction[];\r\n  /** Screen corner position. @default 'bottom-right' */\r\n  position?: 'bottom-right' | 'bottom-left';\r\n}\r\n\r\n// ─── Page Header ─────────────────────────────────────────────────────\r\n\r\n/** Default page-header settings applied to every route unless overridden. */\r\nexport interface UiLayoutPageHeaderConfig {\r\n  /** Whether to show the page header (breadcrumbs + title). @default true */\r\n  show?: boolean;\r\n  /** Home breadcrumb route path. @default '/' */\r\n  homeRoute?: string;\r\n  /** Whether to show the Home link in breadcrumbs. @default true */\r\n  showHome?: boolean;\r\n  /** Whether to sync `document.title` with the current page. @default false */\r\n  updateDocumentTitle?: boolean;\r\n  /** Suffix appended to `document.title`. */\r\n  titleSuffix?: string;\r\n}\r\n\r\n// ─── Loader ──────────────────────────────────────────────────────────\r\n\r\n/** Overlay loader configuration. */\r\nexport interface UiLayoutLoaderConfig {\r\n  /** Show the loader when the layout mounts (hides on first NavigationEnd). @default false */\r\n  showOnInit?: boolean;\r\n  /** Minimum display duration in milliseconds (prevents flicker). @default 0 */\r\n  minDuration?: number;\r\n}\r\n\r\n// ─── Topbar (multi-bar header) ───────────────────────────────────────\r\n\r\n/** Discriminator for topbar bar type. */\r\nexport type UiBarType = 'notification' | 'brand' | 'navigation' | 'links';\r\n\r\n/**\r\n * Notification bar displayed at the top of the page.\r\n *\r\n * @usageNotes\r\n * ```typescript\r\n * { text: 'Manutenzione prevista alle 22:00', variant: 'warn', dismissible: true }\r\n * ```\r\n */\r\nexport interface UiNotificationBarConfig {\r\n  /** Notification message text. */\r\n  text: string;\r\n  /** Visual variant. @default 'primary' */\r\n  variant?: UiVariant;\r\n  /** Whether the user can dismiss this bar. @default true */\r\n  dismissible?: boolean;\r\n  /** Lucide icon shown before the text. */\r\n  icon?: UiIconName;\r\n}\r\n\r\n/**\r\n * Brand bar with logo, title, and optional user dropdown.\r\n *\r\n * @usageNotes\r\n * ```typescript\r\n * {\r\n *   logo: { src: '/assets/logo.svg', alt: 'My App' },\r\n *   title: 'My App',\r\n *   userDropdown: {\r\n *     label: 'Mario Rossi',\r\n *     avatar: { src: '/avatar.jpg' },\r\n *     items: [\r\n *       { id: 'profile', label: 'Profilo', icon: 'user', route: '/profile' },\r\n *       { id: 'logout', label: 'Logout', icon: 'log-out', action: () => auth.logout() },\r\n *     ],\r\n *   },\r\n * }\r\n * ```\r\n */\r\nexport interface UiBrandBarConfig {\r\n  /** Logo displayed on the left. */\r\n  logo?: UiLayoutLogo;\r\n  /** Application title. */\r\n  title?: string;\r\n  /** User dropdown aligned to the right. */\r\n  userDropdown?: UiUserDropdownConfig;\r\n}\r\n\r\n/** User dropdown menu configuration for the brand bar. */\r\nexport interface UiUserDropdownConfig {\r\n  /** Display label (e.g. user's name). */\r\n  label: string;\r\n  /** Lucide icon name (fallback when no avatar). */\r\n  icon?: UiIconName;\r\n  /** User avatar image. */\r\n  avatar?: { src: string; alt?: string };\r\n  /** Dropdown menu items. */\r\n  items: UiDropdownItem[];\r\n}\r\n\r\n/** Single item in a dropdown menu. */\r\nexport interface UiDropdownItem {\r\n  /** Unique item identifier. */\r\n  id: string;\r\n  /** Display label. */\r\n  label: string;\r\n  /** Lucide icon name. */\r\n  icon?: UiIconName;\r\n  /** Angular router path. */\r\n  route?: string;\r\n  /** Callback invoked on click. */\r\n  action?: () => void;\r\n  /** Render a divider above this item. */\r\n  divider?: boolean;\r\n}\r\n\r\n/** Navigation bar (used in topbar mode to render horizontal nav links). */\r\nexport interface UiNavigationBarConfig {\r\n  /** Override items for this bar (defaults to `schema.navigation.items`). */\r\n  items?: UiNavItem[];\r\n}\r\n\r\n/**\r\n * Links bar with icon links (e.g. social media, external resources).\r\n *\r\n * @usageNotes\r\n * ```typescript\r\n * {\r\n *   align: 'end',\r\n *   items: [\r\n *     { icon: 'github', href: 'https://github.com/...', ariaLabel: 'GitHub' },\r\n *     { label: 'Docs', icon: 'book-open', href: 'https://docs.example.com' },\r\n *   ],\r\n * }\r\n * ```\r\n */\r\nexport interface UiLinksBarConfig {\r\n  /** Horizontal alignment of items. @default 'end' */\r\n  align?: 'start' | 'center' | 'end' | 'space-between';\r\n  /** Link items. */\r\n  items: UiBarLinkItem[];\r\n}\r\n\r\n/** Single link item in a links bar. */\r\nexport interface UiBarLinkItem {\r\n  /** Display label (optional, icon-only if omitted). */\r\n  label?: string;\r\n  /** Lucide icon name. */\r\n  icon?: UiIconName;\r\n  /** External URL. */\r\n  href?: string;\r\n  /** Angular router path. */\r\n  route?: string;\r\n  /** Link target. @default '_self' */\r\n  target?: '_blank' | '_self';\r\n  /** Accessible label for icon-only items. */\r\n  ariaLabel?: string;\r\n}\r\n\r\n/**\r\n * Single bar descriptor in the topbar bar stack.\r\n *\r\n * @usageNotes\r\n * ```typescript\r\n * const bars: UiBarConfig[] = [\r\n *   { id: 'promo', type: 'notification', notification: { text: 'Promo attiva!', dismissible: true } },\r\n *   { id: 'main', type: 'brand', brand: { logo: { text: 'UI' }, title: 'My App' } },\r\n *   { id: 'nav', type: 'navigation' },\r\n *   { id: 'social', type: 'links', links: { align: 'end', items: [...] } },\r\n * ];\r\n * ```\r\n */\r\nexport interface UiBarConfig {\r\n  /** Unique bar identifier. */\r\n  id: string;\r\n  /** Bar type discriminator. */\r\n  type: UiBarType;\r\n  /** Whether the bar is initially visible. @default true */\r\n  visible?: boolean;\r\n  /** Configuration when `type === 'notification'`. */\r\n  notification?: UiNotificationBarConfig;\r\n  /** Configuration when `type === 'brand'`. */\r\n  brand?: UiBrandBarConfig;\r\n  /** Configuration when `type === 'navigation'`. */\r\n  navigation?: UiNavigationBarConfig;\r\n  /** Configuration when `type === 'links'`. */\r\n  links?: UiLinksBarConfig;\r\n}\r\n\r\n/** Topbar configuration (bar stack). */\r\nexport interface UiTopbarConfig {\r\n  /** Ordered list of bars rendered top-to-bottom. */\r\n  bars: UiBarConfig[];\r\n}\r\n\r\n// ─── Root Schema ─────────────────────────────────────────────────────\r\n\r\n/**\r\n * Root descriptor for the `ui-layout-builder` component.\r\n *\r\n * @usageNotes\r\n * ### Minimal sidebar example\r\n * ```typescript\r\n * const schema: UiLayoutSchema = {\r\n *   header: { title: 'My App' },\r\n *   navigation: {\r\n *     items: [\r\n *       { id: 'home', label: 'Home', icon: 'home', route: '/', routeActiveExact: true },\r\n *       { id: 'users', label: 'Utenti', icon: 'users', route: '/users' },\r\n *     ],\r\n *   },\r\n * };\r\n * ```\r\n *\r\n * ### Topbar mode example\r\n * ```typescript\r\n * const schema: UiLayoutSchema = {\r\n *   mode: 'topbar',\r\n *   header: { logo: { text: 'UI' }, title: 'My App' },\r\n *   navigation: { items: [...] },\r\n *   topbar: {\r\n *     bars: [\r\n *       { id: 'alert', type: 'notification', notification: { text: 'Beta!', dismissible: true } },\r\n *       { id: 'main', type: 'brand', brand: { logo: { text: 'UI' }, title: 'My App' } },\r\n *       { id: 'nav', type: 'navigation' },\r\n *     ],\r\n *   },\r\n * };\r\n * ```\r\n */\r\nexport interface UiLayoutSchema {\r\n  /** Layout mode. @default 'sidebar' */\r\n  mode?: UiLayoutMode;\r\n  /** Header / top-bar branding (used in sidebar mode and as fallback). */\r\n  header: UiLayoutHeaderConfig;\r\n  /** Navigation structure and behaviour. */\r\n  navigation: UiLayoutNavigationConfig;\r\n  /** Footer content (optional). */\r\n  footer?: UiLayoutFooterConfig;\r\n  /** Floating action button / speed dial (optional). */\r\n  fab?: UiLayoutFabConfig;\r\n  /** Page header defaults applied to every page (optional). */\r\n  pageHeader?: UiLayoutPageHeaderConfig;\r\n  /** Overlay loader configuration (optional). */\r\n  loader?: UiLayoutLoaderConfig;\r\n  /** Content area sizing strategy. @default 'fluid' */\r\n  contentType?: UiContentType;\r\n  /** Topbar configuration (only used when `mode === 'topbar'`). */\r\n  topbar?: UiTopbarConfig;\r\n}\r\n"]}
|
|
@@ -0,0 +1,239 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Central service for programmatic control of the `ui-layout-builder` shell.
|
|
3
|
+
*
|
|
4
|
+
* Provides reactive state management for:
|
|
5
|
+
* - Overlay loader (with body scroll locking)
|
|
6
|
+
* - Sidebar open / collapsed state
|
|
7
|
+
* - Context-sensitive FAB actions
|
|
8
|
+
* - Navigation badge and visibility overrides
|
|
9
|
+
* - Layout mode switching (sidebar / topbar)
|
|
10
|
+
* - Content type (fluid / boxed / fullscreen)
|
|
11
|
+
* - Topbar bar visibility (notification bars show/hide)
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```typescript
|
|
15
|
+
* export class MyPageComponent {
|
|
16
|
+
* private layout = inject(UiLayoutService);
|
|
17
|
+
*
|
|
18
|
+
* async loadData() {
|
|
19
|
+
* this.layout.showLoader();
|
|
20
|
+
* try {
|
|
21
|
+
* await this.api.fetchAll();
|
|
22
|
+
* } finally {
|
|
23
|
+
* this.layout.hideLoader();
|
|
24
|
+
* }
|
|
25
|
+
* }
|
|
26
|
+
* }
|
|
27
|
+
* ```
|
|
28
|
+
*/
|
|
29
|
+
import { Injectable, inject, PLATFORM_ID } from '@angular/core';
|
|
30
|
+
import { isPlatformBrowser } from '@angular/common';
|
|
31
|
+
import { BehaviorSubject } from 'rxjs';
|
|
32
|
+
import * as i0 from "@angular/core";
|
|
33
|
+
export class UiLayoutService {
|
|
34
|
+
constructor() {
|
|
35
|
+
this.platformId = inject(PLATFORM_ID);
|
|
36
|
+
// ── Loader ───────────────────────────────────────────────────────
|
|
37
|
+
this._loading$ = new BehaviorSubject(false);
|
|
38
|
+
/** Whether the overlay loader is currently visible. */
|
|
39
|
+
this.loading$ = this._loading$.asObservable();
|
|
40
|
+
// ── Layout Mode ─────────────────────────────────────────────────
|
|
41
|
+
this._layoutMode$ = new BehaviorSubject('sidebar');
|
|
42
|
+
/** Current layout mode (sidebar or topbar). */
|
|
43
|
+
this.layoutMode$ = this._layoutMode$.asObservable();
|
|
44
|
+
// ── Content Type ────────────────────────────────────────────────
|
|
45
|
+
this._contentType$ = new BehaviorSubject('fluid');
|
|
46
|
+
/** Current content type (fluid, boxed, fullscreen). */
|
|
47
|
+
this.contentType$ = this._contentType$.asObservable();
|
|
48
|
+
// ── Sidebar ──────────────────────────────────────────────────────
|
|
49
|
+
this._sidebarOpen$ = new BehaviorSubject(false);
|
|
50
|
+
/** Whether the mobile sidebar drawer is open. */
|
|
51
|
+
this.sidebarOpen$ = this._sidebarOpen$.asObservable();
|
|
52
|
+
this._sidebarCollapsed$ = new BehaviorSubject(false);
|
|
53
|
+
/** Whether the desktop sidebar is in collapsed (icon-only) mode. */
|
|
54
|
+
this.sidebarCollapsed$ = this._sidebarCollapsed$.asObservable();
|
|
55
|
+
// ── FAB (context-sensitive) ──────────────────────────────────────
|
|
56
|
+
this._defaultFabConfig = null;
|
|
57
|
+
this._fabConfig$ = new BehaviorSubject(null);
|
|
58
|
+
/** Current FAB configuration (schema default merged with page overrides). */
|
|
59
|
+
this.fabConfig$ = this._fabConfig$.asObservable();
|
|
60
|
+
// ── Navigation helpers ───────────────────────────────────────────
|
|
61
|
+
this._navBadges$ = new BehaviorSubject(new Map());
|
|
62
|
+
/** Observable map of nav-item badges keyed by item ID. */
|
|
63
|
+
this.navBadges$ = this._navBadges$.asObservable();
|
|
64
|
+
this._navVisibility$ = new BehaviorSubject(new Map());
|
|
65
|
+
/** Observable map of nav-item visibility overrides keyed by item ID. */
|
|
66
|
+
this.navVisibility$ = this._navVisibility$.asObservable();
|
|
67
|
+
// ── Bar visibility (topbar notification bars) ───────────────────
|
|
68
|
+
this._barVisibility$ = new BehaviorSubject(new Map());
|
|
69
|
+
/** Observable map of topbar bar visibility keyed by bar ID. */
|
|
70
|
+
this.barVisibility$ = this._barVisibility$.asObservable();
|
|
71
|
+
}
|
|
72
|
+
/** Show the full-screen overlay loader and block body scroll. */
|
|
73
|
+
showLoader() {
|
|
74
|
+
this._loading$.next(true);
|
|
75
|
+
this.setBodyOverflow(true);
|
|
76
|
+
}
|
|
77
|
+
/** Hide the overlay loader and restore body scroll. */
|
|
78
|
+
hideLoader() {
|
|
79
|
+
this._loading$.next(false);
|
|
80
|
+
this.setBodyOverflow(false);
|
|
81
|
+
}
|
|
82
|
+
/** Toggle the overlay loader. */
|
|
83
|
+
toggleLoader() {
|
|
84
|
+
if (this._loading$.value) {
|
|
85
|
+
this.hideLoader();
|
|
86
|
+
}
|
|
87
|
+
else {
|
|
88
|
+
this.showLoader();
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
/** Synchronous snapshot of the loading state. */
|
|
92
|
+
isLoading() {
|
|
93
|
+
return this._loading$.value;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Switch layout mode with loader masking the transition.
|
|
97
|
+
* Shows the loader, applies the mode change, then hides the loader
|
|
98
|
+
* after a short delay to prevent visible layout shift.
|
|
99
|
+
*/
|
|
100
|
+
setLayoutMode(mode, transitionMs = 300) {
|
|
101
|
+
if (this._layoutMode$.value === mode)
|
|
102
|
+
return;
|
|
103
|
+
this.showLoader();
|
|
104
|
+
this._layoutMode$.next(mode);
|
|
105
|
+
setTimeout(() => this.hideLoader(), transitionMs);
|
|
106
|
+
}
|
|
107
|
+
/** Synchronous snapshot of the current layout mode. */
|
|
108
|
+
getLayoutMode() {
|
|
109
|
+
return this._layoutMode$.value;
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* @internal Called by the layout-builder component to set the initial mode
|
|
113
|
+
* from the schema without triggering the loader transition.
|
|
114
|
+
*/
|
|
115
|
+
_setInitialLayoutMode(mode) {
|
|
116
|
+
this._layoutMode$.next(mode);
|
|
117
|
+
}
|
|
118
|
+
/** Set the content type per-route or globally. */
|
|
119
|
+
setContentType(type) {
|
|
120
|
+
this._contentType$.next(type);
|
|
121
|
+
}
|
|
122
|
+
/** Synchronous snapshot of the current content type. */
|
|
123
|
+
getContentType() {
|
|
124
|
+
return this._contentType$.value;
|
|
125
|
+
}
|
|
126
|
+
/** @internal */
|
|
127
|
+
_setInitialContentType(type) {
|
|
128
|
+
this._contentType$.next(type);
|
|
129
|
+
}
|
|
130
|
+
/** Toggle the desktop sidebar between expanded and collapsed. */
|
|
131
|
+
toggleSidebar() {
|
|
132
|
+
this._sidebarCollapsed$.next(!this._sidebarCollapsed$.value);
|
|
133
|
+
}
|
|
134
|
+
/** Set the desktop sidebar collapsed state explicitly. */
|
|
135
|
+
collapseSidebar(collapsed = true) {
|
|
136
|
+
this._sidebarCollapsed$.next(collapsed);
|
|
137
|
+
}
|
|
138
|
+
/** Open the mobile sidebar drawer. */
|
|
139
|
+
openMobileSidebar() {
|
|
140
|
+
this._sidebarOpen$.next(true);
|
|
141
|
+
}
|
|
142
|
+
/** Close the mobile sidebar drawer. */
|
|
143
|
+
closeMobileSidebar() {
|
|
144
|
+
this._sidebarOpen$.next(false);
|
|
145
|
+
}
|
|
146
|
+
/** Synchronous snapshot: is the mobile sidebar open? */
|
|
147
|
+
isSidebarOpen() {
|
|
148
|
+
return this._sidebarOpen$.value;
|
|
149
|
+
}
|
|
150
|
+
/** Synchronous snapshot: is the desktop sidebar collapsed? */
|
|
151
|
+
isSidebarCollapsed() {
|
|
152
|
+
return this._sidebarCollapsed$.value;
|
|
153
|
+
}
|
|
154
|
+
/**
|
|
155
|
+
* Called internally by the layout-builder component to register the
|
|
156
|
+
* schema-level default FAB config.
|
|
157
|
+
* @internal
|
|
158
|
+
*/
|
|
159
|
+
_setDefaultFabConfig(config) {
|
|
160
|
+
this._defaultFabConfig = config;
|
|
161
|
+
this._fabConfig$.next(config);
|
|
162
|
+
}
|
|
163
|
+
/** Override the FAB main action for the current page context. */
|
|
164
|
+
setMainFabAction(action) {
|
|
165
|
+
const current = this._fabConfig$.value;
|
|
166
|
+
this._fabConfig$.next({
|
|
167
|
+
mainAction: action,
|
|
168
|
+
secondaryActions: current?.secondaryActions ?? [],
|
|
169
|
+
position: current?.position ?? 'bottom-right',
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
/** Override all FAB actions for the current page context. */
|
|
173
|
+
setFabActions(main, secondary) {
|
|
174
|
+
const current = this._fabConfig$.value;
|
|
175
|
+
this._fabConfig$.next({
|
|
176
|
+
mainAction: main,
|
|
177
|
+
secondaryActions: secondary ?? [],
|
|
178
|
+
position: current?.position ?? 'bottom-right',
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
/** Reset FAB to the schema-level default configuration. */
|
|
182
|
+
clearFabActions() {
|
|
183
|
+
this._fabConfig$.next(this._defaultFabConfig);
|
|
184
|
+
}
|
|
185
|
+
/** Set or clear a badge on a navigation item. Pass `null` to remove. */
|
|
186
|
+
setNavBadge(itemId, badge) {
|
|
187
|
+
const map = new Map(this._navBadges$.value);
|
|
188
|
+
if (badge === null) {
|
|
189
|
+
map.delete(itemId);
|
|
190
|
+
}
|
|
191
|
+
else {
|
|
192
|
+
map.set(itemId, badge);
|
|
193
|
+
}
|
|
194
|
+
this._navBadges$.next(map);
|
|
195
|
+
}
|
|
196
|
+
/** Override visibility of a navigation item. */
|
|
197
|
+
setNavItemVisibility(itemId, visible) {
|
|
198
|
+
const map = new Map(this._navVisibility$.value);
|
|
199
|
+
map.set(itemId, visible);
|
|
200
|
+
this._navVisibility$.next(map);
|
|
201
|
+
}
|
|
202
|
+
/** Show a topbar bar by ID. */
|
|
203
|
+
showBar(barId) {
|
|
204
|
+
const map = new Map(this._barVisibility$.value);
|
|
205
|
+
map.set(barId, true);
|
|
206
|
+
this._barVisibility$.next(map);
|
|
207
|
+
}
|
|
208
|
+
/** Hide a topbar bar by ID. */
|
|
209
|
+
hideBar(barId) {
|
|
210
|
+
const map = new Map(this._barVisibility$.value);
|
|
211
|
+
map.set(barId, false);
|
|
212
|
+
this._barVisibility$.next(map);
|
|
213
|
+
}
|
|
214
|
+
/** Toggle visibility of a topbar bar by ID. */
|
|
215
|
+
toggleBar(barId) {
|
|
216
|
+
const map = new Map(this._barVisibility$.value);
|
|
217
|
+
const current = map.get(barId);
|
|
218
|
+
map.set(barId, current === undefined ? false : !current);
|
|
219
|
+
this._barVisibility$.next(map);
|
|
220
|
+
}
|
|
221
|
+
/** Synchronous check of bar visibility (schema `visible` fallback). */
|
|
222
|
+
isBarVisible(barId, schemaVisible = true) {
|
|
223
|
+
const override = this._barVisibility$.value.get(barId);
|
|
224
|
+
return override !== undefined ? override : schemaVisible;
|
|
225
|
+
}
|
|
226
|
+
// ── Private helpers ──────────────────────────────────────────────
|
|
227
|
+
setBodyOverflow(locked) {
|
|
228
|
+
if (isPlatformBrowser(this.platformId)) {
|
|
229
|
+
document.body.style.overflow = locked ? 'hidden' : '';
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiLayoutService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
233
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiLayoutService, providedIn: 'root' }); }
|
|
234
|
+
}
|
|
235
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiLayoutService, decorators: [{
|
|
236
|
+
type: Injectable,
|
|
237
|
+
args: [{ providedIn: 'root' }]
|
|
238
|
+
}] });
|
|
239
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"layout.service.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/layout-builder/layout.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Central service for programmatic control of the `ui-layout-builder` shell.\r\n *\r\n * Provides reactive state management for:\r\n * - Overlay loader (with body scroll locking)\r\n * - Sidebar open / collapsed state\r\n * - Context-sensitive FAB actions\r\n * - Navigation badge and visibility overrides\r\n * - Layout mode switching (sidebar / topbar)\r\n * - Content type (fluid / boxed / fullscreen)\r\n * - Topbar bar visibility (notification bars show/hide)\r\n *\r\n * @example\r\n * ```typescript\r\n * export class MyPageComponent {\r\n *   private layout = inject(UiLayoutService);\r\n *\r\n *   async loadData() {\r\n *     this.layout.showLoader();\r\n *     try {\r\n *       await this.api.fetchAll();\r\n *     } finally {\r\n *       this.layout.hideLoader();\r\n *     }\r\n *   }\r\n * }\r\n * ```\r\n */\r\nimport { Injectable, inject, PLATFORM_ID } from '@angular/core';\r\nimport { isPlatformBrowser } from '@angular/common';\r\nimport { BehaviorSubject, Observable } from 'rxjs';\r\nimport {\r\n  UiFabAction,\r\n  UiLayoutFabConfig,\r\n  UiLayoutMode,\r\n  UiContentType,\r\n} from './layout-builder.types';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class UiLayoutService {\r\n  private readonly platformId = inject(PLATFORM_ID);\r\n\r\n  // ── Loader ───────────────────────────────────────────────────────\r\n\r\n  private readonly _loading$ = new BehaviorSubject<boolean>(false);\r\n  /** Whether the overlay loader is currently visible. */\r\n  readonly loading$: Observable<boolean> = this._loading$.asObservable();\r\n\r\n  /** Show the full-screen overlay loader and block body scroll. */\r\n  showLoader(): void {\r\n    this._loading$.next(true);\r\n    this.setBodyOverflow(true);\r\n  }\r\n\r\n  /** Hide the overlay loader and restore body scroll. */\r\n  hideLoader(): void {\r\n    this._loading$.next(false);\r\n    this.setBodyOverflow(false);\r\n  }\r\n\r\n  /** Toggle the overlay loader. */\r\n  toggleLoader(): void {\r\n    if (this._loading$.value) {\r\n      this.hideLoader();\r\n    } else {\r\n      this.showLoader();\r\n    }\r\n  }\r\n\r\n  /** Synchronous snapshot of the loading state. */\r\n  isLoading(): boolean {\r\n    return this._loading$.value;\r\n  }\r\n\r\n  // ── Layout Mode ─────────────────────────────────────────────────\r\n\r\n  private readonly _layoutMode$ = new BehaviorSubject<UiLayoutMode>('sidebar');\r\n  /** Current layout mode (sidebar or topbar). */\r\n  readonly layoutMode$: Observable<UiLayoutMode> = this._layoutMode$.asObservable();\r\n\r\n  /**\r\n   * Switch layout mode with loader masking the transition.\r\n   * Shows the loader, applies the mode change, then hides the loader\r\n   * after a short delay to prevent visible layout shift.\r\n   */\r\n  setLayoutMode(mode: UiLayoutMode, transitionMs = 300): void {\r\n    if (this._layoutMode$.value === mode) return;\r\n    this.showLoader();\r\n    this._layoutMode$.next(mode);\r\n    setTimeout(() => this.hideLoader(), transitionMs);\r\n  }\r\n\r\n  /** Synchronous snapshot of the current layout mode. */\r\n  getLayoutMode(): UiLayoutMode {\r\n    return this._layoutMode$.value;\r\n  }\r\n\r\n  /**\r\n   * @internal Called by the layout-builder component to set the initial mode\r\n   * from the schema without triggering the loader transition.\r\n   */\r\n  _setInitialLayoutMode(mode: UiLayoutMode): void {\r\n    this._layoutMode$.next(mode);\r\n  }\r\n\r\n  // ── Content Type ────────────────────────────────────────────────\r\n\r\n  private readonly _contentType$ = new BehaviorSubject<UiContentType>('fluid');\r\n  /** Current content type (fluid, boxed, fullscreen). */\r\n  readonly contentType$: Observable<UiContentType> = this._contentType$.asObservable();\r\n\r\n  /** Set the content type per-route or globally. */\r\n  setContentType(type: UiContentType): void {\r\n    this._contentType$.next(type);\r\n  }\r\n\r\n  /** Synchronous snapshot of the current content type. */\r\n  getContentType(): UiContentType {\r\n    return this._contentType$.value;\r\n  }\r\n\r\n  /** @internal */\r\n  _setInitialContentType(type: UiContentType): void {\r\n    this._contentType$.next(type);\r\n  }\r\n\r\n  // ── Sidebar ──────────────────────────────────────────────────────\r\n\r\n  private readonly _sidebarOpen$ = new BehaviorSubject<boolean>(false);\r\n  /** Whether the mobile sidebar drawer is open. */\r\n  readonly sidebarOpen$: Observable<boolean> = this._sidebarOpen$.asObservable();\r\n\r\n  private readonly _sidebarCollapsed$ = new BehaviorSubject<boolean>(false);\r\n  /** Whether the desktop sidebar is in collapsed (icon-only) mode. */\r\n  readonly sidebarCollapsed$: Observable<boolean> = this._sidebarCollapsed$.asObservable();\r\n\r\n  /** Toggle the desktop sidebar between expanded and collapsed. */\r\n  toggleSidebar(): void {\r\n    this._sidebarCollapsed$.next(!this._sidebarCollapsed$.value);\r\n  }\r\n\r\n  /** Set the desktop sidebar collapsed state explicitly. */\r\n  collapseSidebar(collapsed = true): void {\r\n    this._sidebarCollapsed$.next(collapsed);\r\n  }\r\n\r\n  /** Open the mobile sidebar drawer. */\r\n  openMobileSidebar(): void {\r\n    this._sidebarOpen$.next(true);\r\n  }\r\n\r\n  /** Close the mobile sidebar drawer. */\r\n  closeMobileSidebar(): void {\r\n    this._sidebarOpen$.next(false);\r\n  }\r\n\r\n  /** Synchronous snapshot: is the mobile sidebar open? */\r\n  isSidebarOpen(): boolean {\r\n    return this._sidebarOpen$.value;\r\n  }\r\n\r\n  /** Synchronous snapshot: is the desktop sidebar collapsed? */\r\n  isSidebarCollapsed(): boolean {\r\n    return this._sidebarCollapsed$.value;\r\n  }\r\n\r\n  // ── FAB (context-sensitive) ──────────────────────────────────────\r\n\r\n  private _defaultFabConfig: UiLayoutFabConfig | null = null;\r\n  private readonly _fabConfig$ = new BehaviorSubject<UiLayoutFabConfig | null>(null);\r\n  /** Current FAB configuration (schema default merged with page overrides). */\r\n  readonly fabConfig$: Observable<UiLayoutFabConfig | null> = this._fabConfig$.asObservable();\r\n\r\n  /**\r\n   * Called internally by the layout-builder component to register the\r\n   * schema-level default FAB config.\r\n   * @internal\r\n   */\r\n  _setDefaultFabConfig(config: UiLayoutFabConfig | null): void {\r\n    this._defaultFabConfig = config;\r\n    this._fabConfig$.next(config);\r\n  }\r\n\r\n  /** Override the FAB main action for the current page context. */\r\n  setMainFabAction(action: UiFabAction): void {\r\n    const current = this._fabConfig$.value;\r\n    this._fabConfig$.next({\r\n      mainAction: action,\r\n      secondaryActions: current?.secondaryActions ?? [],\r\n      position: current?.position ?? 'bottom-right',\r\n    });\r\n  }\r\n\r\n  /** Override all FAB actions for the current page context. */\r\n  setFabActions(main: UiFabAction, secondary?: UiFabAction[]): void {\r\n    const current = this._fabConfig$.value;\r\n    this._fabConfig$.next({\r\n      mainAction: main,\r\n      secondaryActions: secondary ?? [],\r\n      position: current?.position ?? 'bottom-right',\r\n    });\r\n  }\r\n\r\n  /** Reset FAB to the schema-level default configuration. */\r\n  clearFabActions(): void {\r\n    this._fabConfig$.next(this._defaultFabConfig);\r\n  }\r\n\r\n  // ── Navigation helpers ───────────────────────────────────────────\r\n\r\n  private readonly _navBadges$ = new BehaviorSubject<Map<string, string | number | null>>(new Map());\r\n  /** Observable map of nav-item badges keyed by item ID. */\r\n  readonly navBadges$: Observable<Map<string, string | number | null>> = this._navBadges$.asObservable();\r\n\r\n  private readonly _navVisibility$ = new BehaviorSubject<Map<string, boolean>>(new Map());\r\n  /** Observable map of nav-item visibility overrides keyed by item ID. */\r\n  readonly navVisibility$: Observable<Map<string, boolean>> = this._navVisibility$.asObservable();\r\n\r\n  /** Set or clear a badge on a navigation item. Pass `null` to remove. */\r\n  setNavBadge(itemId: string, badge: string | number | null): void {\r\n    const map = new Map(this._navBadges$.value);\r\n    if (badge === null) {\r\n      map.delete(itemId);\r\n    } else {\r\n      map.set(itemId, badge);\r\n    }\r\n    this._navBadges$.next(map);\r\n  }\r\n\r\n  /** Override visibility of a navigation item. */\r\n  setNavItemVisibility(itemId: string, visible: boolean): void {\r\n    const map = new Map(this._navVisibility$.value);\r\n    map.set(itemId, visible);\r\n    this._navVisibility$.next(map);\r\n  }\r\n\r\n  // ── Bar visibility (topbar notification bars) ───────────────────\r\n\r\n  private readonly _barVisibility$ = new BehaviorSubject<Map<string, boolean>>(new Map());\r\n  /** Observable map of topbar bar visibility keyed by bar ID. */\r\n  readonly barVisibility$: Observable<Map<string, boolean>> = this._barVisibility$.asObservable();\r\n\r\n  /** Show a topbar bar by ID. */\r\n  showBar(barId: string): void {\r\n    const map = new Map(this._barVisibility$.value);\r\n    map.set(barId, true);\r\n    this._barVisibility$.next(map);\r\n  }\r\n\r\n  /** Hide a topbar bar by ID. */\r\n  hideBar(barId: string): void {\r\n    const map = new Map(this._barVisibility$.value);\r\n    map.set(barId, false);\r\n    this._barVisibility$.next(map);\r\n  }\r\n\r\n  /** Toggle visibility of a topbar bar by ID. */\r\n  toggleBar(barId: string): void {\r\n    const map = new Map(this._barVisibility$.value);\r\n    const current = map.get(barId);\r\n    map.set(barId, current === undefined ? false : !current);\r\n    this._barVisibility$.next(map);\r\n  }\r\n\r\n  /** Synchronous check of bar visibility (schema `visible` fallback). */\r\n  isBarVisible(barId: string, schemaVisible = true): boolean {\r\n    const override = this._barVisibility$.value.get(barId);\r\n    return override !== undefined ? override : schemaVisible;\r\n  }\r\n\r\n  // ── Private helpers ──────────────────────────────────────────────\r\n\r\n  private setBodyOverflow(locked: boolean): void {\r\n    if (isPlatformBrowser(this.platformId)) {\r\n      document.body.style.overflow = locked ? 'hidden' : '';\r\n    }\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
import { Component, ChangeDetectionStrategy, ViewEncapsulation, inject } from '@angular/core';
|
|
2
|
+
import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';
|
|
3
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
4
|
+
import { UiButtonComponent } from '../button/button.component';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "lucide-angular";
|
|
7
|
+
/** @internal Mappatura variante -> icona Lucide di default. */
|
|
8
|
+
const VARIANT_ICON_MAP = {
|
|
9
|
+
confirm: 'check-circle',
|
|
10
|
+
delete: 'trash-2',
|
|
11
|
+
warn: 'alert-triangle',
|
|
12
|
+
info: 'info',
|
|
13
|
+
};
|
|
14
|
+
/** @internal Mappatura variante -> colore pulsante di conferma. */
|
|
15
|
+
const VARIANT_BUTTON_MAP = {
|
|
16
|
+
confirm: 'primary',
|
|
17
|
+
delete: 'warn',
|
|
18
|
+
warn: 'warn',
|
|
19
|
+
info: 'primary',
|
|
20
|
+
};
|
|
21
|
+
/**
|
|
22
|
+
* Dialogo di conferma leggero per azioni condizionali.
|
|
23
|
+
*
|
|
24
|
+
* Non va utilizzato direttamente nel template: viene aperto
|
|
25
|
+
* programmaticamente tramite `UiModalService.confirm()` e i suoi
|
|
26
|
+
* metodi scorciatoia (`confirmDelete`, `confirmDiscard`, `confirmSave`).
|
|
27
|
+
*
|
|
28
|
+
* Supporta quattro varianti semantiche — confirm, delete, warn, info —
|
|
29
|
+
* ciascuna con icona e colore preconfigurati.
|
|
30
|
+
*
|
|
31
|
+
* @selector ui-confirm-dialog
|
|
32
|
+
*
|
|
33
|
+
* @example
|
|
34
|
+
* ```typescript
|
|
35
|
+
* // Apertura tramite servizio
|
|
36
|
+
* this.modalService.confirm({
|
|
37
|
+
* title: 'Conferma eliminazione',
|
|
38
|
+
* message: 'Questa azione non puo essere annullata.',
|
|
39
|
+
* variant: 'delete',
|
|
40
|
+
* }).subscribe(confirmed => {
|
|
41
|
+
* if (confirmed) this.deleteItem();
|
|
42
|
+
* });
|
|
43
|
+
* ```
|
|
44
|
+
*/
|
|
45
|
+
export class UiConfirmDialogComponent {
|
|
46
|
+
constructor() {
|
|
47
|
+
/** @internal Dati iniettati dal servizio modale. */
|
|
48
|
+
this.data = inject(MAT_DIALOG_DATA);
|
|
49
|
+
/** @internal Riferimento al dialogo per la chiusura. */
|
|
50
|
+
this.dialogRef = inject(MatDialogRef);
|
|
51
|
+
}
|
|
52
|
+
/** Variante semantica corrente. */
|
|
53
|
+
get variant() {
|
|
54
|
+
return this.data.variant || 'confirm';
|
|
55
|
+
}
|
|
56
|
+
/** Nome icona Lucide da visualizzare. */
|
|
57
|
+
get iconName() {
|
|
58
|
+
return this.data.icon || VARIANT_ICON_MAP[this.variant];
|
|
59
|
+
}
|
|
60
|
+
/** Variante del pulsante di conferma. */
|
|
61
|
+
get confirmButtonVariant() {
|
|
62
|
+
return VARIANT_BUTTON_MAP[this.variant];
|
|
63
|
+
}
|
|
64
|
+
/** Classi CSS per il contenitore icona. */
|
|
65
|
+
get iconClasses() {
|
|
66
|
+
return `ui-confirm-dialog__icon ui-confirm-dialog__icon--${this.variant}`;
|
|
67
|
+
}
|
|
68
|
+
/** Conferma e chiude il dialogo con `true`. */
|
|
69
|
+
confirm() {
|
|
70
|
+
this.dialogRef.close(true);
|
|
71
|
+
}
|
|
72
|
+
/** Annulla e chiude il dialogo con `false`. */
|
|
73
|
+
cancel() {
|
|
74
|
+
this.dialogRef.close(false);
|
|
75
|
+
}
|
|
76
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiConfirmDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
77
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiConfirmDialogComponent, isStandalone: true, selector: "ui-confirm-dialog", host: { attributes: { "role": "alertdialog", "aria-modal": "true" }, properties: { "attr.aria-labelledby": "\"ui-confirm-title\"", "attr.aria-describedby": "\"ui-confirm-message\"" }, classAttribute: "ui-confirm-dialog-host" }, ngImport: i0, template: `
|
|
78
|
+
<div class="ui-confirm-dialog">
|
|
79
|
+
<!-- Icona -->
|
|
80
|
+
<div [class]="iconClasses">
|
|
81
|
+
<lucide-icon [name]="iconName" [size]="32" aria-hidden="true" />
|
|
82
|
+
</div>
|
|
83
|
+
|
|
84
|
+
<!-- Contenuto -->
|
|
85
|
+
<div class="ui-confirm-dialog__content">
|
|
86
|
+
@if (data.title) {
|
|
87
|
+
<h3 class="ui-confirm-dialog__title" id="ui-confirm-title">{{ data.title }}</h3>
|
|
88
|
+
}
|
|
89
|
+
<p class="ui-confirm-dialog__message" id="ui-confirm-message">{{ data.message }}</p>
|
|
90
|
+
</div>
|
|
91
|
+
|
|
92
|
+
<!-- Azioni -->
|
|
93
|
+
<div class="ui-confirm-dialog__actions">
|
|
94
|
+
<ui-button
|
|
95
|
+
[label]="data.cancelText || 'Annulla'"
|
|
96
|
+
variant="ghost"
|
|
97
|
+
size="md"
|
|
98
|
+
(click)="cancel()"
|
|
99
|
+
/>
|
|
100
|
+
<ui-button
|
|
101
|
+
[label]="data.confirmText || 'Conferma'"
|
|
102
|
+
[variant]="confirmButtonVariant"
|
|
103
|
+
size="md"
|
|
104
|
+
(click)="confirm()"
|
|
105
|
+
/>
|
|
106
|
+
</div>
|
|
107
|
+
</div>
|
|
108
|
+
`, isInline: true, styles: [".ui-confirm-dialog-host{display:block}.ui-confirm-dialog{display:flex;flex-direction:column;align-items:center;text-align:center;padding:var(--ui-spacing-8) var(--ui-spacing-6);background:var(--ui-color-surface);border-radius:var(--ui-radius-lg);box-shadow:var(--ui-shadow-lg)}.ui-confirm-dialog__icon{display:flex;align-items:center;justify-content:center;width:56px;height:56px;border-radius:var(--ui-radius-full);margin-bottom:var(--ui-spacing-4)}.ui-confirm-dialog__icon--confirm{background:var(--ui-color-primary-light);color:var(--ui-color-primary)}.ui-confirm-dialog__icon--delete{background:var(--ui-color-warn-light);color:var(--ui-color-warn)}.ui-confirm-dialog__icon--warn{background:var(--ui-color-warning-light);color:var(--ui-color-warning)}.ui-confirm-dialog__icon--info{background:var(--ui-color-info-light);color:var(--ui-color-info)}.ui-confirm-dialog__content{margin-bottom:var(--ui-spacing-6)}.ui-confirm-dialog__title{margin:0 0 var(--ui-spacing-2);font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);line-height:var(--ui-line-height-tight)}.ui-confirm-dialog__message{margin:0;font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);line-height:var(--ui-line-height-normal);max-width:340px}.ui-confirm-dialog__actions{display:flex;gap:var(--ui-spacing-2);width:100%;justify-content:center}.ui-confirm-panel .mat-mdc-dialog-container{--mdc-dialog-container-color: transparent;--mdc-dialog-container-shape: var(--ui-radius-lg)}.ui-confirm-panel .mat-mdc-dialog-container .mdc-dialog__surface{background:transparent!important;box-shadow:none!important;border-radius:var(--ui-radius-lg)!important;overflow:visible}@media (max-width: 480px){.ui-confirm-dialog{padding:var(--ui-spacing-6) var(--ui-spacing-4)}.ui-confirm-dialog__actions{flex-direction:column-reverse;gap:var(--ui-spacing-2)}.ui-confirm-dialog__actions .ui-button-host{width:100%}}\n"], dependencies: [{ kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i1.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { 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 }); }
|
|
109
|
+
}
|
|
110
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiConfirmDialogComponent, decorators: [{
|
|
111
|
+
type: Component,
|
|
112
|
+
args: [{ selector: 'ui-confirm-dialog', standalone: true, imports: [LucideAngularModule, UiButtonComponent], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
|
|
113
|
+
class: 'ui-confirm-dialog-host',
|
|
114
|
+
role: 'alertdialog',
|
|
115
|
+
'aria-modal': 'true',
|
|
116
|
+
'[attr.aria-labelledby]': '"ui-confirm-title"',
|
|
117
|
+
'[attr.aria-describedby]': '"ui-confirm-message"',
|
|
118
|
+
}, template: `
|
|
119
|
+
<div class="ui-confirm-dialog">
|
|
120
|
+
<!-- Icona -->
|
|
121
|
+
<div [class]="iconClasses">
|
|
122
|
+
<lucide-icon [name]="iconName" [size]="32" aria-hidden="true" />
|
|
123
|
+
</div>
|
|
124
|
+
|
|
125
|
+
<!-- Contenuto -->
|
|
126
|
+
<div class="ui-confirm-dialog__content">
|
|
127
|
+
@if (data.title) {
|
|
128
|
+
<h3 class="ui-confirm-dialog__title" id="ui-confirm-title">{{ data.title }}</h3>
|
|
129
|
+
}
|
|
130
|
+
<p class="ui-confirm-dialog__message" id="ui-confirm-message">{{ data.message }}</p>
|
|
131
|
+
</div>
|
|
132
|
+
|
|
133
|
+
<!-- Azioni -->
|
|
134
|
+
<div class="ui-confirm-dialog__actions">
|
|
135
|
+
<ui-button
|
|
136
|
+
[label]="data.cancelText || 'Annulla'"
|
|
137
|
+
variant="ghost"
|
|
138
|
+
size="md"
|
|
139
|
+
(click)="cancel()"
|
|
140
|
+
/>
|
|
141
|
+
<ui-button
|
|
142
|
+
[label]="data.confirmText || 'Conferma'"
|
|
143
|
+
[variant]="confirmButtonVariant"
|
|
144
|
+
size="md"
|
|
145
|
+
(click)="confirm()"
|
|
146
|
+
/>
|
|
147
|
+
</div>
|
|
148
|
+
</div>
|
|
149
|
+
`, styles: [".ui-confirm-dialog-host{display:block}.ui-confirm-dialog{display:flex;flex-direction:column;align-items:center;text-align:center;padding:var(--ui-spacing-8) var(--ui-spacing-6);background:var(--ui-color-surface);border-radius:var(--ui-radius-lg);box-shadow:var(--ui-shadow-lg)}.ui-confirm-dialog__icon{display:flex;align-items:center;justify-content:center;width:56px;height:56px;border-radius:var(--ui-radius-full);margin-bottom:var(--ui-spacing-4)}.ui-confirm-dialog__icon--confirm{background:var(--ui-color-primary-light);color:var(--ui-color-primary)}.ui-confirm-dialog__icon--delete{background:var(--ui-color-warn-light);color:var(--ui-color-warn)}.ui-confirm-dialog__icon--warn{background:var(--ui-color-warning-light);color:var(--ui-color-warning)}.ui-confirm-dialog__icon--info{background:var(--ui-color-info-light);color:var(--ui-color-info)}.ui-confirm-dialog__content{margin-bottom:var(--ui-spacing-6)}.ui-confirm-dialog__title{margin:0 0 var(--ui-spacing-2);font-size:var(--ui-font-size-lg);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);line-height:var(--ui-line-height-tight)}.ui-confirm-dialog__message{margin:0;font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);line-height:var(--ui-line-height-normal);max-width:340px}.ui-confirm-dialog__actions{display:flex;gap:var(--ui-spacing-2);width:100%;justify-content:center}.ui-confirm-panel .mat-mdc-dialog-container{--mdc-dialog-container-color: transparent;--mdc-dialog-container-shape: var(--ui-radius-lg)}.ui-confirm-panel .mat-mdc-dialog-container .mdc-dialog__surface{background:transparent!important;box-shadow:none!important;border-radius:var(--ui-radius-lg)!important;overflow:visible}@media (max-width: 480px){.ui-confirm-dialog{padding:var(--ui-spacing-6) var(--ui-spacing-4)}.ui-confirm-dialog__actions{flex-direction:column-reverse;gap:var(--ui-spacing-2)}.ui-confirm-dialog__actions .ui-button-host{width:100%}}\n"] }]
|
|
150
|
+
}] });
|
|
151
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"confirm-dialog.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/modal/confirm-dialog.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,uBAAuB,EAAE,iBAAiB,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC9F,OAAO,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,0BAA0B,CAAC;AACzE,OAAO,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AACrD,OAAO,EAAE,iBAAiB,EAAE,MAAM,4BAA4B,CAAC;;;AAG/D,+DAA+D;AAC/D,MAAM,gBAAgB,GAAyC;IAC7D,OAAO,EAAE,cAAc;IACvB,MAAM,EAAE,SAAS;IACjB,IAAI,EAAE,gBAAgB;IACtB,IAAI,EAAE,MAAM;CACb,CAAC;AAEF,mEAAmE;AACnE,MAAM,kBAAkB,GAAwC;IAC9D,OAAO,EAAE,SAAS;IAClB,MAAM,EAAE,MAAM;IACd,IAAI,EAAE,MAAM;IACZ,IAAI,EAAE,SAAS;CAChB,CAAC;AAEF;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AAgDH,MAAM,OAAO,wBAAwB;IA/CrC;QAgDE,oDAAoD;QAC3C,SAAI,GAAwB,MAAM,CAAC,eAAe,CAAC,CAAC;QAE7D,wDAAwD;QACvC,cAAS,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;KA+BnD;IA7BC,mCAAmC;IACnC,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,IAAI,CAAC,OAAO,IAAI,SAAS,CAAC;IACxC,CAAC;IAED,yCAAyC;IACzC,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC1D,CAAC;IAED,yCAAyC;IACzC,IAAI,oBAAoB;QACtB,OAAO,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC1C,CAAC;IAED,2CAA2C;IAC3C,IAAI,WAAW;QACb,OAAO,oDAAoD,IAAI,CAAC,OAAO,EAAE,CAAC;IAC5E,CAAC;IAED,+CAA+C;IAC/C,OAAO;QACL,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAC7B,CAAC;IAED,+CAA+C;IAC/C,MAAM;QACJ,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IAC9B,CAAC;+GAnCU,wBAAwB;mGAAxB,wBAAwB,iTAlCzB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BT,m9DAzCS,mBAAmB,gPAAE,iBAAiB;;4FA4CrC,wBAAwB;kBA/CpC,SAAS;+BACE,mBAAmB,cACjB,IAAI,WACP,CAAC,mBAAmB,EAAE,iBAAiB,CAAC,mBAChC,uBAAuB,CAAC,MAAM,iBAChC,iBAAiB,CAAC,IAAI,QAC/B;wBACJ,KAAK,EAAE,wBAAwB;wBAC/B,IAAI,EAAE,aAAa;wBACnB,YAAY,EAAE,MAAM;wBACpB,wBAAwB,EAAE,oBAAoB;wBAC9C,yBAAyB,EAAE,sBAAsB;qBAClD,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BT","sourcesContent":["import { Component, ChangeDetectionStrategy, ViewEncapsulation, inject } from '@angular/core';\r\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { UiButtonComponent } from '../button/button.component';\r\nimport { UiConfirmDialogData, UiConfirmVariant, UiIconName, UiVariant } from './modal.types';\r\n\r\n/** @internal Mappatura variante -> icona Lucide di default. */\r\nconst VARIANT_ICON_MAP: Record<UiConfirmVariant, UiIconName> = {\r\n  confirm: 'check-circle',\r\n  delete: 'trash-2',\r\n  warn: 'alert-triangle',\r\n  info: 'info',\r\n};\r\n\r\n/** @internal Mappatura variante -> colore pulsante di conferma. */\r\nconst VARIANT_BUTTON_MAP: Record<UiConfirmVariant, UiVariant> = {\r\n  confirm: 'primary',\r\n  delete: 'warn',\r\n  warn: 'warn',\r\n  info: 'primary',\r\n};\r\n\r\n/**\r\n * Dialogo di conferma leggero per azioni condizionali.\r\n *\r\n * Non va utilizzato direttamente nel template: viene aperto\r\n * programmaticamente tramite `UiModalService.confirm()` e i suoi\r\n * metodi scorciatoia (`confirmDelete`, `confirmDiscard`, `confirmSave`).\r\n *\r\n * Supporta quattro varianti semantiche — confirm, delete, warn, info —\r\n * ciascuna con icona e colore preconfigurati.\r\n *\r\n * @selector ui-confirm-dialog\r\n *\r\n * @example\r\n * ```typescript\r\n * // Apertura tramite servizio\r\n * this.modalService.confirm({\r\n *   title: 'Conferma eliminazione',\r\n *   message: 'Questa azione non puo essere annullata.',\r\n *   variant: 'delete',\r\n * }).subscribe(confirmed => {\r\n *   if (confirmed) this.deleteItem();\r\n * });\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-confirm-dialog',\r\n  standalone: true,\r\n  imports: [LucideAngularModule, UiButtonComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: {\r\n    class: 'ui-confirm-dialog-host',\r\n    role: 'alertdialog',\r\n    'aria-modal': 'true',\r\n    '[attr.aria-labelledby]': '\"ui-confirm-title\"',\r\n    '[attr.aria-describedby]': '\"ui-confirm-message\"',\r\n  },\r\n  template: `\r\n    <div class=\"ui-confirm-dialog\">\r\n      <!-- Icona -->\r\n      <div [class]=\"iconClasses\">\r\n        <lucide-icon [name]=\"iconName\" [size]=\"32\" aria-hidden=\"true\" />\r\n      </div>\r\n\r\n      <!-- Contenuto -->\r\n      <div class=\"ui-confirm-dialog__content\">\r\n        @if (data.title) {\r\n          <h3 class=\"ui-confirm-dialog__title\" id=\"ui-confirm-title\">{{ data.title }}</h3>\r\n        }\r\n        <p class=\"ui-confirm-dialog__message\" id=\"ui-confirm-message\">{{ data.message }}</p>\r\n      </div>\r\n\r\n      <!-- Azioni -->\r\n      <div class=\"ui-confirm-dialog__actions\">\r\n        <ui-button\r\n          [label]=\"data.cancelText || 'Annulla'\"\r\n          variant=\"ghost\"\r\n          size=\"md\"\r\n          (click)=\"cancel()\"\r\n        />\r\n        <ui-button\r\n          [label]=\"data.confirmText || 'Conferma'\"\r\n          [variant]=\"confirmButtonVariant\"\r\n          size=\"md\"\r\n          (click)=\"confirm()\"\r\n        />\r\n      </div>\r\n    </div>\r\n  `,\r\n  styleUrl: './confirm-dialog.component.scss',\r\n})\r\nexport class UiConfirmDialogComponent {\r\n  /** @internal Dati iniettati dal servizio modale. */\r\n  readonly data: UiConfirmDialogData = inject(MAT_DIALOG_DATA);\r\n\r\n  /** @internal Riferimento al dialogo per la chiusura. */\r\n  private readonly dialogRef = inject(MatDialogRef);\r\n\r\n  /** Variante semantica corrente. */\r\n  get variant(): UiConfirmVariant {\r\n    return this.data.variant || 'confirm';\r\n  }\r\n\r\n  /** Nome icona Lucide da visualizzare. */\r\n  get iconName(): UiIconName {\r\n    return this.data.icon || VARIANT_ICON_MAP[this.variant];\r\n  }\r\n\r\n  /** Variante del pulsante di conferma. */\r\n  get confirmButtonVariant(): UiVariant {\r\n    return VARIANT_BUTTON_MAP[this.variant];\r\n  }\r\n\r\n  /** Classi CSS per il contenitore icona. */\r\n  get iconClasses(): string {\r\n    return `ui-confirm-dialog__icon ui-confirm-dialog__icon--${this.variant}`;\r\n  }\r\n\r\n  /** Conferma e chiude il dialogo con `true`. */\r\n  confirm(): void {\r\n    this.dialogRef.close(true);\r\n  }\r\n\r\n  /** Annulla e chiude il dialogo con `false`. */\r\n  cancel(): void {\r\n    this.dialogRef.close(false);\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
export { UiModalComponent } from './modal.component';
|
|
2
|
+
export { UiConfirmDialogComponent } from './confirm-dialog.component';
|
|
3
|
+
export { UiModalService } from './modal.service';
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL21vZGFsL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBQ3JELE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQ3RFLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFVpTW9kYWxDb21wb25lbnQgfSBmcm9tICcuL21vZGFsLmNvbXBvbmVudCc7XHJcbmV4cG9ydCB7IFVpQ29uZmlybURpYWxvZ0NvbXBvbmVudCB9IGZyb20gJy4vY29uZmlybS1kaWFsb2cuY29tcG9uZW50JztcclxuZXhwb3J0IHsgVWlNb2RhbFNlcnZpY2UgfSBmcm9tICcuL21vZGFsLnNlcnZpY2UnO1xyXG5leHBvcnQge1xyXG4gIFVpTW9kYWxDb25maWcsXHJcbiAgVWlNb2RhbE9wZW5Db25maWcsXHJcbiAgVWlNb2RhbFNpemUsXHJcbiAgVWlDb25maXJtRGlhbG9nRGF0YSxcclxuICBVaUNvbmZpcm1WYXJpYW50LFxyXG59IGZyb20gJy4vbW9kYWwudHlwZXMnO1xyXG4iXX0=
|