@fidurcode/dashboard-widgets-skeleton 2.0.10 → 2.0.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,6 +1,6 @@
1
1
  # @fidurcode/dashboard-widgets-skeleton
2
2
 
3
- Reusable Angular components and services for building dynamic dashboard widgets with configurable layout, sizing, colors, and interactive controls.
3
+ Reusable Angular components and services for building dynamic dashboard widgets with configurable layout, responsive sizing, and focused edit controls.
4
4
 
5
5
  ---
6
6
 
@@ -54,14 +54,14 @@ Main container component. Manages layout, persistence (localStorage), and edit m
54
54
  | `title` | `string` | _(required)_ | Dashboard header title |
55
55
  | `widgets` | `Widget[]` | _(required)_ | Available widget definitions |
56
56
  | `storageKey` | `string` | `'dashboard-widgets'`| localStorage key for persisting layout |
57
- | `columns` | `number or 'auto'` | `'auto'` | Kept for API compatibility; layout uses the 5-column widget scale |
58
- | `minTileWidth` | `number` | `200` | Kept for API compatibility |
57
+ | `columns` | `number or 'auto'` | `'auto'` | Grid column count or `'auto'` for responsive fit |
58
+ | `minTileWidth` | `number` | `200` | Minimum tile width in px (used in auto mode) |
59
59
  | `rowHeight` | `number` | `80` | Row height in px |
60
60
  | `gap` | `number` | `12` | Gap between widgets in px |
61
- | `maxColumns` | `number` | `5` | Maximum columns available in widget options |
62
- | `maxRows` | `number` | `5` | Maximum rows available in widget options |
61
+ | `maxColumns` | `number` | `4` | Maximum columns available in widget options |
62
+ | `maxRows` | `number` | `4` | Maximum rows available in widget options |
63
63
  | `widgetPadding`| `number` | `12` | Padding inside widget content area in px |
64
- | `editMode` | `boolean` | `true` | Shows widget settings and add-widget controls |
64
+ | `editMode` | `boolean` | `true` | Initial edit mode state |
65
65
 
66
66
  #### Outputs
67
67
 
@@ -83,10 +83,38 @@ Individual widget wrapper. Renders dynamic content and provides a header with ac
83
83
  |--------|----------|----------|--------------------------------|
84
84
  | `data` | `Widget` | ✅ Yes | Widget configuration object |
85
85
 
86
- #### Features
86
+ #### Content contract
87
+
88
+ Every component passed as widget content must expose a `widgetSize` input. The dashboard updates it with the actual content area available inside the widget.
89
+
90
+ ```typescript
91
+ import { Component, input, InputSignal } from '@angular/core';
92
+ import {
93
+ DashboardWidgetContentComponent,
94
+ DashboardWidgetContentSize,
95
+ } from '@fidurcode/dashboard-widgets-skeleton';
96
+
97
+ @Component({
98
+ selector: 'app-sales-widget',
99
+ template: '<section class="sales-widget">...</section>',
100
+ styles: [`
101
+ :host,
102
+ .sales-widget {
103
+ display: block;
104
+ width: 100%;
105
+ height: 100%;
106
+ min-width: 0;
107
+ min-height: 0;
108
+ box-sizing: border-box;
109
+ }
110
+ `],
111
+ })
112
+ export class SalesWidgetComponent implements DashboardWidgetContentComponent {
113
+ widgetSize: InputSignal<DashboardWidgetContentSize> = input.required();
114
+ }
115
+ ```
87
116
 
88
- - **Inline rename** double-click the title to rename (edit mode only)
89
- - **Edit mode guard** — settings button hidden when `editMode` is false
117
+ The content component should render within `width: 100%` and `height: 100%`; use the numeric `widgetSize.width`, `widgetSize.height`, `widgetSize.rows`, and `widgetSize.columns` values for charts, canvases, tables, or other components that need explicit dimensions.
90
118
 
91
119
  ---
92
120
 
@@ -98,8 +126,7 @@ Settings overlay rendered inside the widget. Visible only in edit mode.
98
126
 
99
127
  #### Features
100
128
 
101
- - **Rename** — overrides the widget label with a custom name
102
- - **Resize** — toggle group for columns (width) and rows (height), on a 1-5 scale. Width `5` spans the full row
129
+ - **Resize** — toggle group for columns (width) and rows (height)
103
130
  - **Delete** — removes widget from dashboard
104
131
  - **Move left/right** — reorders widget in the grid
105
132
 
@@ -111,13 +138,11 @@ Settings overlay rendered inside the widget. Visible only in edit mode.
111
138
  interface Widget {
112
139
  id: number;
113
140
  label: string; // translation key
114
- content: Type<unknown>; // Angular component to render as content
115
- rows?: number; // 1-5; defaults to 3
116
- columns?: number; // 1-5; defaults to 3, 5 spans the full row
141
+ content: Type<DashboardWidgetContentComponent>;
142
+ rows?: number;
143
+ columns?: number;
117
144
  defaultRows?: number; // initial rows when widget is first added
118
145
  defaultColumns?: number; // initial columns when widget is first added
119
- color?: string; // optional explicit widget background override
120
- customLabel?: string; // user-defined label override (not a translation key)
121
146
  }
122
147
  ```
123
148
 
@@ -131,12 +156,13 @@ Add the following keys to your ngx-translate translation files:
131
156
  {
132
157
  "DASHBOARD": {
133
158
  "ADD_WIDGET": "Add widget",
134
- "NO_WIDGETS": "No widgets available"
159
+ "NO_WIDGETS": "No widgets available",
160
+ "LOCK": "Lock dashboard",
161
+ "UNLOCK": "Unlock dashboard"
135
162
  },
136
163
  "WIDGET": {
137
164
  "WIDTH": "Width",
138
165
  "HEIGHT": "Height",
139
- "RENAME": "Name",
140
166
  "DELETE": "Delete",
141
167
  "MOVE_LEFT": "Move left",
142
168
  "MOVE_RIGHT": "Move right",
@@ -152,6 +178,6 @@ Widget `label` fields are also treated as translation keys and resolved at runti
152
178
 
153
179
  ## 💾 Layout Persistence
154
180
 
155
- The dashboard automatically saves and restores widget state via `localStorage` using the `storageKey` input. Persisted state includes: position order, rows, columns, color, and custom label.
181
+ The dashboard automatically saves and restores widget state via `localStorage` using the `storageKey` input. Persisted state includes position order, rows, and columns.
156
182
 
157
183
  The format is backward compatible — dashboards saved with v1.4.x (IDs only) are migrated automatically on first load.
@@ -1,7 +1,7 @@
1
1
  import * as i0 from '@angular/core';
2
- import { signal, computed, Injectable, input, model, inject, Component, PLATFORM_ID, output, effect, ViewChild, NgModule } from '@angular/core';
2
+ import { signal, computed, Injectable, input, model, inject, Component, ViewChild, output, effect, NgModule } from '@angular/core';
3
3
  import * as i1$1 from '@angular/common';
4
- import { isPlatformBrowser, CommonModule } from '@angular/common';
4
+ import { CommonModule } from '@angular/common';
5
5
  import * as i2 from '@angular/material/button';
6
6
  import { MatButtonModule } from '@angular/material/button';
7
7
  import * as i3 from '@angular/material/icon';
@@ -10,8 +10,6 @@ import * as i4 from '@angular/material/tooltip';
10
10
  import { MatTooltipModule } from '@angular/material/tooltip';
11
11
  import * as i1 from '@angular/material/button-toggle';
12
12
  import { MatButtonToggleModule } from '@angular/material/button-toggle';
13
- import * as i5 from '@angular/material/divider';
14
- import { MatDividerModule } from '@angular/material/divider';
15
13
  import * as i6 from '@ngx-translate/core';
16
14
  import { TranslateModule } from '@ngx-translate/core';
17
15
  import { wrapGrid } from 'animate-css-grid';
@@ -20,23 +18,23 @@ import { MatMenuModule } from '@angular/material/menu';
20
18
 
21
19
  const DEFAULT_DASHBOARD_CONFIG = {
22
20
  columns: 'auto',
21
+ activeColumns: 1,
23
22
  minTileWidth: 200,
24
23
  rowHeight: 80,
25
24
  gap: 12,
26
- maxColumns: 5,
27
- maxRows: 5,
25
+ maxColumns: 4,
26
+ maxRows: 4,
28
27
  widgetPadding: 12,
29
28
  };
30
29
 
31
30
  class WidgetService {
32
- DEFAULT_WIDGET_SIZE = 3;
33
31
  widgets = signal([], ...(ngDevMode ? [{ debugName: "widgets" }] : /* istanbul ignore next */ []));
34
32
  addedWidgets = signal([], ...(ngDevMode ? [{ debugName: "addedWidgets" }] : /* istanbul ignore next */ []));
35
33
  config = signal(DEFAULT_DASHBOARD_CONFIG, ...(ngDevMode ? [{ debugName: "config" }] : /* istanbul ignore next */ []));
36
34
  editMode = signal(true, ...(ngDevMode ? [{ debugName: "editMode" }] : /* istanbul ignore next */ []));
37
35
  widgetsToAdd = computed(() => {
38
- const addedIds = new Set(this.addedWidgets().map((widget) => widget.id));
39
- return this.widgets().filter((w) => !addedIds.has(w.id));
36
+ const addedIds = this.addedWidgets().map((widget) => widget.id);
37
+ return this.widgets().filter((w) => !addedIds.includes(w.id));
40
38
  }, ...(ngDevMode ? [{ debugName: "widgetsToAdd" }] : /* istanbul ignore next */ []));
41
39
  setAvailableWidgets(widgets) {
42
40
  this.widgets.set(widgets);
@@ -45,14 +43,12 @@ class WidgetService {
45
43
  this.config.set({ ...DEFAULT_DASHBOARD_CONFIG, ...config });
46
44
  }
47
45
  addWidget(widget) {
48
- if (this.addedWidgets().some((addedWidget) => addedWidget.id === widget.id))
49
- return;
50
46
  this.addedWidgets.set([
51
47
  ...this.addedWidgets(),
52
48
  {
53
49
  ...widget,
54
- rows: this.clampScale(widget.defaultRows ?? widget.rows ?? this.DEFAULT_WIDGET_SIZE, this.config().maxRows),
55
- columns: this.clampScale(widget.defaultColumns ?? widget.columns ?? this.DEFAULT_WIDGET_SIZE, this.config().maxColumns),
50
+ rows: this.normalizeRows(widget.defaultRows ?? widget.rows),
51
+ columns: this.normalizeColumns(widget.defaultColumns ?? widget.columns),
56
52
  },
57
53
  ]);
58
54
  }
@@ -60,29 +56,16 @@ class WidgetService {
60
56
  const index = this.addedWidgets().findIndex((w) => w.id === id);
61
57
  if (index !== -1) {
62
58
  const newWidgets = [...this.addedWidgets()];
63
- newWidgets[index] = {
64
- ...newWidgets[index],
65
- ...widget,
66
- rows: widget.rows === undefined ? newWidgets[index].rows : this.clampScale(widget.rows, this.config().maxRows),
67
- columns: widget.columns === undefined
68
- ? newWidgets[index].columns
69
- : this.clampScale(widget.columns, this.config().maxColumns),
70
- };
59
+ newWidgets[index] = this.normalizeWidgetSize({ ...newWidgets[index], ...widget });
71
60
  this.addedWidgets.set(newWidgets);
72
61
  }
73
62
  }
74
- renameWidget(id, customLabel) {
75
- this.updateWidget(id, { customLabel: customLabel.trim() || undefined });
76
- }
77
- setWidgetColor(id, color) {
78
- this.updateWidget(id, { color });
79
- }
80
63
  toggleEditMode() {
81
64
  this.editMode.set(!this.editMode());
82
65
  }
83
66
  moveWidgetToLeft(id) {
84
67
  const index = this.addedWidgets().findIndex((w) => w.id === id);
85
- if (index <= 0)
68
+ if (index === 0)
86
69
  return;
87
70
  const newWidgets = [...this.addedWidgets()];
88
71
  [newWidgets[index], newWidgets[index - 1]] = [{ ...newWidgets[index - 1] }, { ...newWidgets[index] }];
@@ -90,7 +73,7 @@ class WidgetService {
90
73
  }
91
74
  moveWidgetToRight(id) {
92
75
  const index = this.addedWidgets().findIndex((w) => w.id === id);
93
- if (index === -1 || index === this.addedWidgets().length - 1)
76
+ if (index === this.addedWidgets().length - 1)
94
77
  return;
95
78
  const newWidgets = [...this.addedWidgets()];
96
79
  [newWidgets[index], newWidgets[index + 1]] = [{ ...newWidgets[index + 1] }, { ...newWidgets[index] }];
@@ -99,10 +82,21 @@ class WidgetService {
99
82
  deleteWidget(id) {
100
83
  this.addedWidgets.set(this.addedWidgets().filter((w) => w.id !== id));
101
84
  }
102
- clampScale(value, max) {
103
- const safeValue = Number.isFinite(value) ? Math.trunc(value) : this.DEFAULT_WIDGET_SIZE;
104
- const safeMax = Number.isFinite(max) ? Math.trunc(max) : this.DEFAULT_WIDGET_SIZE;
105
- return Math.max(1, Math.min(Math.max(1, safeMax), safeValue));
85
+ normalizeWidgetSize(widget) {
86
+ return {
87
+ ...widget,
88
+ rows: this.normalizeRows(widget.rows),
89
+ columns: this.normalizeColumns(widget.columns),
90
+ };
91
+ }
92
+ normalizeColumns(columns) {
93
+ return this.clamp(columns ?? 1, 1, this.config().maxColumns);
94
+ }
95
+ normalizeRows(rows) {
96
+ return this.clamp(rows ?? 1, 1, this.config().maxRows);
97
+ }
98
+ clamp(value, min, max) {
99
+ return Math.min(Math.max(Math.round(value), min), Math.max(min, max));
106
100
  }
107
101
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: WidgetService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
108
102
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: WidgetService });
@@ -112,82 +106,91 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.10", ngImpo
112
106
  }] });
113
107
 
114
108
  class WidgetOptionsComponent {
115
- DEFAULT_WIDGET_SIZE = 3;
116
- MAX_WIDGET_SCALE = 5;
117
109
  widget = input.required(...(ngDevMode ? [{ debugName: "widget" }] : /* istanbul ignore next */ []));
118
110
  showOptions = model(false, ...(ngDevMode ? [{ debugName: "showOptions" }] : /* istanbul ignore next */ []));
119
111
  store = inject(WidgetService);
120
- colOptions = computed(() => Array.from({ length: this.clampOptionCount(this.store.config().maxColumns) }, (_, i) => i + 1), ...(ngDevMode ? [{ debugName: "colOptions" }] : /* istanbul ignore next */ []));
121
- rowOptions = computed(() => Array.from({ length: this.clampOptionCount(this.store.config().maxRows) }, (_, i) => i + 1), ...(ngDevMode ? [{ debugName: "rowOptions" }] : /* istanbul ignore next */ []));
122
- selectedColumns = computed(() => this.clampOptionCount(this.widget().columns ?? this.DEFAULT_WIDGET_SIZE), ...(ngDevMode ? [{ debugName: "selectedColumns" }] : /* istanbul ignore next */ []));
123
- selectedRows = computed(() => this.clampOptionCount(this.widget().rows ?? this.DEFAULT_WIDGET_SIZE), ...(ngDevMode ? [{ debugName: "selectedRows" }] : /* istanbul ignore next */ []));
124
- canMoveLeft = computed(() => this.widgetIndex() > 0, ...(ngDevMode ? [{ debugName: "canMoveLeft" }] : /* istanbul ignore next */ []));
125
- canMoveRight = computed(() => {
126
- const index = this.widgetIndex();
127
- return index !== -1 && index < this.store.addedWidgets().length - 1;
128
- }, ...(ngDevMode ? [{ debugName: "canMoveRight" }] : /* istanbul ignore next */ []));
129
- onRenameBlur(event) {
130
- const input = event.target;
131
- this.store.renameWidget(this.widget().id, input.value);
112
+ colOptions = computed(() => Array.from({ length: this.store.config().maxColumns }, (_, i) => i + 1), ...(ngDevMode ? [{ debugName: "colOptions" }] : /* istanbul ignore next */ []));
113
+ rowOptions = computed(() => Array.from({ length: this.store.config().maxRows }, (_, i) => i + 1), ...(ngDevMode ? [{ debugName: "rowOptions" }] : /* istanbul ignore next */ []));
114
+ canMoveLeft() {
115
+ return this.store.addedWidgets().findIndex((w) => w.id === this.widget().id) > 0;
132
116
  }
133
- clampOptionCount(value) {
134
- return Math.max(1, Math.min(this.MAX_WIDGET_SCALE, Math.trunc(value)));
135
- }
136
- widgetIndex() {
137
- return this.store.addedWidgets().findIndex((widget) => widget.id === this.widget().id);
117
+ canMoveRight() {
118
+ const index = this.store.addedWidgets().findIndex((w) => w.id === this.widget().id);
119
+ return index >= 0 && index < this.store.addedWidgets().length - 1;
138
120
  }
139
121
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: WidgetOptionsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
140
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.10", type: WidgetOptionsComponent, isStandalone: false, selector: "app-widget-options", inputs: { widget: { classPropertyName: "widget", publicName: "widget", isSignal: true, isRequired: true, transformFunction: null }, showOptions: { classPropertyName: "showOptions", publicName: "showOptions", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { showOptions: "showOptionsChange" }, ngImport: i0, template: "<div class=\"options-top-bar\">\n <button\n mat-icon-button\n class=\"delete-btn\"\n (click)=\"store.deleteWidget(widget().id)\"\n [attr.aria-label]=\"'WIDGET.DELETE' | translate\"\n [matTooltip]=\"'WIDGET.DELETE' | translate\">\n <mat-icon>delete</mat-icon>\n </button>\n <span class=\"spacer\"></span>\n <button\n mat-icon-button\n (click)=\"showOptions.set(false)\"\n [attr.aria-label]=\"'WIDGET.CLOSE_OPTIONS' | translate\"\n [matTooltip]=\"'WIDGET.CLOSE_OPTIONS' | translate\">\n <mat-icon>close</mat-icon>\n </button>\n</div>\n\n<div class=\"options-body\">\n <div class=\"option-group\">\n <span class=\"option-label\">{{ 'WIDGET.RENAME' | translate }}</span>\n <input\n class=\"rename-input\"\n [value]=\"widget().customLabel || ''\"\n (blur)=\"onRenameBlur($event)\"\n (keydown.enter)=\"$any($event.target).blur()\"\n (keydown.escape)=\"showOptions.set(false)\"\n [attr.aria-label]=\"'WIDGET.RENAME' | translate\"\n [placeholder]=\"widget().label | translate\"\n />\n </div>\n\n <mat-divider/>\n\n <div class=\"option-group\">\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.WIDTH' | translate }}</span>\n <mat-button-toggle-group\n class=\"size-toggle-group\"\n (change)=\"store.updateWidget(widget().id, { columns: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"selectedColumns()\">\n @for (n of colOptions(); track n) {\n <mat-button-toggle class=\"size-toggle\" [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.HEIGHT' | translate }}</span>\n <mat-button-toggle-group\n class=\"size-toggle-group\"\n (change)=\"store.updateWidget(widget().id, { rows: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"selectedRows()\">\n @for (n of rowOptions(); track n) {\n <mat-button-toggle class=\"size-toggle\" [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n </div>\n\n</div>\n\n<button\n class=\"move-left-btn\"\n mat-icon-button\n (click)=\"store.moveWidgetToLeft(widget().id)\"\n [disabled]=\"!canMoveLeft()\"\n [attr.aria-label]=\"'WIDGET.MOVE_LEFT' | translate\"\n [matTooltip]=\"'WIDGET.MOVE_LEFT' | translate\">\n <mat-icon>chevron_left</mat-icon>\n</button>\n<button\n class=\"move-right-btn\"\n mat-icon-button\n (click)=\"store.moveWidgetToRight(widget().id)\"\n [disabled]=\"!canMoveRight()\"\n [attr.aria-label]=\"'WIDGET.MOVE_RIGHT' | translate\"\n [matTooltip]=\"'WIDGET.MOVE_RIGHT' | translate\">\n <mat-icon>chevron_right</mat-icon>\n</button>\n", styles: [":host{position:absolute;z-index:2;top:0;left:0;border-radius:inherit;width:100%;height:100%;display:flex;flex-direction:column;box-sizing:border-box;overflow:hidden;color:inherit;background-color:inherit;font-family:inherit;opacity:1;box-shadow:inset 0 0 0 1px color-mix(in srgb,currentColor 10%,transparent)}.options-top-bar{display:flex;align-items:center;padding:4px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent)}.options-top-bar button{color:inherit;--mdc-icon-button-icon-color: currentColor;--mat-icon-button-state-layer-color: currentColor}.options-top-bar .spacer{flex:1}.options-top-bar .delete-btn{color:inherit;opacity:.85}.options-body{flex:1;overflow:auto;display:flex;flex-direction:column;gap:16px;padding:12px;min-height:0;scrollbar-gutter:stable}.option-group{display:flex;flex-direction:column;gap:6px}.option-label{font-size:.7rem;font-weight:600;opacity:.6;text-transform:uppercase;letter-spacing:.06em}.rename-input{border:1px solid color-mix(in srgb,currentColor 20%,transparent);border-radius:6px;background:transparent;color:inherit;font-size:.875rem;font-family:inherit;padding:6px 8px;width:100%;box-sizing:border-box;outline:none;transition:border-color .15s}.rename-input:focus{border-color:currentColor}.rename-input:focus-visible{outline:2px solid currentColor;outline-offset:2px}.size-row{display:flex;align-items:flex-start;gap:10px;min-width:0}.size-row .size-label{min-width:52px;padding-top:8px}.size-row .size-toggle-group{display:grid;grid-template-columns:repeat(auto-fit,minmax(34px,1fr));flex:1;min-width:0;border-radius:6px;overflow:hidden}.size-row .size-toggle-group .size-toggle{min-width:0}.size-row .size-toggle-group ::ng-deep .mat-button-toggle{min-width:0;color:inherit;background-color:inherit;--mat-standard-button-toggle-text-color: currentColor;--mat-standard-button-toggle-selected-state-text-color: currentColor;--mat-standard-button-toggle-selected-state-background-color: color-mix(in srgb, currentColor 12%, transparent)}.size-row .size-toggle-group ::ng-deep .mat-button-toggle-button{width:100%;height:32px;line-height:32px;padding:0;font-size:.8125rem}.move-left-btn,.move-right-btn{color:inherit;background-color:inherit;--mdc-icon-button-icon-color: currentColor;--mat-icon-button-state-layer-color: currentColor}.move-left-btn:disabled,.move-right-btn:disabled{opacity:.35}.move-left-btn{position:absolute;top:50%;transform:translateY(-50%);left:-16px}.move-right-btn{position:absolute;top:50%;transform:translateY(-50%);right:-16px}@media(max-width:599px){:host{inset:0;width:100%;height:100%;max-height:none;border-radius:8px}.size-row{flex-direction:column;gap:6px}.size-row .size-label{min-width:0;padding-top:0}.move-left-btn,.move-right-btn{display:none}}\n"], dependencies: [{ kind: "directive", type: i1.MatButtonToggleGroup, selector: "mat-button-toggle-group", inputs: ["appearance", "name", "vertical", "value", "multiple", "disabled", "disabledInteractive", "hideSingleSelectionIndicator", "hideMultipleSelectionIndicator"], outputs: ["valueChange", "change"], exportAs: ["matButtonToggleGroup"] }, { kind: "component", type: i1.MatButtonToggle, selector: "mat-button-toggle", inputs: ["aria-label", "aria-labelledby", "id", "name", "value", "tabIndex", "disableRipple", "appearance", "checked", "disabled", "disabledInteractive"], outputs: ["change"], exportAs: ["matButtonToggle"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button], a[mat-icon-button], button[matIconButton], a[matIconButton]", exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: i5.MatDivider, selector: "mat-divider", inputs: ["vertical", "inset"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] });
122
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.10", type: WidgetOptionsComponent, isStandalone: false, selector: "app-widget-options", inputs: { widget: { classPropertyName: "widget", publicName: "widget", isSignal: true, isRequired: true, transformFunction: null }, showOptions: { classPropertyName: "showOptions", publicName: "showOptions", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { showOptions: "showOptionsChange" }, ngImport: i0, template: "<div class=\"options-top-bar\">\n <button\n mat-icon-button\n class=\"delete-btn\"\n (click)=\"store.deleteWidget(widget().id)\"\n [matTooltip]=\"'WIDGET.DELETE' | translate\">\n <mat-icon>delete</mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"store.moveWidgetToLeft(widget().id)\"\n [disabled]=\"!canMoveLeft()\"\n [matTooltip]=\"'WIDGET.MOVE_LEFT' | translate\">\n <mat-icon>chevron_left</mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"store.moveWidgetToRight(widget().id)\"\n [disabled]=\"!canMoveRight()\"\n [matTooltip]=\"'WIDGET.MOVE_RIGHT' | translate\">\n <mat-icon>chevron_right</mat-icon>\n </button>\n <span class=\"spacer\"></span>\n <button\n mat-icon-button\n (click)=\"showOptions.set(false)\"\n [matTooltip]=\"'WIDGET.CLOSE_OPTIONS' | translate\">\n <mat-icon>close</mat-icon>\n </button>\n</div>\n\n<div class=\"options-body\">\n <div class=\"option-group\">\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.WIDTH' | translate }}</span>\n <mat-button-toggle-group\n (change)=\"store.updateWidget(widget().id, { columns: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"widget().columns ?? 1\">\n @for (n of colOptions(); track n) {\n <mat-button-toggle [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.HEIGHT' | translate }}</span>\n <mat-button-toggle-group\n (change)=\"store.updateWidget(widget().id, { rows: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"widget().rows ?? 1\">\n @for (n of rowOptions(); track n) {\n <mat-button-toggle [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n </div>\n\n</div>\n", styles: [":host{position:absolute;z-index:2;top:0;left:0;border-radius:inherit;width:100%;height:100%;display:flex;flex-direction:column;box-sizing:border-box;overflow:hidden;background:inherit;background-color:inherit;color:inherit;font-family:inherit;min-width:0;min-height:0}.options-top-bar{display:flex;align-items:center;padding:4px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent)}.options-top-bar .spacer{flex:1}.options-top-bar .delete-btn{color:var(--mat-sys-error, #ef4444)}.options-body{flex:1;overflow:auto;display:flex;flex-direction:column;gap:20px;padding:12px}.option-group{display:flex;flex-direction:column;gap:6px}.option-label{font-size:.7rem;font-weight:600;opacity:.6;text-transform:uppercase;letter-spacing:.06em}.size-row{display:flex;align-items:center;gap:8px}.size-row .size-label{min-width:52px}.size-row mat-button-toggle-group{height:28px}.size-row mat-button-toggle-group ::ng-deep .mat-button-toggle-button{height:28px;line-height:28px;padding:0 10px;font-size:.8rem}\n"], dependencies: [{ kind: "directive", type: i1.MatButtonToggleGroup, selector: "mat-button-toggle-group", inputs: ["appearance", "name", "vertical", "value", "multiple", "disabled", "disabledInteractive", "hideSingleSelectionIndicator", "hideMultipleSelectionIndicator"], outputs: ["valueChange", "change"], exportAs: ["matButtonToggleGroup"] }, { kind: "component", type: i1.MatButtonToggle, selector: "mat-button-toggle", inputs: ["aria-label", "aria-labelledby", "id", "name", "value", "tabIndex", "disableRipple", "appearance", "checked", "disabled", "disabledInteractive"], outputs: ["change"], exportAs: ["matButtonToggle"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button], a[mat-icon-button], button[matIconButton], a[matIconButton]", exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] });
141
123
  }
142
124
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: WidgetOptionsComponent, decorators: [{
143
125
  type: Component,
144
- args: [{ selector: 'app-widget-options', standalone: false, template: "<div class=\"options-top-bar\">\n <button\n mat-icon-button\n class=\"delete-btn\"\n (click)=\"store.deleteWidget(widget().id)\"\n [attr.aria-label]=\"'WIDGET.DELETE' | translate\"\n [matTooltip]=\"'WIDGET.DELETE' | translate\">\n <mat-icon>delete</mat-icon>\n </button>\n <span class=\"spacer\"></span>\n <button\n mat-icon-button\n (click)=\"showOptions.set(false)\"\n [attr.aria-label]=\"'WIDGET.CLOSE_OPTIONS' | translate\"\n [matTooltip]=\"'WIDGET.CLOSE_OPTIONS' | translate\">\n <mat-icon>close</mat-icon>\n </button>\n</div>\n\n<div class=\"options-body\">\n <div class=\"option-group\">\n <span class=\"option-label\">{{ 'WIDGET.RENAME' | translate }}</span>\n <input\n class=\"rename-input\"\n [value]=\"widget().customLabel || ''\"\n (blur)=\"onRenameBlur($event)\"\n (keydown.enter)=\"$any($event.target).blur()\"\n (keydown.escape)=\"showOptions.set(false)\"\n [attr.aria-label]=\"'WIDGET.RENAME' | translate\"\n [placeholder]=\"widget().label | translate\"\n />\n </div>\n\n <mat-divider/>\n\n <div class=\"option-group\">\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.WIDTH' | translate }}</span>\n <mat-button-toggle-group\n class=\"size-toggle-group\"\n (change)=\"store.updateWidget(widget().id, { columns: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"selectedColumns()\">\n @for (n of colOptions(); track n) {\n <mat-button-toggle class=\"size-toggle\" [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.HEIGHT' | translate }}</span>\n <mat-button-toggle-group\n class=\"size-toggle-group\"\n (change)=\"store.updateWidget(widget().id, { rows: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"selectedRows()\">\n @for (n of rowOptions(); track n) {\n <mat-button-toggle class=\"size-toggle\" [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n </div>\n\n</div>\n\n<button\n class=\"move-left-btn\"\n mat-icon-button\n (click)=\"store.moveWidgetToLeft(widget().id)\"\n [disabled]=\"!canMoveLeft()\"\n [attr.aria-label]=\"'WIDGET.MOVE_LEFT' | translate\"\n [matTooltip]=\"'WIDGET.MOVE_LEFT' | translate\">\n <mat-icon>chevron_left</mat-icon>\n</button>\n<button\n class=\"move-right-btn\"\n mat-icon-button\n (click)=\"store.moveWidgetToRight(widget().id)\"\n [disabled]=\"!canMoveRight()\"\n [attr.aria-label]=\"'WIDGET.MOVE_RIGHT' | translate\"\n [matTooltip]=\"'WIDGET.MOVE_RIGHT' | translate\">\n <mat-icon>chevron_right</mat-icon>\n</button>\n", styles: [":host{position:absolute;z-index:2;top:0;left:0;border-radius:inherit;width:100%;height:100%;display:flex;flex-direction:column;box-sizing:border-box;overflow:hidden;color:inherit;background-color:inherit;font-family:inherit;opacity:1;box-shadow:inset 0 0 0 1px color-mix(in srgb,currentColor 10%,transparent)}.options-top-bar{display:flex;align-items:center;padding:4px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent)}.options-top-bar button{color:inherit;--mdc-icon-button-icon-color: currentColor;--mat-icon-button-state-layer-color: currentColor}.options-top-bar .spacer{flex:1}.options-top-bar .delete-btn{color:inherit;opacity:.85}.options-body{flex:1;overflow:auto;display:flex;flex-direction:column;gap:16px;padding:12px;min-height:0;scrollbar-gutter:stable}.option-group{display:flex;flex-direction:column;gap:6px}.option-label{font-size:.7rem;font-weight:600;opacity:.6;text-transform:uppercase;letter-spacing:.06em}.rename-input{border:1px solid color-mix(in srgb,currentColor 20%,transparent);border-radius:6px;background:transparent;color:inherit;font-size:.875rem;font-family:inherit;padding:6px 8px;width:100%;box-sizing:border-box;outline:none;transition:border-color .15s}.rename-input:focus{border-color:currentColor}.rename-input:focus-visible{outline:2px solid currentColor;outline-offset:2px}.size-row{display:flex;align-items:flex-start;gap:10px;min-width:0}.size-row .size-label{min-width:52px;padding-top:8px}.size-row .size-toggle-group{display:grid;grid-template-columns:repeat(auto-fit,minmax(34px,1fr));flex:1;min-width:0;border-radius:6px;overflow:hidden}.size-row .size-toggle-group .size-toggle{min-width:0}.size-row .size-toggle-group ::ng-deep .mat-button-toggle{min-width:0;color:inherit;background-color:inherit;--mat-standard-button-toggle-text-color: currentColor;--mat-standard-button-toggle-selected-state-text-color: currentColor;--mat-standard-button-toggle-selected-state-background-color: color-mix(in srgb, currentColor 12%, transparent)}.size-row .size-toggle-group ::ng-deep .mat-button-toggle-button{width:100%;height:32px;line-height:32px;padding:0;font-size:.8125rem}.move-left-btn,.move-right-btn{color:inherit;background-color:inherit;--mdc-icon-button-icon-color: currentColor;--mat-icon-button-state-layer-color: currentColor}.move-left-btn:disabled,.move-right-btn:disabled{opacity:.35}.move-left-btn{position:absolute;top:50%;transform:translateY(-50%);left:-16px}.move-right-btn{position:absolute;top:50%;transform:translateY(-50%);right:-16px}@media(max-width:599px){:host{inset:0;width:100%;height:100%;max-height:none;border-radius:8px}.size-row{flex-direction:column;gap:6px}.size-row .size-label{min-width:0;padding-top:0}.move-left-btn,.move-right-btn{display:none}}\n"] }]
126
+ args: [{ selector: 'app-widget-options', standalone: false, template: "<div class=\"options-top-bar\">\n <button\n mat-icon-button\n class=\"delete-btn\"\n (click)=\"store.deleteWidget(widget().id)\"\n [matTooltip]=\"'WIDGET.DELETE' | translate\">\n <mat-icon>delete</mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"store.moveWidgetToLeft(widget().id)\"\n [disabled]=\"!canMoveLeft()\"\n [matTooltip]=\"'WIDGET.MOVE_LEFT' | translate\">\n <mat-icon>chevron_left</mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"store.moveWidgetToRight(widget().id)\"\n [disabled]=\"!canMoveRight()\"\n [matTooltip]=\"'WIDGET.MOVE_RIGHT' | translate\">\n <mat-icon>chevron_right</mat-icon>\n </button>\n <span class=\"spacer\"></span>\n <button\n mat-icon-button\n (click)=\"showOptions.set(false)\"\n [matTooltip]=\"'WIDGET.CLOSE_OPTIONS' | translate\">\n <mat-icon>close</mat-icon>\n </button>\n</div>\n\n<div class=\"options-body\">\n <div class=\"option-group\">\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.WIDTH' | translate }}</span>\n <mat-button-toggle-group\n (change)=\"store.updateWidget(widget().id, { columns: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"widget().columns ?? 1\">\n @for (n of colOptions(); track n) {\n <mat-button-toggle [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.HEIGHT' | translate }}</span>\n <mat-button-toggle-group\n (change)=\"store.updateWidget(widget().id, { rows: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"widget().rows ?? 1\">\n @for (n of rowOptions(); track n) {\n <mat-button-toggle [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n </div>\n\n</div>\n", styles: [":host{position:absolute;z-index:2;top:0;left:0;border-radius:inherit;width:100%;height:100%;display:flex;flex-direction:column;box-sizing:border-box;overflow:hidden;background:inherit;background-color:inherit;color:inherit;font-family:inherit;min-width:0;min-height:0}.options-top-bar{display:flex;align-items:center;padding:4px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent)}.options-top-bar .spacer{flex:1}.options-top-bar .delete-btn{color:var(--mat-sys-error, #ef4444)}.options-body{flex:1;overflow:auto;display:flex;flex-direction:column;gap:20px;padding:12px}.option-group{display:flex;flex-direction:column;gap:6px}.option-label{font-size:.7rem;font-weight:600;opacity:.6;text-transform:uppercase;letter-spacing:.06em}.size-row{display:flex;align-items:center;gap:8px}.size-row .size-label{min-width:52px}.size-row mat-button-toggle-group{height:28px}.size-row mat-button-toggle-group ::ng-deep .mat-button-toggle-button{height:28px;line-height:28px;padding:0 10px;font-size:.8rem}\n"] }]
145
127
  }], propDecorators: { widget: [{ type: i0.Input, args: [{ isSignal: true, alias: "widget", required: true }] }], showOptions: [{ type: i0.Input, args: [{ isSignal: true, alias: "showOptions", required: false }] }, { type: i0.Output, args: ["showOptionsChange"] }] } });
146
128
 
147
129
  class WidgetComponent {
148
- DEFAULT_WIDGET_SIZE = 3;
149
130
  data = input.required(...(ngDevMode ? [{ debugName: "data" }] : /* istanbul ignore next */ []));
150
131
  store = inject(WidgetService);
132
+ contentHost;
151
133
  showWidgetOptions = signal(false, ...(ngDevMode ? [{ debugName: "showWidgetOptions" }] : /* istanbul ignore next */ []));
152
- isRenaming = signal(false, ...(ngDevMode ? [{ debugName: "isRenaming" }] : /* istanbul ignore next */ []));
134
+ contentSize = signal({
135
+ width: 0,
136
+ height: 0,
137
+ rows: 1,
138
+ columns: 1,
139
+ }, ...(ngDevMode ? [{ debugName: "contentSize" }] : /* istanbul ignore next */ []));
140
+ resizeObserver;
153
141
  gridArea = computed(() => {
154
142
  const widget = this.data();
155
- const rows = this.clampScale(widget.rows ?? this.DEFAULT_WIDGET_SIZE, this.store.config().maxRows);
156
- const columns = this.clampScale(widget.columns ?? this.DEFAULT_WIDGET_SIZE, this.store.config().maxColumns);
143
+ const rows = this.normalizeSpan(widget.rows, this.store.config().maxRows);
144
+ const columns = this.normalizeSpan(widget.columns, Math.min(this.store.config().maxColumns, this.store.config().activeColumns));
157
145
  return `span ${rows} / span ${columns}`;
158
146
  }, ...(ngDevMode ? [{ debugName: "gridArea" }] : /* istanbul ignore next */ []));
159
147
  widgetPadding = computed(() => `${this.store.config().widgetPadding}px`, ...(ngDevMode ? [{ debugName: "widgetPadding" }] : /* istanbul ignore next */ []));
160
- startRename() {
161
- if (!this.store.editMode())
148
+ contentInputs = computed(() => ({
149
+ widgetSize: this.contentSize(),
150
+ }), ...(ngDevMode ? [{ debugName: "contentInputs" }] : /* istanbul ignore next */ []));
151
+ ngAfterViewInit() {
152
+ this.updateContentSize();
153
+ const element = this.contentHost?.nativeElement;
154
+ if (!element)
162
155
  return;
163
- this.isRenaming.set(true);
156
+ this.resizeObserver = new ResizeObserver(() => this.updateContentSize());
157
+ this.resizeObserver.observe(element);
164
158
  }
165
- finishRename(event) {
166
- const input = event.target;
167
- this.store.renameWidget(this.data().id, input.value);
168
- this.isRenaming.set(false);
159
+ ngOnDestroy() {
160
+ this.resizeObserver?.disconnect();
169
161
  }
170
- clampScale(value, max) {
171
- return Math.max(1, Math.min(max, Math.trunc(value)));
162
+ updateContentSize() {
163
+ const element = this.contentHost?.nativeElement;
164
+ if (!element)
165
+ return;
166
+ const rect = element.getBoundingClientRect();
167
+ this.contentSize.set({
168
+ width: Math.round(rect.width),
169
+ height: Math.round(rect.height),
170
+ rows: this.normalizeSpan(this.data().rows, this.store.config().maxRows),
171
+ columns: this.normalizeSpan(this.data().columns, Math.min(this.store.config().maxColumns, this.store.config().activeColumns)),
172
+ });
173
+ }
174
+ normalizeSpan(value, max) {
175
+ return Math.min(Math.max(Math.round(value ?? 1), 1), Math.max(max, 1));
172
176
  }
173
177
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: WidgetComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
174
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.10", type: WidgetComponent, isStandalone: false, selector: "app-widget", inputs: { data: { classPropertyName: "data", publicName: "data", isSignal: true, isRequired: true, transformFunction: null } }, host: { properties: { "style.grid-area": "gridArea()", "style.--widget-padding": "widgetPadding()" } }, ngImport: i0, template: "<div class=\"widget-container mat-elevation-z3\" [style.--widget-color]=\"data().color\">\n @if (!showWidgetOptions()) {\n <div class=\"widget-header\">\n @if (isRenaming()) {\n <input\n class=\"rename-input\"\n [value]=\"data().customLabel || ''\"\n (blur)=\"finishRename($event)\"\n (keydown.enter)=\"finishRename($event)\"\n (keydown.escape)=\"isRenaming.set(false)\"\n [attr.aria-label]=\"'WIDGET.RENAME' | translate\"\n [placeholder]=\"data().label | translate\"\n autofocus\n />\n } @else {\n <h3 class=\"widget-title\" (dblclick)=\"startRename()\">\n @if (data().customLabel) {\n {{ data().customLabel }}\n } @else {\n {{ data().label | translate }}\n }\n </h3>\n }\n <div class=\"widget-actions\">\n @if (store.editMode()) {\n <button\n mat-icon-button\n (click)=\"showWidgetOptions.set(true)\"\n [attr.aria-label]=\"'WIDGET.SETTINGS' | translate\"\n [matTooltip]=\"'WIDGET.SETTINGS' | translate\">\n <mat-icon>settings</mat-icon>\n </button>\n }\n </div>\n </div>\n\n <div class=\"widget-content\">\n <ng-container [ngComponentOutlet]=\"data().content\"/>\n </div>\n }\n\n @if (showWidgetOptions()) {\n <app-widget-options [(showOptions)]=\"showWidgetOptions\" [widget]=\"data()\"/>\n }\n</div>\n", styles: [":host{display:block;border-radius:8px;font-family:inherit;color:inherit;background-color:inherit;align-self:stretch;justify-self:stretch;width:100%;min-width:0;min-height:0;contain:layout style}.widget-container{position:relative;min-height:100%;width:100%;box-sizing:border-box;border-radius:inherit;overflow:visible;color:inherit;background-color:var(--widget-color, inherit);display:flex;flex-direction:column;box-shadow:0 1px 2px color-mix(in srgb,currentColor 12%,transparent);transition:background-color .2s,box-shadow .2s;min-width:0;min-height:0}.widget-header{display:flex;align-items:center;padding:4px 4px 4px 12px;gap:6px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent);min-height:44px}.widget-title{flex:1;margin:0;font-size:1rem;font-weight:600;line-height:1.3;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:flex;align-items:center;gap:4px;cursor:text;-webkit-user-select:none;user-select:none}.rename-input{flex:1;min-width:0;border:none;border-bottom:1px solid currentColor;outline:none;background:transparent;color:inherit;font-size:.875rem;font-weight:500;font-family:inherit;padding:2px 0}.rename-input:focus-visible{outline:2px solid currentColor;outline-offset:2px}.widget-actions{display:flex;align-items:center;flex-shrink:0}.widget-actions button{width:32px;height:32px;line-height:32px;color:inherit;--mdc-icon-button-icon-color: currentColor;--mat-icon-button-state-layer-color: currentColor}.widget-actions button mat-icon{font-size:22px;height:22px;width:22px;line-height:22px}.widget-content{flex:1;min-height:0;min-width:0;overflow:visible;padding:var(--widget-padding, 12px);box-sizing:border-box;width:100%}:host ::ng-deep .widget-content>*{display:block;width:100%;max-width:100%;min-width:0;box-sizing:border-box}@media(max-width:599px){:host{grid-column:1/-1!important}.widget-header{padding-left:10px}}\n"], dependencies: [{ kind: "directive", type: i1$1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletEnvironmentInjector", "ngComponentOutletContent", "ngComponentOutletNgModule"], exportAs: ["ngComponentOutlet"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button], a[mat-icon-button], button[matIconButton], a[matIconButton]", exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: WidgetOptionsComponent, selector: "app-widget-options", inputs: ["widget", "showOptions"], outputs: ["showOptionsChange"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] });
178
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.10", type: WidgetComponent, isStandalone: false, selector: "app-widget", inputs: { data: { classPropertyName: "data", publicName: "data", isSignal: true, isRequired: true, transformFunction: null } }, host: { properties: { "style.grid-area": "gridArea()", "style.--widget-padding": "widgetPadding()" } }, viewQueries: [{ propertyName: "contentHost", first: true, predicate: ["contentHost"], descendants: true }], ngImport: i0, template: "<div class=\"widget-container mat-elevation-z3\">\n <div class=\"widget-header\">\n <h3 class=\"widget-title\">{{ data().label | translate }}</h3>\n <div class=\"widget-actions\">\n @if (store.editMode()) {\n <button\n mat-icon-button\n (click)=\"showWidgetOptions.set(true)\"\n [matTooltip]=\"'WIDGET.SETTINGS' | translate\">\n <mat-icon>settings</mat-icon>\n </button>\n }\n </div>\n </div>\n\n <div #contentHost class=\"widget-content\">\n <ng-container [ngComponentOutlet]=\"data().content\" [ngComponentOutletInputs]=\"contentInputs()\"/>\n </div>\n\n @if (showWidgetOptions()) {\n <app-widget-options [(showOptions)]=\"showWidgetOptions\" [widget]=\"data()\"/>\n }\n</div>\n", styles: [":host{display:block;border-radius:12px;font-family:inherit;color:inherit;background:inherit;background-color:inherit;align-self:stretch;min-width:0;min-height:0;height:100%}.widget-container{position:relative;height:100%;width:100%;min-width:0;min-height:0;box-sizing:border-box;border-radius:inherit;overflow:hidden;background:inherit;background-color:inherit;color:inherit;display:flex;flex-direction:column}.widget-header{display:flex;align-items:center;padding:4px 4px 4px 12px;gap:2px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent);min-height:44px}.widget-title{flex:1;margin:0;font-size:1rem;font-weight:600;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:flex;align-items:center;gap:4px;-webkit-user-select:none;user-select:none}.widget-actions{display:flex;align-items:center;flex-shrink:0}.widget-actions button{width:32px;height:32px;line-height:32px}.widget-actions button mat-icon{font-size:22px;height:22px;width:22px;line-height:22px}.widget-content{flex:1;overflow:auto;padding:var(--widget-padding, 12px);min-width:0;min-height:0;box-sizing:border-box}.widget-content ::ng-deep>*{display:block;width:100%;height:100%;min-width:0;min-height:0;box-sizing:border-box;color:inherit;background:inherit;background-color:inherit}\n"], dependencies: [{ kind: "directive", type: i1$1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletEnvironmentInjector", "ngComponentOutletContent", "ngComponentOutletNgModule"], exportAs: ["ngComponentOutlet"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button], a[mat-icon-button], button[matIconButton], a[matIconButton]", exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: WidgetOptionsComponent, selector: "app-widget-options", inputs: ["widget", "showOptions"], outputs: ["showOptionsChange"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] });
175
179
  }
176
180
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: WidgetComponent, decorators: [{
177
181
  type: Component,
178
182
  args: [{ selector: 'app-widget', standalone: false, host: {
179
183
  '[style.grid-area]': 'gridArea()',
180
184
  '[style.--widget-padding]': 'widgetPadding()',
181
- }, template: "<div class=\"widget-container mat-elevation-z3\" [style.--widget-color]=\"data().color\">\n @if (!showWidgetOptions()) {\n <div class=\"widget-header\">\n @if (isRenaming()) {\n <input\n class=\"rename-input\"\n [value]=\"data().customLabel || ''\"\n (blur)=\"finishRename($event)\"\n (keydown.enter)=\"finishRename($event)\"\n (keydown.escape)=\"isRenaming.set(false)\"\n [attr.aria-label]=\"'WIDGET.RENAME' | translate\"\n [placeholder]=\"data().label | translate\"\n autofocus\n />\n } @else {\n <h3 class=\"widget-title\" (dblclick)=\"startRename()\">\n @if (data().customLabel) {\n {{ data().customLabel }}\n } @else {\n {{ data().label | translate }}\n }\n </h3>\n }\n <div class=\"widget-actions\">\n @if (store.editMode()) {\n <button\n mat-icon-button\n (click)=\"showWidgetOptions.set(true)\"\n [attr.aria-label]=\"'WIDGET.SETTINGS' | translate\"\n [matTooltip]=\"'WIDGET.SETTINGS' | translate\">\n <mat-icon>settings</mat-icon>\n </button>\n }\n </div>\n </div>\n\n <div class=\"widget-content\">\n <ng-container [ngComponentOutlet]=\"data().content\"/>\n </div>\n }\n\n @if (showWidgetOptions()) {\n <app-widget-options [(showOptions)]=\"showWidgetOptions\" [widget]=\"data()\"/>\n }\n</div>\n", styles: [":host{display:block;border-radius:8px;font-family:inherit;color:inherit;background-color:inherit;align-self:stretch;justify-self:stretch;width:100%;min-width:0;min-height:0;contain:layout style}.widget-container{position:relative;min-height:100%;width:100%;box-sizing:border-box;border-radius:inherit;overflow:visible;color:inherit;background-color:var(--widget-color, inherit);display:flex;flex-direction:column;box-shadow:0 1px 2px color-mix(in srgb,currentColor 12%,transparent);transition:background-color .2s,box-shadow .2s;min-width:0;min-height:0}.widget-header{display:flex;align-items:center;padding:4px 4px 4px 12px;gap:6px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent);min-height:44px}.widget-title{flex:1;margin:0;font-size:1rem;font-weight:600;line-height:1.3;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:flex;align-items:center;gap:4px;cursor:text;-webkit-user-select:none;user-select:none}.rename-input{flex:1;min-width:0;border:none;border-bottom:1px solid currentColor;outline:none;background:transparent;color:inherit;font-size:.875rem;font-weight:500;font-family:inherit;padding:2px 0}.rename-input:focus-visible{outline:2px solid currentColor;outline-offset:2px}.widget-actions{display:flex;align-items:center;flex-shrink:0}.widget-actions button{width:32px;height:32px;line-height:32px;color:inherit;--mdc-icon-button-icon-color: currentColor;--mat-icon-button-state-layer-color: currentColor}.widget-actions button mat-icon{font-size:22px;height:22px;width:22px;line-height:22px}.widget-content{flex:1;min-height:0;min-width:0;overflow:visible;padding:var(--widget-padding, 12px);box-sizing:border-box;width:100%}:host ::ng-deep .widget-content>*{display:block;width:100%;max-width:100%;min-width:0;box-sizing:border-box}@media(max-width:599px){:host{grid-column:1/-1!important}.widget-header{padding-left:10px}}\n"] }]
182
- }], propDecorators: { data: [{ type: i0.Input, args: [{ isSignal: true, alias: "data", required: true }] }] } });
185
+ }, template: "<div class=\"widget-container mat-elevation-z3\">\n <div class=\"widget-header\">\n <h3 class=\"widget-title\">{{ data().label | translate }}</h3>\n <div class=\"widget-actions\">\n @if (store.editMode()) {\n <button\n mat-icon-button\n (click)=\"showWidgetOptions.set(true)\"\n [matTooltip]=\"'WIDGET.SETTINGS' | translate\">\n <mat-icon>settings</mat-icon>\n </button>\n }\n </div>\n </div>\n\n <div #contentHost class=\"widget-content\">\n <ng-container [ngComponentOutlet]=\"data().content\" [ngComponentOutletInputs]=\"contentInputs()\"/>\n </div>\n\n @if (showWidgetOptions()) {\n <app-widget-options [(showOptions)]=\"showWidgetOptions\" [widget]=\"data()\"/>\n }\n</div>\n", styles: [":host{display:block;border-radius:12px;font-family:inherit;color:inherit;background:inherit;background-color:inherit;align-self:stretch;min-width:0;min-height:0;height:100%}.widget-container{position:relative;height:100%;width:100%;min-width:0;min-height:0;box-sizing:border-box;border-radius:inherit;overflow:hidden;background:inherit;background-color:inherit;color:inherit;display:flex;flex-direction:column}.widget-header{display:flex;align-items:center;padding:4px 4px 4px 12px;gap:2px;flex-shrink:0;border-bottom:1px solid color-mix(in srgb,currentColor 10%,transparent);min-height:44px}.widget-title{flex:1;margin:0;font-size:1rem;font-weight:600;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;display:flex;align-items:center;gap:4px;-webkit-user-select:none;user-select:none}.widget-actions{display:flex;align-items:center;flex-shrink:0}.widget-actions button{width:32px;height:32px;line-height:32px}.widget-actions button mat-icon{font-size:22px;height:22px;width:22px;line-height:22px}.widget-content{flex:1;overflow:auto;padding:var(--widget-padding, 12px);min-width:0;min-height:0;box-sizing:border-box}.widget-content ::ng-deep>*{display:block;width:100%;height:100%;min-width:0;min-height:0;box-sizing:border-box;color:inherit;background:inherit;background-color:inherit}\n"] }]
186
+ }], propDecorators: { data: [{ type: i0.Input, args: [{ isSignal: true, alias: "data", required: true }] }], contentHost: [{
187
+ type: ViewChild,
188
+ args: ['contentHost']
189
+ }] } });
183
190
 
184
191
  class DashboardWidgetsComponent {
185
192
  DEFAULT_STORAGE_KEY = 'dashboard-widgets';
186
193
  ERROR_PARSING_MESSAGE = 'Error parsing widgets from localStorage';
187
- MAX_WIDGET_SCALE = 5;
188
- platformId = inject(PLATFORM_ID);
189
- isBrowser = isPlatformBrowser(this.platformId);
190
- hasLoadedStoredWidgets = signal(false, ...(ngDevMode ? [{ debugName: "hasLoadedStoredWidgets" }] : /* istanbul ignore next */ []));
191
194
  title = input.required(...(ngDevMode ? [{ debugName: "title" }] : /* istanbul ignore next */ []));
192
195
  widgets = input.required(...(ngDevMode ? [{ debugName: "widgets" }] : /* istanbul ignore next */ []));
193
196
  storageKey = input(this.DEFAULT_STORAGE_KEY, ...(ngDevMode ? [{ debugName: "storageKey" }] : /* istanbul ignore next */ []));
@@ -195,48 +198,55 @@ class DashboardWidgetsComponent {
195
198
  minTileWidth = input(200, ...(ngDevMode ? [{ debugName: "minTileWidth" }] : /* istanbul ignore next */ []));
196
199
  rowHeight = input(80, ...(ngDevMode ? [{ debugName: "rowHeight" }] : /* istanbul ignore next */ []));
197
200
  gap = input(12, ...(ngDevMode ? [{ debugName: "gap" }] : /* istanbul ignore next */ []));
198
- maxColumns = input(this.MAX_WIDGET_SCALE, ...(ngDevMode ? [{ debugName: "maxColumns" }] : /* istanbul ignore next */ []));
199
- maxRows = input(this.MAX_WIDGET_SCALE, ...(ngDevMode ? [{ debugName: "maxRows" }] : /* istanbul ignore next */ []));
201
+ maxColumns = input(4, ...(ngDevMode ? [{ debugName: "maxColumns" }] : /* istanbul ignore next */ []));
202
+ maxRows = input(4, ...(ngDevMode ? [{ debugName: "maxRows" }] : /* istanbul ignore next */ []));
200
203
  widgetPadding = input(12, ...(ngDevMode ? [{ debugName: "widgetPadding" }] : /* istanbul ignore next */ []));
201
204
  editMode = input(true, ...(ngDevMode ? [{ debugName: "editMode" }] : /* istanbul ignore next */ []));
202
205
  widgetChange = output();
203
206
  store = inject(WidgetService);
204
207
  dashboard;
208
+ dashboardWidth = signal(0, ...(ngDevMode ? [{ debugName: "dashboardWidth" }] : /* istanbul ignore next */ []));
209
+ storageReady = signal(false, ...(ngDevMode ? [{ debugName: "storageReady" }] : /* istanbul ignore next */ []));
210
+ resizeObserver;
211
+ activeColumns = computed(() => {
212
+ const configuredColumns = this.columns();
213
+ const maxColumns = Math.max(1, this.maxColumns());
214
+ if (configuredColumns !== 'auto') {
215
+ return Math.max(1, configuredColumns);
216
+ }
217
+ const width = this.dashboardWidth();
218
+ if (width <= 0)
219
+ return 1;
220
+ const minTileWidth = Math.max(1, this.minTileWidth());
221
+ const gap = Math.max(0, this.gap());
222
+ const columns = Math.floor((width + gap) / (minTileWidth + gap));
223
+ return Math.max(1, Math.min(columns, maxColumns));
224
+ }, ...(ngDevMode ? [{ debugName: "activeColumns" }] : /* istanbul ignore next */ []));
205
225
  gridStyles = computed(() => ({
206
- gridTemplateColumns: this.getGridTemplateColumns(),
207
- gridAutoRows: `minmax(${this.clampPositiveNumber(this.rowHeight(), 80)}px, auto)`,
208
- gap: `${this.clampPositiveNumber(this.gap(), 12)}px`,
209
- padding: `${this.clampPositiveNumber(this.gap(), 12)}px`,
226
+ gridTemplateColumns: `repeat(${this.activeColumns()}, minmax(0, 1fr))`,
227
+ gridAutoRows: `${this.rowHeight()}px`,
228
+ gap: `${this.gap()}px`,
210
229
  }), ...(ngDevMode ? [{ debugName: "gridStyles" }] : /* istanbul ignore next */ []));
211
230
  constructor() {
212
231
  effect(() => {
213
- if (!this.isBrowser || !this.hasLoadedStoredWidgets())
232
+ if (!this.storageReady())
214
233
  return;
215
234
  const state = this.store.addedWidgets().map((w) => ({
216
235
  id: w.id,
217
236
  rows: w.rows,
218
237
  columns: w.columns,
219
- color: w.color,
220
- customLabel: w.customLabel,
221
238
  }));
222
239
  localStorage.setItem(this.storageKey(), JSON.stringify(state));
223
240
  });
224
- effect(() => {
225
- const widgets = this.widgets();
226
- this.setAvailableWidgets(widgets);
227
- if (!this.hasLoadedStoredWidgets() && widgets.length > 0) {
228
- this.loadFromStorage();
229
- this.hasLoadedStoredWidgets.set(true);
230
- }
231
- });
232
241
  effect(() => {
233
242
  this.store.setConfig({
234
243
  columns: this.columns(),
244
+ activeColumns: this.activeColumns(),
235
245
  minTileWidth: this.minTileWidth(),
236
246
  rowHeight: this.rowHeight(),
237
247
  gap: this.gap(),
238
- maxColumns: this.clampScale(this.maxColumns()),
239
- maxRows: this.clampScale(this.maxRows()),
248
+ maxColumns: this.maxColumns(),
249
+ maxRows: this.maxRows(),
240
250
  widgetPadding: this.widgetPadding(),
241
251
  });
242
252
  });
@@ -247,38 +257,32 @@ class DashboardWidgetsComponent {
247
257
  this.widgetChange.emit(this.store.addedWidgets());
248
258
  });
249
259
  }
260
+ ngOnInit() {
261
+ this.setAvailableWidgets();
262
+ this.loadFromStorage();
263
+ this.storageReady.set(true);
264
+ }
250
265
  ngAfterViewInit() {
251
- if (!this.isBrowser)
252
- return;
253
- wrapGrid(this.dashboard.nativeElement, { duration: 220 });
266
+ wrapGrid(this.dashboard.nativeElement, { duration: 300 });
267
+ this.updateDashboardWidth();
268
+ this.resizeObserver = new ResizeObserver(() => this.updateDashboardWidth());
269
+ this.resizeObserver.observe(this.dashboard.nativeElement);
254
270
  }
255
- setAvailableWidgets(widgets) {
256
- this.store.setAvailableWidgets(widgets ?? []);
271
+ ngOnDestroy() {
272
+ this.resizeObserver?.disconnect();
257
273
  }
258
- getGridTemplateColumns() {
259
- if (this.columns() === 'auto') {
260
- const minTileWidth = this.clampPositiveNumber(this.minTileWidth(), 200);
261
- return `repeat(auto-fit, minmax(min(${minTileWidth}px, 100%), 1fr))`;
274
+ setAvailableWidgets() {
275
+ const widgets = this.widgets();
276
+ if (widgets?.length > 0) {
277
+ this.store.setAvailableWidgets(widgets);
262
278
  }
263
- return `repeat(${this.clampScale(this.columns())}, minmax(0, 1fr))`;
264
- }
265
- clampScale(value) {
266
- return Math.max(1, Math.min(this.MAX_WIDGET_SCALE, Math.trunc(value)));
267
- }
268
- clampPositiveNumber(value, fallback) {
269
- return Number.isFinite(value) && value > 0 ? Math.trunc(value) : fallback;
270
279
  }
271
280
  loadFromStorage() {
272
- if (!this.isBrowser)
273
- return;
274
281
  const stored = localStorage.getItem(this.storageKey());
275
- if (!stored) {
282
+ if (!stored)
276
283
  return;
277
- }
278
284
  try {
279
285
  const parsed = JSON.parse(stored);
280
- if (!Array.isArray(parsed))
281
- return;
282
286
  const widgetMap = new Map(this.store.widgets().map((w) => [w.id, w]));
283
287
  if (parsed.length > 0 && typeof parsed[0] === 'number') {
284
288
  const restoredWidgets = parsed
@@ -302,12 +306,15 @@ class DashboardWidgetsComponent {
302
306
  console.error(this.ERROR_PARSING_MESSAGE, e);
303
307
  }
304
308
  }
309
+ updateDashboardWidth() {
310
+ this.dashboardWidth.set(this.dashboard.nativeElement.clientWidth);
311
+ }
305
312
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: DashboardWidgetsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
306
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.10", type: DashboardWidgetsComponent, isStandalone: false, selector: "fidurcode-dashboard-widgets", inputs: { title: { classPropertyName: "title", publicName: "title", isSignal: true, isRequired: true, transformFunction: null }, widgets: { classPropertyName: "widgets", publicName: "widgets", isSignal: true, isRequired: true, transformFunction: null }, storageKey: { classPropertyName: "storageKey", publicName: "storageKey", isSignal: true, isRequired: false, transformFunction: null }, columns: { classPropertyName: "columns", publicName: "columns", isSignal: true, isRequired: false, transformFunction: null }, minTileWidth: { classPropertyName: "minTileWidth", publicName: "minTileWidth", isSignal: true, isRequired: false, transformFunction: null }, rowHeight: { classPropertyName: "rowHeight", publicName: "rowHeight", isSignal: true, isRequired: false, transformFunction: null }, gap: { classPropertyName: "gap", publicName: "gap", isSignal: true, isRequired: false, transformFunction: null }, maxColumns: { classPropertyName: "maxColumns", publicName: "maxColumns", isSignal: true, isRequired: false, transformFunction: null }, maxRows: { classPropertyName: "maxRows", publicName: "maxRows", isSignal: true, isRequired: false, transformFunction: null }, widgetPadding: { classPropertyName: "widgetPadding", publicName: "widgetPadding", isSignal: true, isRequired: false, transformFunction: null }, editMode: { classPropertyName: "editMode", publicName: "editMode", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { widgetChange: "widgetChange" }, providers: [WidgetService], viewQueries: [{ propertyName: "dashboard", first: true, predicate: ["dashboard"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"dashboard-widgets-header\">\n <h2 class=\"dashboard-title\">{{title()}}</h2>\n <div class=\"dashboard-header-actions\">\n @if (store.editMode()) {\n <button\n [matMenuTriggerFor]=\"addMenu\"\n [disabled]=\"store.widgetsToAdd().length === 0\"\n mat-raised-button>\n <mat-icon>add_circle</mat-icon>\n {{ 'DASHBOARD.ADD_WIDGET' | translate }}\n </button>\n <mat-menu #addMenu=\"matMenu\">\n @for (widget of store.widgetsToAdd(); track widget.id) {\n <button mat-menu-item (click)=\"store.addWidget(widget)\">{{ widget.label | translate }}</button>\n } @empty {\n <button mat-menu-item disabled>{{ 'DASHBOARD.NO_WIDGETS' | translate }}</button>\n }\n </mat-menu>\n }\n </div>\n</div>\n<div #dashboard class=\"dashboard-widgets\" [style]=\"gridStyles()\">\n @for (w of store.addedWidgets(); track w.id) {\n <app-widget [data]=\"w\"/>\n }\n</div>\n", styles: [":host{display:block;color:inherit;background-color:inherit}.dashboard-widgets-header{display:flex;justify-content:space-between;align-items:center;gap:12px;margin-bottom:8px;flex-wrap:wrap}.dashboard-title{margin:0;min-width:0;overflow-wrap:anywhere;line-height:1.2}.dashboard-header-actions{display:flex;align-items:center;gap:8px;flex-wrap:wrap}.dashboard-header-actions button{color:inherit;background-color:inherit;--mdc-protected-button-container-color: inherit;--mdc-protected-button-label-text-color: currentColor;--mat-protected-button-state-layer-color: currentColor}.dashboard-widgets{display:grid;grid-auto-flow:dense;align-items:stretch;justify-items:stretch;width:100%;box-sizing:border-box;min-width:0;overflow:visible;contain:layout style}@media(max-width:599px){.dashboard-widgets-header{align-items:stretch;flex-direction:column}.dashboard-header-actions,.dashboard-header-actions button{width:100%}.dashboard-widgets{padding-inline:0!important}}\n"], dependencies: [{ kind: "component", type: i1$2.MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: i1$2.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i1$2.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "component", type: i2.MatButton, selector: " button[matButton], a[matButton], button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button], a[mat-button], a[mat-raised-button], a[mat-flat-button], a[mat-stroked-button] ", inputs: ["matButton"], exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: WidgetComponent, selector: "app-widget", inputs: ["data"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] });
313
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.10", type: DashboardWidgetsComponent, isStandalone: false, selector: "fidurcode-dashboard-widgets", inputs: { title: { classPropertyName: "title", publicName: "title", isSignal: true, isRequired: true, transformFunction: null }, widgets: { classPropertyName: "widgets", publicName: "widgets", isSignal: true, isRequired: true, transformFunction: null }, storageKey: { classPropertyName: "storageKey", publicName: "storageKey", isSignal: true, isRequired: false, transformFunction: null }, columns: { classPropertyName: "columns", publicName: "columns", isSignal: true, isRequired: false, transformFunction: null }, minTileWidth: { classPropertyName: "minTileWidth", publicName: "minTileWidth", isSignal: true, isRequired: false, transformFunction: null }, rowHeight: { classPropertyName: "rowHeight", publicName: "rowHeight", isSignal: true, isRequired: false, transformFunction: null }, gap: { classPropertyName: "gap", publicName: "gap", isSignal: true, isRequired: false, transformFunction: null }, maxColumns: { classPropertyName: "maxColumns", publicName: "maxColumns", isSignal: true, isRequired: false, transformFunction: null }, maxRows: { classPropertyName: "maxRows", publicName: "maxRows", isSignal: true, isRequired: false, transformFunction: null }, widgetPadding: { classPropertyName: "widgetPadding", publicName: "widgetPadding", isSignal: true, isRequired: false, transformFunction: null }, editMode: { classPropertyName: "editMode", publicName: "editMode", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { widgetChange: "widgetChange" }, providers: [WidgetService], viewQueries: [{ propertyName: "dashboard", first: true, predicate: ["dashboard"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"dashboard-widgets-header\">\n <h2 class=\"dashboard-title\">{{title()}}</h2>\n <div class=\"dashboard-header-actions\">\n <button\n mat-icon-button\n (click)=\"store.toggleEditMode()\"\n [matTooltip]=\"(store.editMode() ? 'DASHBOARD.LOCK' : 'DASHBOARD.UNLOCK') | translate\">\n <mat-icon>{{ store.editMode() ? 'lock_open' : 'lock' }}</mat-icon>\n </button>\n @if (store.editMode()) {\n <button [matMenuTriggerFor]=\"addMenu\" mat-raised-button>\n <mat-icon>add_circle</mat-icon>\n {{ 'DASHBOARD.ADD_WIDGET' | translate }}\n </button>\n <mat-menu #addMenu=\"matMenu\">\n @for (widget of store.widgetsToAdd(); track widget.id) {\n <button mat-menu-item (click)=\"store.addWidget(widget)\">{{ widget.label | translate }}</button>\n } @empty {\n <button mat-menu-item disabled>{{ 'DASHBOARD.NO_WIDGETS' | translate }}</button>\n }\n </mat-menu>\n }\n </div>\n</div>\n<div #dashboard class=\"dashboard-widgets\" [style]=\"gridStyles()\">\n @for (w of store.addedWidgets(); track w.id) {\n <app-widget [data]=\"w\"/>\n }\n</div>", styles: [".dashboard-widgets-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:8px}.dashboard-title{margin:0}.dashboard-header-actions{display:flex;align-items:center;gap:8px}.dashboard-widgets{display:grid;width:100%;min-width:0;align-items:stretch}\n"], dependencies: [{ kind: "component", type: i1$2.MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: i1$2.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i1$2.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "component", type: i2.MatButton, selector: " button[matButton], a[matButton], button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button], a[mat-button], a[mat-raised-button], a[mat-flat-button], a[mat-stroked-button] ", inputs: ["matButton"], exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button], a[mat-icon-button], button[matIconButton], a[matIconButton]", exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: WidgetComponent, selector: "app-widget", inputs: ["data"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] });
307
314
  }
308
315
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: DashboardWidgetsComponent, decorators: [{
309
316
  type: Component,
310
- args: [{ selector: 'fidurcode-dashboard-widgets', standalone: false, providers: [WidgetService], template: "<div class=\"dashboard-widgets-header\">\n <h2 class=\"dashboard-title\">{{title()}}</h2>\n <div class=\"dashboard-header-actions\">\n @if (store.editMode()) {\n <button\n [matMenuTriggerFor]=\"addMenu\"\n [disabled]=\"store.widgetsToAdd().length === 0\"\n mat-raised-button>\n <mat-icon>add_circle</mat-icon>\n {{ 'DASHBOARD.ADD_WIDGET' | translate }}\n </button>\n <mat-menu #addMenu=\"matMenu\">\n @for (widget of store.widgetsToAdd(); track widget.id) {\n <button mat-menu-item (click)=\"store.addWidget(widget)\">{{ widget.label | translate }}</button>\n } @empty {\n <button mat-menu-item disabled>{{ 'DASHBOARD.NO_WIDGETS' | translate }}</button>\n }\n </mat-menu>\n }\n </div>\n</div>\n<div #dashboard class=\"dashboard-widgets\" [style]=\"gridStyles()\">\n @for (w of store.addedWidgets(); track w.id) {\n <app-widget [data]=\"w\"/>\n }\n</div>\n", styles: [":host{display:block;color:inherit;background-color:inherit}.dashboard-widgets-header{display:flex;justify-content:space-between;align-items:center;gap:12px;margin-bottom:8px;flex-wrap:wrap}.dashboard-title{margin:0;min-width:0;overflow-wrap:anywhere;line-height:1.2}.dashboard-header-actions{display:flex;align-items:center;gap:8px;flex-wrap:wrap}.dashboard-header-actions button{color:inherit;background-color:inherit;--mdc-protected-button-container-color: inherit;--mdc-protected-button-label-text-color: currentColor;--mat-protected-button-state-layer-color: currentColor}.dashboard-widgets{display:grid;grid-auto-flow:dense;align-items:stretch;justify-items:stretch;width:100%;box-sizing:border-box;min-width:0;overflow:visible;contain:layout style}@media(max-width:599px){.dashboard-widgets-header{align-items:stretch;flex-direction:column}.dashboard-header-actions,.dashboard-header-actions button{width:100%}.dashboard-widgets{padding-inline:0!important}}\n"] }]
317
+ args: [{ selector: 'fidurcode-dashboard-widgets', standalone: false, providers: [WidgetService], template: "<div class=\"dashboard-widgets-header\">\n <h2 class=\"dashboard-title\">{{title()}}</h2>\n <div class=\"dashboard-header-actions\">\n <button\n mat-icon-button\n (click)=\"store.toggleEditMode()\"\n [matTooltip]=\"(store.editMode() ? 'DASHBOARD.LOCK' : 'DASHBOARD.UNLOCK') | translate\">\n <mat-icon>{{ store.editMode() ? 'lock_open' : 'lock' }}</mat-icon>\n </button>\n @if (store.editMode()) {\n <button [matMenuTriggerFor]=\"addMenu\" mat-raised-button>\n <mat-icon>add_circle</mat-icon>\n {{ 'DASHBOARD.ADD_WIDGET' | translate }}\n </button>\n <mat-menu #addMenu=\"matMenu\">\n @for (widget of store.widgetsToAdd(); track widget.id) {\n <button mat-menu-item (click)=\"store.addWidget(widget)\">{{ widget.label | translate }}</button>\n } @empty {\n <button mat-menu-item disabled>{{ 'DASHBOARD.NO_WIDGETS' | translate }}</button>\n }\n </mat-menu>\n }\n </div>\n</div>\n<div #dashboard class=\"dashboard-widgets\" [style]=\"gridStyles()\">\n @for (w of store.addedWidgets(); track w.id) {\n <app-widget [data]=\"w\"/>\n }\n</div>", styles: [".dashboard-widgets-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:8px}.dashboard-title{margin:0}.dashboard-header-actions{display:flex;align-items:center;gap:8px}.dashboard-widgets{display:grid;width:100%;min-width:0;align-items:stretch}\n"] }]
311
318
  }], ctorParameters: () => [], propDecorators: { title: [{ type: i0.Input, args: [{ isSignal: true, alias: "title", required: true }] }], widgets: [{ type: i0.Input, args: [{ isSignal: true, alias: "widgets", required: true }] }], storageKey: [{ type: i0.Input, args: [{ isSignal: true, alias: "storageKey", required: false }] }], columns: [{ type: i0.Input, args: [{ isSignal: true, alias: "columns", required: false }] }], minTileWidth: [{ type: i0.Input, args: [{ isSignal: true, alias: "minTileWidth", required: false }] }], rowHeight: [{ type: i0.Input, args: [{ isSignal: true, alias: "rowHeight", required: false }] }], gap: [{ type: i0.Input, args: [{ isSignal: true, alias: "gap", required: false }] }], maxColumns: [{ type: i0.Input, args: [{ isSignal: true, alias: "maxColumns", required: false }] }], maxRows: [{ type: i0.Input, args: [{ isSignal: true, alias: "maxRows", required: false }] }], widgetPadding: [{ type: i0.Input, args: [{ isSignal: true, alias: "widgetPadding", required: false }] }], editMode: [{ type: i0.Input, args: [{ isSignal: true, alias: "editMode", required: false }] }], widgetChange: [{ type: i0.Output, args: ["widgetChange"] }], dashboard: [{
312
319
  type: ViewChild,
313
320
  args: ['dashboard', { static: true }]
@@ -321,29 +328,25 @@ class DashboardWidgetsSkeletonModule {
321
328
  MatButtonToggleModule,
322
329
  MatButtonModule,
323
330
  MatIconModule,
324
- MatTooltipModule,
325
- MatDividerModule], exports: [WidgetComponent,
331
+ MatTooltipModule], exports: [WidgetComponent,
326
332
  WidgetOptionsComponent,
327
333
  DashboardWidgetsComponent,
328
334
  MatMenuModule,
329
335
  MatButtonToggleModule,
330
336
  MatButtonModule,
331
337
  MatIconModule,
332
- MatTooltipModule,
333
- MatDividerModule] });
338
+ MatTooltipModule] });
334
339
  static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: DashboardWidgetsSkeletonModule, imports: [CommonModule,
335
340
  TranslateModule.forChild(),
336
341
  MatMenuModule,
337
342
  MatButtonToggleModule,
338
343
  MatButtonModule,
339
344
  MatIconModule,
340
- MatTooltipModule,
341
- MatDividerModule, MatMenuModule,
345
+ MatTooltipModule, MatMenuModule,
342
346
  MatButtonToggleModule,
343
347
  MatButtonModule,
344
348
  MatIconModule,
345
- MatTooltipModule,
346
- MatDividerModule] });
349
+ MatTooltipModule] });
347
350
  }
348
351
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.10", ngImport: i0, type: DashboardWidgetsSkeletonModule, decorators: [{
349
352
  type: NgModule,
@@ -361,7 +364,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.10", ngImpo
361
364
  MatButtonModule,
362
365
  MatIconModule,
363
366
  MatTooltipModule,
364
- MatDividerModule,
365
367
  ],
366
368
  exports: [
367
369
  WidgetComponent,
@@ -372,7 +374,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.10", ngImpo
372
374
  MatButtonModule,
373
375
  MatIconModule,
374
376
  MatTooltipModule,
375
- MatDividerModule,
376
377
  ],
377
378
  }]
378
379
  }] });
@@ -1 +1 @@
1
- {"version":3,"file":"fidurcode-dashboard-widgets-skeleton.mjs","sources":["../../../projects/dashboard-widgets-skeleton/src/lib/models/dashboard-config.ts","../../../projects/dashboard-widgets-skeleton/src/lib/services/widget.service.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget-options/widget-options.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget-options/widget-options.component.html","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.component.html","../../../projects/dashboard-widgets-skeleton/src/lib/components/dashboard-widgets/dashboard-widgets.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/dashboard-widgets/dashboard-widgets.component.html","../../../projects/dashboard-widgets-skeleton/src/lib/dashboard-widgets-skeleton.module.ts","../../../projects/dashboard-widgets-skeleton/src/public-api.ts","../../../projects/dashboard-widgets-skeleton/src/fidurcode-dashboard-widgets-skeleton.ts"],"sourcesContent":["export interface DashboardConfig {\n columns: number | 'auto';\n minTileWidth: number;\n rowHeight: number;\n gap: number;\n maxColumns: number;\n maxRows: number;\n widgetPadding: number;\n}\n\nexport const DEFAULT_DASHBOARD_CONFIG: DashboardConfig = {\n columns: 'auto',\n minTileWidth: 200,\n rowHeight: 80,\n gap: 12,\n maxColumns: 5,\n maxRows: 5,\n widgetPadding: 12,\n};\n","import {\n computed,\n Injectable,\n Signal,\n signal,\n WritableSignal,\n} from '@angular/core';\nimport { Widget } from '../components/widget/widget';\nimport { DashboardConfig, DEFAULT_DASHBOARD_CONFIG } from '../models/dashboard-config';\n\n@Injectable()\nexport class WidgetService {\n private readonly DEFAULT_WIDGET_SIZE = 3;\n\n public widgets: WritableSignal<Widget[]> = signal<Widget[]>([]);\n addedWidgets: WritableSignal<Widget[]> = signal<Widget[]>([]);\n config: WritableSignal<DashboardConfig> = signal<DashboardConfig>(DEFAULT_DASHBOARD_CONFIG);\n editMode: WritableSignal<boolean> = signal<boolean>(true);\n\n widgetsToAdd: Signal<Widget[]> = computed((): Widget[] => {\n const addedIds = new Set(this.addedWidgets().map((widget: Widget): number => widget.id));\n return this.widgets().filter((w): boolean => !addedIds.has(w.id));\n });\n\n setAvailableWidgets(widgets: Widget[]): void {\n this.widgets.set(widgets);\n }\n\n setConfig(config: Partial<DashboardConfig>): void {\n this.config.set({ ...DEFAULT_DASHBOARD_CONFIG, ...config });\n }\n\n addWidget(widget: Widget): void {\n if (this.addedWidgets().some((addedWidget) => addedWidget.id === widget.id)) return;\n\n this.addedWidgets.set([\n ...this.addedWidgets(),\n {\n ...widget,\n rows: this.clampScale(widget.defaultRows ?? widget.rows ?? this.DEFAULT_WIDGET_SIZE, this.config().maxRows),\n columns: this.clampScale(\n widget.defaultColumns ?? widget.columns ?? this.DEFAULT_WIDGET_SIZE,\n this.config().maxColumns,\n ),\n },\n ]);\n }\n\n updateWidget(id: number, widget: Partial<Widget>): void {\n const index = this.addedWidgets().findIndex((w) => w.id === id);\n if (index !== -1) {\n const newWidgets = [...this.addedWidgets()];\n newWidgets[index] = {\n ...newWidgets[index],\n ...widget,\n rows: widget.rows === undefined ? newWidgets[index].rows : this.clampScale(widget.rows, this.config().maxRows),\n columns:\n widget.columns === undefined\n ? newWidgets[index].columns\n : this.clampScale(widget.columns, this.config().maxColumns),\n };\n this.addedWidgets.set(newWidgets);\n }\n }\n\n renameWidget(id: number, customLabel: string): void {\n this.updateWidget(id, { customLabel: customLabel.trim() || undefined });\n }\n\n setWidgetColor(id: number, color: string | undefined): void {\n this.updateWidget(id, { color });\n }\n\n toggleEditMode(): void {\n this.editMode.set(!this.editMode());\n }\n\n moveWidgetToLeft(id: number): void {\n const index = this.addedWidgets().findIndex((w) => w.id === id);\n if (index <= 0) return;\n const newWidgets = [...this.addedWidgets()];\n [newWidgets[index], newWidgets[index - 1]] = [{ ...newWidgets[index - 1] }, { ...newWidgets[index] }];\n this.addedWidgets.set(newWidgets);\n }\n\n moveWidgetToRight(id: number): void {\n const index = this.addedWidgets().findIndex((w) => w.id === id);\n if (index === -1 || index === this.addedWidgets().length - 1) return;\n const newWidgets = [...this.addedWidgets()];\n [newWidgets[index], newWidgets[index + 1]] = [{ ...newWidgets[index + 1] }, { ...newWidgets[index] }];\n this.addedWidgets.set(newWidgets);\n }\n\n deleteWidget(id: number): void {\n this.addedWidgets.set(this.addedWidgets().filter((w) => w.id !== id));\n }\n\n private clampScale(value: number, max: number): number {\n const safeValue = Number.isFinite(value) ? Math.trunc(value) : this.DEFAULT_WIDGET_SIZE;\n const safeMax = Number.isFinite(max) ? Math.trunc(max) : this.DEFAULT_WIDGET_SIZE;\n return Math.max(1, Math.min(Math.max(1, safeMax), safeValue));\n }\n}\n","import {\n Component,\n computed,\n inject,\n input,\n InputSignal,\n model,\n ModelSignal,\n Signal,\n} from '@angular/core';\nimport { Widget } from '../widget/widget';\nimport { WidgetService } from '../../services/widget.service';\n\n@Component({\n selector: 'app-widget-options',\n standalone: false,\n templateUrl: './widget-options.component.html',\n styleUrl: './widget-options.component.scss',\n})\nexport class WidgetOptionsComponent {\n private readonly DEFAULT_WIDGET_SIZE = 3;\n private readonly MAX_WIDGET_SCALE = 5;\n\n widget: InputSignal<Widget> = input.required<Widget>();\n showOptions: ModelSignal<boolean> = model<boolean>(false);\n\n store = inject(WidgetService);\n\n colOptions: Signal<number[]> = computed(() =>\n Array.from({ length: this.clampOptionCount(this.store.config().maxColumns) }, (_, i) => i + 1),\n );\n\n rowOptions: Signal<number[]> = computed(() =>\n Array.from({ length: this.clampOptionCount(this.store.config().maxRows) }, (_, i) => i + 1),\n );\n\n selectedColumns: Signal<number> = computed(() =>\n this.clampOptionCount(this.widget().columns ?? this.DEFAULT_WIDGET_SIZE),\n );\n selectedRows: Signal<number> = computed(() =>\n this.clampOptionCount(this.widget().rows ?? this.DEFAULT_WIDGET_SIZE),\n );\n canMoveLeft: Signal<boolean> = computed(() => this.widgetIndex() > 0);\n canMoveRight: Signal<boolean> = computed(() => {\n const index = this.widgetIndex();\n return index !== -1 && index < this.store.addedWidgets().length - 1;\n });\n\n onRenameBlur(event: Event): void {\n const input = event.target as HTMLInputElement;\n this.store.renameWidget(this.widget().id, input.value);\n }\n\n private clampOptionCount(value: number): number {\n return Math.max(1, Math.min(this.MAX_WIDGET_SCALE, Math.trunc(value)));\n }\n\n private widgetIndex(): number {\n return this.store.addedWidgets().findIndex((widget) => widget.id === this.widget().id);\n }\n}\n","<div class=\"options-top-bar\">\n <button\n mat-icon-button\n class=\"delete-btn\"\n (click)=\"store.deleteWidget(widget().id)\"\n [attr.aria-label]=\"'WIDGET.DELETE' | translate\"\n [matTooltip]=\"'WIDGET.DELETE' | translate\">\n <mat-icon>delete</mat-icon>\n </button>\n <span class=\"spacer\"></span>\n <button\n mat-icon-button\n (click)=\"showOptions.set(false)\"\n [attr.aria-label]=\"'WIDGET.CLOSE_OPTIONS' | translate\"\n [matTooltip]=\"'WIDGET.CLOSE_OPTIONS' | translate\">\n <mat-icon>close</mat-icon>\n </button>\n</div>\n\n<div class=\"options-body\">\n <div class=\"option-group\">\n <span class=\"option-label\">{{ 'WIDGET.RENAME' | translate }}</span>\n <input\n class=\"rename-input\"\n [value]=\"widget().customLabel || ''\"\n (blur)=\"onRenameBlur($event)\"\n (keydown.enter)=\"$any($event.target).blur()\"\n (keydown.escape)=\"showOptions.set(false)\"\n [attr.aria-label]=\"'WIDGET.RENAME' | translate\"\n [placeholder]=\"widget().label | translate\"\n />\n </div>\n\n <mat-divider/>\n\n <div class=\"option-group\">\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.WIDTH' | translate }}</span>\n <mat-button-toggle-group\n class=\"size-toggle-group\"\n (change)=\"store.updateWidget(widget().id, { columns: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"selectedColumns()\">\n @for (n of colOptions(); track n) {\n <mat-button-toggle class=\"size-toggle\" [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.HEIGHT' | translate }}</span>\n <mat-button-toggle-group\n class=\"size-toggle-group\"\n (change)=\"store.updateWidget(widget().id, { rows: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"selectedRows()\">\n @for (n of rowOptions(); track n) {\n <mat-button-toggle class=\"size-toggle\" [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n </div>\n\n</div>\n\n<button\n class=\"move-left-btn\"\n mat-icon-button\n (click)=\"store.moveWidgetToLeft(widget().id)\"\n [disabled]=\"!canMoveLeft()\"\n [attr.aria-label]=\"'WIDGET.MOVE_LEFT' | translate\"\n [matTooltip]=\"'WIDGET.MOVE_LEFT' | translate\">\n <mat-icon>chevron_left</mat-icon>\n</button>\n<button\n class=\"move-right-btn\"\n mat-icon-button\n (click)=\"store.moveWidgetToRight(widget().id)\"\n [disabled]=\"!canMoveRight()\"\n [attr.aria-label]=\"'WIDGET.MOVE_RIGHT' | translate\"\n [matTooltip]=\"'WIDGET.MOVE_RIGHT' | translate\">\n <mat-icon>chevron_right</mat-icon>\n</button>\n","import {\n Component,\n computed,\n inject,\n input,\n InputSignal,\n Signal,\n signal,\n WritableSignal,\n} from '@angular/core';\nimport { Widget } from './widget';\nimport { WidgetService } from '../../services/widget.service';\n\n@Component({\n selector: 'app-widget',\n standalone: false,\n templateUrl: './widget.component.html',\n styleUrl: './widget.component.scss',\n host: {\n '[style.grid-area]': 'gridArea()',\n '[style.--widget-padding]': 'widgetPadding()',\n },\n})\nexport class WidgetComponent {\n private readonly DEFAULT_WIDGET_SIZE = 3;\n\n data: InputSignal<Widget> = input.required<Widget>();\n\n store = inject(WidgetService);\n\n showWidgetOptions: WritableSignal<boolean> = signal<boolean>(false);\n isRenaming: WritableSignal<boolean> = signal<boolean>(false);\n\n gridArea: Signal<string> = computed(() => {\n const widget = this.data();\n const rows = this.clampScale(widget.rows ?? this.DEFAULT_WIDGET_SIZE, this.store.config().maxRows);\n const columns = this.clampScale(widget.columns ?? this.DEFAULT_WIDGET_SIZE, this.store.config().maxColumns);\n return `span ${rows} / span ${columns}`;\n });\n\n widgetPadding: Signal<string> = computed(() => `${this.store.config().widgetPadding}px`);\n\n startRename(): void {\n if (!this.store.editMode()) return;\n this.isRenaming.set(true);\n }\n\n finishRename(event: Event): void {\n const input = event.target as HTMLInputElement;\n this.store.renameWidget(this.data().id, input.value);\n this.isRenaming.set(false);\n }\n\n private clampScale(value: number, max: number): number {\n return Math.max(1, Math.min(max, Math.trunc(value)));\n }\n}\n","<div class=\"widget-container mat-elevation-z3\" [style.--widget-color]=\"data().color\">\n @if (!showWidgetOptions()) {\n <div class=\"widget-header\">\n @if (isRenaming()) {\n <input\n class=\"rename-input\"\n [value]=\"data().customLabel || ''\"\n (blur)=\"finishRename($event)\"\n (keydown.enter)=\"finishRename($event)\"\n (keydown.escape)=\"isRenaming.set(false)\"\n [attr.aria-label]=\"'WIDGET.RENAME' | translate\"\n [placeholder]=\"data().label | translate\"\n autofocus\n />\n } @else {\n <h3 class=\"widget-title\" (dblclick)=\"startRename()\">\n @if (data().customLabel) {\n {{ data().customLabel }}\n } @else {\n {{ data().label | translate }}\n }\n </h3>\n }\n <div class=\"widget-actions\">\n @if (store.editMode()) {\n <button\n mat-icon-button\n (click)=\"showWidgetOptions.set(true)\"\n [attr.aria-label]=\"'WIDGET.SETTINGS' | translate\"\n [matTooltip]=\"'WIDGET.SETTINGS' | translate\">\n <mat-icon>settings</mat-icon>\n </button>\n }\n </div>\n </div>\n\n <div class=\"widget-content\">\n <ng-container [ngComponentOutlet]=\"data().content\"/>\n </div>\n }\n\n @if (showWidgetOptions()) {\n <app-widget-options [(showOptions)]=\"showWidgetOptions\" [widget]=\"data()\"/>\n }\n</div>\n","import {\n AfterViewInit,\n Component,\n computed,\n effect,\n ElementRef,\n inject,\n input,\n InputSignal,\n output,\n OutputEmitterRef,\n Signal,\n signal,\n ViewChild,\n WritableSignal,\n PLATFORM_ID,\n} from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { WidgetService } from '../../services/widget.service';\nimport { wrapGrid } from 'animate-css-grid';\nimport { Widget } from '../widget/widget';\n\ntype StoredWidgetState = Pick<Widget, 'id' | 'rows' | 'columns' | 'color' | 'customLabel'>;\n\n@Component({\n selector: 'fidurcode-dashboard-widgets',\n standalone: false,\n templateUrl: './dashboard-widgets.component.html',\n styleUrls: ['./dashboard-widgets.component.scss'],\n providers: [WidgetService],\n})\nexport class DashboardWidgetsComponent implements AfterViewInit {\n private readonly DEFAULT_STORAGE_KEY = 'dashboard-widgets';\n private readonly ERROR_PARSING_MESSAGE = 'Error parsing widgets from localStorage';\n private readonly MAX_WIDGET_SCALE = 5;\n private readonly platformId = inject(PLATFORM_ID);\n private readonly isBrowser = isPlatformBrowser(this.platformId);\n private hasLoadedStoredWidgets: WritableSignal<boolean> = signal<boolean>(false);\n\n title: InputSignal<string> = input.required<string>();\n widgets: InputSignal<Widget[]> = input.required<Widget[]>();\n storageKey: InputSignal<string> = input<string>(this.DEFAULT_STORAGE_KEY);\n\n columns: InputSignal<number | 'auto'> = input<number | 'auto'>('auto');\n minTileWidth: InputSignal<number> = input<number>(200);\n rowHeight: InputSignal<number> = input<number>(80);\n gap: InputSignal<number> = input<number>(12);\n maxColumns: InputSignal<number> = input<number>(this.MAX_WIDGET_SCALE);\n maxRows: InputSignal<number> = input<number>(this.MAX_WIDGET_SCALE);\n widgetPadding: InputSignal<number> = input<number>(12);\n editMode: InputSignal<boolean> = input<boolean>(true);\n\n widgetChange: OutputEmitterRef<Widget[]> = output<Widget[]>();\n\n public store = inject(WidgetService);\n\n @ViewChild('dashboard', { static: true }) dashboard!: ElementRef;\n\n gridStyles: Signal<Record<string, string>> = computed(() => ({\n gridTemplateColumns: this.getGridTemplateColumns(),\n gridAutoRows: `minmax(${this.clampPositiveNumber(this.rowHeight(), 80)}px, auto)`,\n gap: `${this.clampPositiveNumber(this.gap(), 12)}px`,\n padding: `${this.clampPositiveNumber(this.gap(), 12)}px`,\n }));\n\n constructor() {\n effect((): void => {\n if (!this.isBrowser || !this.hasLoadedStoredWidgets()) return;\n\n const state: StoredWidgetState[] = this.store.addedWidgets().map((w) => ({\n id: w.id,\n rows: w.rows,\n columns: w.columns,\n color: w.color,\n customLabel: w.customLabel,\n }));\n localStorage.setItem(this.storageKey(), JSON.stringify(state));\n });\n\n effect((): void => {\n const widgets = this.widgets();\n this.setAvailableWidgets(widgets);\n\n if (!this.hasLoadedStoredWidgets() && widgets.length > 0) {\n this.loadFromStorage();\n this.hasLoadedStoredWidgets.set(true);\n }\n });\n\n effect((): void => {\n this.store.setConfig({\n columns: this.columns(),\n minTileWidth: this.minTileWidth(),\n rowHeight: this.rowHeight(),\n gap: this.gap(),\n maxColumns: this.clampScale(this.maxColumns()),\n maxRows: this.clampScale(this.maxRows()),\n widgetPadding: this.widgetPadding(),\n });\n });\n\n effect((): void => {\n this.store.editMode.set(this.editMode());\n });\n\n effect((): void => {\n this.widgetChange.emit(this.store.addedWidgets());\n });\n }\n\n ngAfterViewInit(): void {\n if (!this.isBrowser) return;\n\n wrapGrid(this.dashboard.nativeElement, { duration: 220 });\n }\n\n private setAvailableWidgets(widgets: Widget[]): void {\n this.store.setAvailableWidgets(widgets ?? []);\n }\n\n private getGridTemplateColumns(): string {\n if (this.columns() === 'auto') {\n const minTileWidth = this.clampPositiveNumber(this.minTileWidth(), 200);\n return `repeat(auto-fit, minmax(min(${minTileWidth}px, 100%), 1fr))`;\n }\n\n return `repeat(${this.clampScale(this.columns() as number)}, minmax(0, 1fr))`;\n }\n\n private clampScale(value: number): number {\n return Math.max(1, Math.min(this.MAX_WIDGET_SCALE, Math.trunc(value)));\n }\n\n private clampPositiveNumber(value: number, fallback: number): number {\n return Number.isFinite(value) && value > 0 ? Math.trunc(value) : fallback;\n }\n\n private loadFromStorage(): void {\n if (!this.isBrowser) return;\n\n const stored = localStorage.getItem(this.storageKey());\n if (!stored) {\n return;\n }\n\n try {\n const parsed = JSON.parse(stored) as unknown;\n if (!Array.isArray(parsed)) return;\n\n const widgetMap = new Map(this.store.widgets().map((w) => [w.id, w]));\n\n if (parsed.length > 0 && typeof parsed[0] === 'number') {\n const restoredWidgets = (parsed as number[])\n .map((id) => widgetMap.get(id))\n .filter((w): w is Widget => w !== undefined);\n this.store.addedWidgets.set(restoredWidgets);\n return;\n }\n\n const states = parsed as StoredWidgetState[];\n const restoredWidgets = states\n .map((state) => {\n const def = widgetMap.get(state.id);\n if (!def) return undefined;\n return { ...def, ...state };\n })\n .filter((w): w is Widget => w !== undefined);\n this.store.addedWidgets.set(restoredWidgets);\n } catch (e) {\n console.error(this.ERROR_PARSING_MESSAGE, e);\n }\n }\n}\n","<div class=\"dashboard-widgets-header\">\n <h2 class=\"dashboard-title\">{{title()}}</h2>\n <div class=\"dashboard-header-actions\">\n @if (store.editMode()) {\n <button\n [matMenuTriggerFor]=\"addMenu\"\n [disabled]=\"store.widgetsToAdd().length === 0\"\n mat-raised-button>\n <mat-icon>add_circle</mat-icon>\n {{ 'DASHBOARD.ADD_WIDGET' | translate }}\n </button>\n <mat-menu #addMenu=\"matMenu\">\n @for (widget of store.widgetsToAdd(); track widget.id) {\n <button mat-menu-item (click)=\"store.addWidget(widget)\">{{ widget.label | translate }}</button>\n } @empty {\n <button mat-menu-item disabled>{{ 'DASHBOARD.NO_WIDGETS' | translate }}</button>\n }\n </mat-menu>\n }\n </div>\n</div>\n<div #dashboard class=\"dashboard-widgets\" [style]=\"gridStyles()\">\n @for (w of store.addedWidgets(); track w.id) {\n <app-widget [data]=\"w\"/>\n }\n</div>\n","import { NgModule } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { WidgetComponent } from './components/widget/widget.component';\nimport { WidgetOptionsComponent } from './components/widget-options/widget-options.component';\nimport { DashboardWidgetsComponent } from './components/dashboard-widgets/dashboard-widgets.component';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { MatDividerModule } from '@angular/material/divider';\nimport { TranslateModule } from '@ngx-translate/core';\n\n@NgModule({\n declarations: [\n WidgetComponent,\n WidgetOptionsComponent,\n DashboardWidgetsComponent,\n ],\n imports: [\n CommonModule,\n TranslateModule.forChild(),\n MatMenuModule,\n MatButtonToggleModule,\n MatButtonModule,\n MatIconModule,\n MatTooltipModule,\n MatDividerModule,\n ],\n exports: [\n WidgetComponent,\n WidgetOptionsComponent,\n DashboardWidgetsComponent,\n MatMenuModule,\n MatButtonToggleModule,\n MatButtonModule,\n MatIconModule,\n MatTooltipModule,\n MatDividerModule,\n ],\n})\nexport class DashboardWidgetsSkeletonModule {}","/*\n * Public API Surface of dashboard-widgets-skeleton\n */\n\nexport * from './lib/components/widget/widget';\nexport * from './lib/components/widget/widget.component';\nexport * from './lib/components/widget-options/widget-options.component';\nexport * from './lib/components/dashboard-widgets/dashboard-widgets.component';\nexport * from './lib/models/dashboard-config';\nexport * from './lib/services/widget.service';\nexport * from './lib/dashboard-widgets-skeleton.module';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["i1","i5.WidgetOptionsComponent","i4.WidgetComponent","i5"],"mappings":"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"}
1
+ {"version":3,"file":"fidurcode-dashboard-widgets-skeleton.mjs","sources":["../../../projects/dashboard-widgets-skeleton/src/lib/models/dashboard-config.ts","../../../projects/dashboard-widgets-skeleton/src/lib/services/widget.service.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget-options/widget-options.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget-options/widget-options.component.html","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.component.html","../../../projects/dashboard-widgets-skeleton/src/lib/components/dashboard-widgets/dashboard-widgets.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/dashboard-widgets/dashboard-widgets.component.html","../../../projects/dashboard-widgets-skeleton/src/lib/dashboard-widgets-skeleton.module.ts","../../../projects/dashboard-widgets-skeleton/src/public-api.ts","../../../projects/dashboard-widgets-skeleton/src/fidurcode-dashboard-widgets-skeleton.ts"],"sourcesContent":["export interface DashboardConfig {\n columns: number | 'auto';\n activeColumns: number;\n minTileWidth: number;\n rowHeight: number;\n gap: number;\n maxColumns: number;\n maxRows: number;\n widgetPadding: number;\n}\n\nexport const DEFAULT_DASHBOARD_CONFIG: DashboardConfig = {\n columns: 'auto',\n activeColumns: 1,\n minTileWidth: 200,\n rowHeight: 80,\n gap: 12,\n maxColumns: 4,\n maxRows: 4,\n widgetPadding: 12,\n};\n","import {\n computed,\n Injectable,\n Signal,\n signal,\n WritableSignal,\n} from '@angular/core';\nimport { Widget } from '../components/widget/widget';\nimport { DashboardConfig, DEFAULT_DASHBOARD_CONFIG } from '../models/dashboard-config';\n\n@Injectable()\nexport class WidgetService {\n public widgets: WritableSignal<Widget[]> = signal<Widget[]>([]);\n addedWidgets: WritableSignal<Widget[]> = signal<Widget[]>([]);\n config: WritableSignal<DashboardConfig> = signal<DashboardConfig>(DEFAULT_DASHBOARD_CONFIG);\n editMode: WritableSignal<boolean> = signal<boolean>(true);\n\n widgetsToAdd: Signal<Widget[]> = computed((): Widget[] => {\n const addedIds: number[] = this.addedWidgets().map((widget: Widget): number => widget.id);\n return this.widgets().filter((w): boolean => !addedIds.includes(w.id));\n });\n\n setAvailableWidgets(widgets: Widget[]): void {\n this.widgets.set(widgets);\n }\n\n setConfig(config: Partial<DashboardConfig>): void {\n this.config.set({ ...DEFAULT_DASHBOARD_CONFIG, ...config });\n }\n\n addWidget(widget: Widget): void {\n this.addedWidgets.set([\n ...this.addedWidgets(),\n {\n ...widget,\n rows: this.normalizeRows(widget.defaultRows ?? widget.rows),\n columns: this.normalizeColumns(widget.defaultColumns ?? widget.columns),\n },\n ]);\n }\n\n updateWidget(id: number, widget: Partial<Widget>): void {\n const index = this.addedWidgets().findIndex((w) => w.id === id);\n if (index !== -1) {\n const newWidgets = [...this.addedWidgets()];\n newWidgets[index] = this.normalizeWidgetSize({ ...newWidgets[index], ...widget });\n this.addedWidgets.set(newWidgets);\n }\n }\n\n toggleEditMode(): void {\n this.editMode.set(!this.editMode());\n }\n\n moveWidgetToLeft(id: number): void {\n const index = this.addedWidgets().findIndex((w) => w.id === id);\n if (index === 0) return;\n const newWidgets = [...this.addedWidgets()];\n [newWidgets[index], newWidgets[index - 1]] = [{ ...newWidgets[index - 1] }, { ...newWidgets[index] }];\n this.addedWidgets.set(newWidgets);\n }\n\n moveWidgetToRight(id: number): void {\n const index = this.addedWidgets().findIndex((w) => w.id === id);\n if (index === this.addedWidgets().length - 1) return;\n const newWidgets = [...this.addedWidgets()];\n [newWidgets[index], newWidgets[index + 1]] = [{ ...newWidgets[index + 1] }, { ...newWidgets[index] }];\n this.addedWidgets.set(newWidgets);\n }\n\n deleteWidget(id: number): void {\n this.addedWidgets.set(this.addedWidgets().filter((w) => w.id !== id));\n }\n\n private normalizeWidgetSize(widget: Widget): Widget {\n return {\n ...widget,\n rows: this.normalizeRows(widget.rows),\n columns: this.normalizeColumns(widget.columns),\n };\n }\n\n private normalizeColumns(columns: number | undefined): number {\n return this.clamp(columns ?? 1, 1, this.config().maxColumns);\n }\n\n private normalizeRows(rows: number | undefined): number {\n return this.clamp(rows ?? 1, 1, this.config().maxRows);\n }\n\n private clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(Math.round(value), min), Math.max(min, max));\n }\n}\n","import {\n Component,\n computed,\n inject,\n input,\n InputSignal,\n model,\n ModelSignal,\n Signal,\n} from '@angular/core';\nimport { Widget } from '../widget/widget';\nimport { WidgetService } from '../../services/widget.service';\n\n@Component({\n selector: 'app-widget-options',\n standalone: false,\n templateUrl: './widget-options.component.html',\n styleUrl: './widget-options.component.scss',\n})\nexport class WidgetOptionsComponent {\n widget: InputSignal<Widget> = input.required<Widget>();\n showOptions: ModelSignal<boolean> = model<boolean>(false);\n\n store = inject(WidgetService);\n\n colOptions: Signal<number[]> = computed(() =>\n Array.from({ length: this.store.config().maxColumns }, (_, i) => i + 1),\n );\n\n rowOptions: Signal<number[]> = computed(() =>\n Array.from({ length: this.store.config().maxRows }, (_, i) => i + 1),\n );\n\n canMoveLeft(): boolean {\n return this.store.addedWidgets().findIndex((w) => w.id === this.widget().id) > 0;\n }\n\n canMoveRight(): boolean {\n const index = this.store.addedWidgets().findIndex((w) => w.id === this.widget().id);\n return index >= 0 && index < this.store.addedWidgets().length - 1;\n }\n}\n","<div class=\"options-top-bar\">\n <button\n mat-icon-button\n class=\"delete-btn\"\n (click)=\"store.deleteWidget(widget().id)\"\n [matTooltip]=\"'WIDGET.DELETE' | translate\">\n <mat-icon>delete</mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"store.moveWidgetToLeft(widget().id)\"\n [disabled]=\"!canMoveLeft()\"\n [matTooltip]=\"'WIDGET.MOVE_LEFT' | translate\">\n <mat-icon>chevron_left</mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"store.moveWidgetToRight(widget().id)\"\n [disabled]=\"!canMoveRight()\"\n [matTooltip]=\"'WIDGET.MOVE_RIGHT' | translate\">\n <mat-icon>chevron_right</mat-icon>\n </button>\n <span class=\"spacer\"></span>\n <button\n mat-icon-button\n (click)=\"showOptions.set(false)\"\n [matTooltip]=\"'WIDGET.CLOSE_OPTIONS' | translate\">\n <mat-icon>close</mat-icon>\n </button>\n</div>\n\n<div class=\"options-body\">\n <div class=\"option-group\">\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.WIDTH' | translate }}</span>\n <mat-button-toggle-group\n (change)=\"store.updateWidget(widget().id, { columns: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"widget().columns ?? 1\">\n @for (n of colOptions(); track n) {\n <mat-button-toggle [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n <div class=\"size-row\">\n <span class=\"option-label size-label\">{{ 'WIDGET.HEIGHT' | translate }}</span>\n <mat-button-toggle-group\n (change)=\"store.updateWidget(widget().id, { rows: +$event.value })\"\n [hideSingleSelectionIndicator]=\"true\"\n [value]=\"widget().rows ?? 1\">\n @for (n of rowOptions(); track n) {\n <mat-button-toggle [value]=\"n\">{{ n }}</mat-button-toggle>\n }\n </mat-button-toggle-group>\n </div>\n </div>\n\n</div>\n","import {\n AfterViewInit,\n Component,\n computed,\n ElementRef,\n inject,\n input,\n InputSignal,\n OnDestroy,\n Signal,\n signal,\n ViewChild,\n WritableSignal,\n} from '@angular/core';\nimport { DashboardWidgetContentSize, Widget } from './widget';\nimport { WidgetService } from '../../services/widget.service';\n\n@Component({\n selector: 'app-widget',\n standalone: false,\n templateUrl: './widget.component.html',\n styleUrl: './widget.component.scss',\n host: {\n '[style.grid-area]': 'gridArea()',\n '[style.--widget-padding]': 'widgetPadding()',\n },\n})\nexport class WidgetComponent implements AfterViewInit, OnDestroy {\n data: InputSignal<Widget> = input.required<Widget>();\n\n store = inject(WidgetService);\n\n @ViewChild('contentHost') contentHost?: ElementRef<HTMLElement>;\n\n showWidgetOptions: WritableSignal<boolean> = signal<boolean>(false);\n contentSize: WritableSignal<DashboardWidgetContentSize> = signal<DashboardWidgetContentSize>({\n width: 0,\n height: 0,\n rows: 1,\n columns: 1,\n });\n\n private resizeObserver?: ResizeObserver;\n\n gridArea: Signal<string> = computed(() => {\n const widget = this.data();\n const rows = this.normalizeSpan(widget.rows, this.store.config().maxRows);\n const columns = this.normalizeSpan(\n widget.columns,\n Math.min(this.store.config().maxColumns, this.store.config().activeColumns),\n );\n return `span ${rows} / span ${columns}`;\n });\n\n widgetPadding: Signal<string> = computed(() => `${this.store.config().widgetPadding}px`);\n\n contentInputs: Signal<{ widgetSize: DashboardWidgetContentSize }> = computed(() => ({\n widgetSize: this.contentSize(),\n }));\n\n ngAfterViewInit(): void {\n this.updateContentSize();\n const element = this.contentHost?.nativeElement;\n if (!element) return;\n this.resizeObserver = new ResizeObserver(() => this.updateContentSize());\n this.resizeObserver.observe(element);\n }\n\n ngOnDestroy(): void {\n this.resizeObserver?.disconnect();\n }\n\n private updateContentSize(): void {\n const element = this.contentHost?.nativeElement;\n if (!element) return;\n const rect = element.getBoundingClientRect();\n this.contentSize.set({\n width: Math.round(rect.width),\n height: Math.round(rect.height),\n rows: this.normalizeSpan(this.data().rows, this.store.config().maxRows),\n columns: this.normalizeSpan(\n this.data().columns,\n Math.min(this.store.config().maxColumns, this.store.config().activeColumns),\n ),\n });\n }\n\n private normalizeSpan(value: number | undefined, max: number): number {\n return Math.min(Math.max(Math.round(value ?? 1), 1), Math.max(max, 1));\n }\n}\n","<div class=\"widget-container mat-elevation-z3\">\n <div class=\"widget-header\">\n <h3 class=\"widget-title\">{{ data().label | translate }}</h3>\n <div class=\"widget-actions\">\n @if (store.editMode()) {\n <button\n mat-icon-button\n (click)=\"showWidgetOptions.set(true)\"\n [matTooltip]=\"'WIDGET.SETTINGS' | translate\">\n <mat-icon>settings</mat-icon>\n </button>\n }\n </div>\n </div>\n\n <div #contentHost class=\"widget-content\">\n <ng-container [ngComponentOutlet]=\"data().content\" [ngComponentOutletInputs]=\"contentInputs()\"/>\n </div>\n\n @if (showWidgetOptions()) {\n <app-widget-options [(showOptions)]=\"showWidgetOptions\" [widget]=\"data()\"/>\n }\n</div>\n","import {\n AfterViewInit,\n Component,\n computed,\n effect,\n ElementRef,\n inject,\n input,\n InputSignal,\n OnDestroy,\n OnInit,\n output,\n OutputEmitterRef,\n Signal,\n signal,\n ViewChild,\n WritableSignal,\n} from '@angular/core';\nimport { WidgetService } from '../../services/widget.service';\nimport { wrapGrid } from 'animate-css-grid';\nimport { Widget } from '../widget/widget';\n\ntype StoredWidgetState = Pick<Widget, 'id' | 'rows' | 'columns'>;\n\n@Component({\n selector: 'fidurcode-dashboard-widgets',\n standalone: false,\n templateUrl: './dashboard-widgets.component.html',\n styleUrls: ['./dashboard-widgets.component.scss'],\n providers: [WidgetService],\n})\nexport class DashboardWidgetsComponent implements OnInit, AfterViewInit, OnDestroy {\n private readonly DEFAULT_STORAGE_KEY = 'dashboard-widgets';\n private readonly ERROR_PARSING_MESSAGE = 'Error parsing widgets from localStorage';\n\n title: InputSignal<string> = input.required<string>();\n widgets: InputSignal<Widget[]> = input.required<Widget[]>();\n storageKey: InputSignal<string> = input<string>(this.DEFAULT_STORAGE_KEY);\n\n columns: InputSignal<number | 'auto'> = input<number | 'auto'>('auto');\n minTileWidth: InputSignal<number> = input<number>(200);\n rowHeight: InputSignal<number> = input<number>(80);\n gap: InputSignal<number> = input<number>(12);\n maxColumns: InputSignal<number> = input<number>(4);\n maxRows: InputSignal<number> = input<number>(4);\n widgetPadding: InputSignal<number> = input<number>(12);\n editMode: InputSignal<boolean> = input<boolean>(true);\n\n widgetChange: OutputEmitterRef<Widget[]> = output<Widget[]>();\n\n public store = inject(WidgetService);\n\n @ViewChild('dashboard', { static: true }) dashboard!: ElementRef;\n\n private dashboardWidth: WritableSignal<number> = signal<number>(0);\n private storageReady: WritableSignal<boolean> = signal<boolean>(false);\n private resizeObserver?: ResizeObserver;\n\n activeColumns: Signal<number> = computed(() => {\n const configuredColumns = this.columns();\n const maxColumns = Math.max(1, this.maxColumns());\n\n if (configuredColumns !== 'auto') {\n return Math.max(1, configuredColumns);\n }\n\n const width = this.dashboardWidth();\n if (width <= 0) return 1;\n\n const minTileWidth = Math.max(1, this.minTileWidth());\n const gap = Math.max(0, this.gap());\n const columns = Math.floor((width + gap) / (minTileWidth + gap));\n\n return Math.max(1, Math.min(columns, maxColumns));\n });\n\n gridStyles: Signal<Record<string, string>> = computed(() => ({\n gridTemplateColumns: `repeat(${this.activeColumns()}, minmax(0, 1fr))`,\n gridAutoRows: `${this.rowHeight()}px`,\n gap: `${this.gap()}px`,\n }));\n\n constructor() {\n effect((): void => {\n if (!this.storageReady()) return;\n const state: StoredWidgetState[] = this.store.addedWidgets().map((w) => ({\n id: w.id,\n rows: w.rows,\n columns: w.columns,\n }));\n localStorage.setItem(this.storageKey(), JSON.stringify(state));\n });\n\n effect((): void => {\n this.store.setConfig({\n columns: this.columns(),\n activeColumns: this.activeColumns(),\n minTileWidth: this.minTileWidth(),\n rowHeight: this.rowHeight(),\n gap: this.gap(),\n maxColumns: this.maxColumns(),\n maxRows: this.maxRows(),\n widgetPadding: this.widgetPadding(),\n });\n });\n\n effect((): void => {\n this.store.editMode.set(this.editMode());\n });\n\n effect((): void => {\n this.widgetChange.emit(this.store.addedWidgets());\n });\n }\n\n ngOnInit(): void {\n this.setAvailableWidgets();\n this.loadFromStorage();\n this.storageReady.set(true);\n }\n\n ngAfterViewInit(): void {\n wrapGrid(this.dashboard.nativeElement, { duration: 300 });\n this.updateDashboardWidth();\n this.resizeObserver = new ResizeObserver(() => this.updateDashboardWidth());\n this.resizeObserver.observe(this.dashboard.nativeElement);\n }\n\n ngOnDestroy(): void {\n this.resizeObserver?.disconnect();\n }\n\n private setAvailableWidgets(): void {\n const widgets = this.widgets();\n if (widgets?.length > 0) {\n this.store.setAvailableWidgets(widgets);\n }\n }\n\n private loadFromStorage(): void {\n const stored = localStorage.getItem(this.storageKey());\n if (!stored) return;\n\n try {\n const parsed: unknown[] = JSON.parse(stored);\n const widgetMap = new Map(this.store.widgets().map((w) => [w.id, w]));\n\n if (parsed.length > 0 && typeof parsed[0] === 'number') {\n const restoredWidgets = (parsed as number[])\n .map((id) => widgetMap.get(id))\n .filter((w): w is Widget => w !== undefined);\n this.store.addedWidgets.set(restoredWidgets);\n return;\n }\n\n const states = parsed as StoredWidgetState[];\n const restoredWidgets = states\n .map((state) => {\n const def = widgetMap.get(state.id);\n if (!def) return undefined;\n return { ...def, ...state };\n })\n .filter((w): w is Widget => w !== undefined);\n this.store.addedWidgets.set(restoredWidgets);\n } catch (e) {\n console.error(this.ERROR_PARSING_MESSAGE, e);\n }\n }\n\n private updateDashboardWidth(): void {\n this.dashboardWidth.set(this.dashboard.nativeElement.clientWidth);\n }\n}\n","<div class=\"dashboard-widgets-header\">\n <h2 class=\"dashboard-title\">{{title()}}</h2>\n <div class=\"dashboard-header-actions\">\n <button\n mat-icon-button\n (click)=\"store.toggleEditMode()\"\n [matTooltip]=\"(store.editMode() ? 'DASHBOARD.LOCK' : 'DASHBOARD.UNLOCK') | translate\">\n <mat-icon>{{ store.editMode() ? 'lock_open' : 'lock' }}</mat-icon>\n </button>\n @if (store.editMode()) {\n <button [matMenuTriggerFor]=\"addMenu\" mat-raised-button>\n <mat-icon>add_circle</mat-icon>\n {{ 'DASHBOARD.ADD_WIDGET' | translate }}\n </button>\n <mat-menu #addMenu=\"matMenu\">\n @for (widget of store.widgetsToAdd(); track widget.id) {\n <button mat-menu-item (click)=\"store.addWidget(widget)\">{{ widget.label | translate }}</button>\n } @empty {\n <button mat-menu-item disabled>{{ 'DASHBOARD.NO_WIDGETS' | translate }}</button>\n }\n </mat-menu>\n }\n </div>\n</div>\n<div #dashboard class=\"dashboard-widgets\" [style]=\"gridStyles()\">\n @for (w of store.addedWidgets(); track w.id) {\n <app-widget [data]=\"w\"/>\n }\n</div>","import { NgModule } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { WidgetComponent } from './components/widget/widget.component';\nimport { WidgetOptionsComponent } from './components/widget-options/widget-options.component';\nimport { DashboardWidgetsComponent } from './components/dashboard-widgets/dashboard-widgets.component';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { TranslateModule } from '@ngx-translate/core';\n\n@NgModule({\n declarations: [\n WidgetComponent,\n WidgetOptionsComponent,\n DashboardWidgetsComponent,\n ],\n imports: [\n CommonModule,\n TranslateModule.forChild(),\n MatMenuModule,\n MatButtonToggleModule,\n MatButtonModule,\n MatIconModule,\n MatTooltipModule,\n ],\n exports: [\n WidgetComponent,\n WidgetOptionsComponent,\n DashboardWidgetsComponent,\n MatMenuModule,\n MatButtonToggleModule,\n MatButtonModule,\n MatIconModule,\n MatTooltipModule,\n ],\n})\nexport class DashboardWidgetsSkeletonModule {}\n","/*\n * Public API Surface of dashboard-widgets-skeleton\n */\n\nexport * from './lib/components/widget/widget';\nexport * from './lib/components/widget/widget.component';\nexport * from './lib/components/widget-options/widget-options.component';\nexport * from './lib/components/dashboard-widgets/dashboard-widgets.component';\nexport * from './lib/models/dashboard-config';\nexport * from './lib/services/widget.service';\nexport * from './lib/dashboard-widgets-skeleton.module';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["i5","i1","i5.WidgetOptionsComponent","i5.WidgetComponent"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@fidurcode/dashboard-widgets-skeleton",
3
- "version": "2.0.10",
3
+ "version": "2.0.11",
4
4
  "license": "MIT",
5
5
  "peerDependencies": {
6
6
  "animate-css-grid": "^1.5.1",
@@ -1,5 +1,5 @@
1
1
  import * as i0 from '@angular/core';
2
- import { Type, WritableSignal, Signal, InputSignal, ModelSignal, AfterViewInit, OutputEmitterRef, ElementRef } from '@angular/core';
2
+ import { InputSignal, Type, WritableSignal, Signal, AfterViewInit, OnDestroy, ElementRef, ModelSignal, OnInit, OutputEmitterRef } from '@angular/core';
3
3
  import * as i4 from '@angular/common';
4
4
  import * as i5 from '@ngx-translate/core';
5
5
  import * as i6 from '@angular/material/menu';
@@ -7,22 +7,29 @@ import * as i7 from '@angular/material/button-toggle';
7
7
  import * as i8 from '@angular/material/button';
8
8
  import * as i9 from '@angular/material/icon';
9
9
  import * as i10 from '@angular/material/tooltip';
10
- import * as i11 from '@angular/material/divider';
11
10
 
11
+ interface DashboardWidgetContentSize {
12
+ width: number;
13
+ height: number;
14
+ rows: number;
15
+ columns: number;
16
+ }
17
+ interface DashboardWidgetContentComponent {
18
+ widgetSize: DashboardWidgetContentSize | InputSignal<DashboardWidgetContentSize>;
19
+ }
12
20
  interface Widget {
13
21
  id: number;
14
22
  label: string;
15
- content: Type<unknown>;
23
+ content: Type<DashboardWidgetContentComponent>;
16
24
  rows?: number;
17
25
  columns?: number;
18
26
  defaultRows?: number;
19
27
  defaultColumns?: number;
20
- color?: string;
21
- customLabel?: string;
22
28
  }
23
29
 
24
30
  interface DashboardConfig {
25
31
  columns: number | 'auto';
32
+ activeColumns: number;
26
33
  minTileWidth: number;
27
34
  rowHeight: number;
28
35
  gap: number;
@@ -33,7 +40,6 @@ interface DashboardConfig {
33
40
  declare const DEFAULT_DASHBOARD_CONFIG: DashboardConfig;
34
41
 
35
42
  declare class WidgetService {
36
- private readonly DEFAULT_WIDGET_SIZE;
37
43
  widgets: WritableSignal<Widget[]>;
38
44
  addedWidgets: WritableSignal<Widget[]>;
39
45
  config: WritableSignal<DashboardConfig>;
@@ -43,58 +49,53 @@ declare class WidgetService {
43
49
  setConfig(config: Partial<DashboardConfig>): void;
44
50
  addWidget(widget: Widget): void;
45
51
  updateWidget(id: number, widget: Partial<Widget>): void;
46
- renameWidget(id: number, customLabel: string): void;
47
- setWidgetColor(id: number, color: string | undefined): void;
48
52
  toggleEditMode(): void;
49
53
  moveWidgetToLeft(id: number): void;
50
54
  moveWidgetToRight(id: number): void;
51
55
  deleteWidget(id: number): void;
52
- private clampScale;
56
+ private normalizeWidgetSize;
57
+ private normalizeColumns;
58
+ private normalizeRows;
59
+ private clamp;
53
60
  static ɵfac: i0.ɵɵFactoryDeclaration<WidgetService, never>;
54
61
  static ɵprov: i0.ɵɵInjectableDeclaration<WidgetService>;
55
62
  }
56
63
 
57
- declare class WidgetComponent {
58
- private readonly DEFAULT_WIDGET_SIZE;
64
+ declare class WidgetComponent implements AfterViewInit, OnDestroy {
59
65
  data: InputSignal<Widget>;
60
66
  store: WidgetService;
67
+ contentHost?: ElementRef<HTMLElement>;
61
68
  showWidgetOptions: WritableSignal<boolean>;
62
- isRenaming: WritableSignal<boolean>;
69
+ contentSize: WritableSignal<DashboardWidgetContentSize>;
70
+ private resizeObserver?;
63
71
  gridArea: Signal<string>;
64
72
  widgetPadding: Signal<string>;
65
- startRename(): void;
66
- finishRename(event: Event): void;
67
- private clampScale;
73
+ contentInputs: Signal<{
74
+ widgetSize: DashboardWidgetContentSize;
75
+ }>;
76
+ ngAfterViewInit(): void;
77
+ ngOnDestroy(): void;
78
+ private updateContentSize;
79
+ private normalizeSpan;
68
80
  static ɵfac: i0.ɵɵFactoryDeclaration<WidgetComponent, never>;
69
81
  static ɵcmp: i0.ɵɵComponentDeclaration<WidgetComponent, "app-widget", never, { "data": { "alias": "data"; "required": true; "isSignal": true; }; }, {}, never, never, false, never>;
70
82
  }
71
83
 
72
84
  declare class WidgetOptionsComponent {
73
- private readonly DEFAULT_WIDGET_SIZE;
74
- private readonly MAX_WIDGET_SCALE;
75
85
  widget: InputSignal<Widget>;
76
86
  showOptions: ModelSignal<boolean>;
77
87
  store: WidgetService;
78
88
  colOptions: Signal<number[]>;
79
89
  rowOptions: Signal<number[]>;
80
- selectedColumns: Signal<number>;
81
- selectedRows: Signal<number>;
82
- canMoveLeft: Signal<boolean>;
83
- canMoveRight: Signal<boolean>;
84
- onRenameBlur(event: Event): void;
85
- private clampOptionCount;
86
- private widgetIndex;
90
+ canMoveLeft(): boolean;
91
+ canMoveRight(): boolean;
87
92
  static ɵfac: i0.ɵɵFactoryDeclaration<WidgetOptionsComponent, never>;
88
93
  static ɵcmp: i0.ɵɵComponentDeclaration<WidgetOptionsComponent, "app-widget-options", never, { "widget": { "alias": "widget"; "required": true; "isSignal": true; }; "showOptions": { "alias": "showOptions"; "required": false; "isSignal": true; }; }, { "showOptions": "showOptionsChange"; }, never, never, false, never>;
89
94
  }
90
95
 
91
- declare class DashboardWidgetsComponent implements AfterViewInit {
96
+ declare class DashboardWidgetsComponent implements OnInit, AfterViewInit, OnDestroy {
92
97
  private readonly DEFAULT_STORAGE_KEY;
93
98
  private readonly ERROR_PARSING_MESSAGE;
94
- private readonly MAX_WIDGET_SCALE;
95
- private readonly platformId;
96
- private readonly isBrowser;
97
- private hasLoadedStoredWidgets;
98
99
  title: InputSignal<string>;
99
100
  widgets: InputSignal<Widget[]>;
100
101
  storageKey: InputSignal<string>;
@@ -109,24 +110,28 @@ declare class DashboardWidgetsComponent implements AfterViewInit {
109
110
  widgetChange: OutputEmitterRef<Widget[]>;
110
111
  store: WidgetService;
111
112
  dashboard: ElementRef;
113
+ private dashboardWidth;
114
+ private storageReady;
115
+ private resizeObserver?;
116
+ activeColumns: Signal<number>;
112
117
  gridStyles: Signal<Record<string, string>>;
113
118
  constructor();
119
+ ngOnInit(): void;
114
120
  ngAfterViewInit(): void;
121
+ ngOnDestroy(): void;
115
122
  private setAvailableWidgets;
116
- private getGridTemplateColumns;
117
- private clampScale;
118
- private clampPositiveNumber;
119
123
  private loadFromStorage;
124
+ private updateDashboardWidth;
120
125
  static ɵfac: i0.ɵɵFactoryDeclaration<DashboardWidgetsComponent, never>;
121
126
  static ɵcmp: i0.ɵɵComponentDeclaration<DashboardWidgetsComponent, "fidurcode-dashboard-widgets", never, { "title": { "alias": "title"; "required": true; "isSignal": true; }; "widgets": { "alias": "widgets"; "required": true; "isSignal": true; }; "storageKey": { "alias": "storageKey"; "required": false; "isSignal": true; }; "columns": { "alias": "columns"; "required": false; "isSignal": true; }; "minTileWidth": { "alias": "minTileWidth"; "required": false; "isSignal": true; }; "rowHeight": { "alias": "rowHeight"; "required": false; "isSignal": true; }; "gap": { "alias": "gap"; "required": false; "isSignal": true; }; "maxColumns": { "alias": "maxColumns"; "required": false; "isSignal": true; }; "maxRows": { "alias": "maxRows"; "required": false; "isSignal": true; }; "widgetPadding": { "alias": "widgetPadding"; "required": false; "isSignal": true; }; "editMode": { "alias": "editMode"; "required": false; "isSignal": true; }; }, { "widgetChange": "widgetChange"; }, never, never, false, never>;
122
127
  }
123
128
 
124
129
  declare class DashboardWidgetsSkeletonModule {
125
130
  static ɵfac: i0.ɵɵFactoryDeclaration<DashboardWidgetsSkeletonModule, never>;
126
- static ɵmod: i0.ɵɵNgModuleDeclaration<DashboardWidgetsSkeletonModule, [typeof WidgetComponent, typeof WidgetOptionsComponent, typeof DashboardWidgetsComponent], [typeof i4.CommonModule, typeof i5.TranslateModule, typeof i6.MatMenuModule, typeof i7.MatButtonToggleModule, typeof i8.MatButtonModule, typeof i9.MatIconModule, typeof i10.MatTooltipModule, typeof i11.MatDividerModule], [typeof WidgetComponent, typeof WidgetOptionsComponent, typeof DashboardWidgetsComponent, typeof i6.MatMenuModule, typeof i7.MatButtonToggleModule, typeof i8.MatButtonModule, typeof i9.MatIconModule, typeof i10.MatTooltipModule, typeof i11.MatDividerModule]>;
131
+ static ɵmod: i0.ɵɵNgModuleDeclaration<DashboardWidgetsSkeletonModule, [typeof WidgetComponent, typeof WidgetOptionsComponent, typeof DashboardWidgetsComponent], [typeof i4.CommonModule, typeof i5.TranslateModule, typeof i6.MatMenuModule, typeof i7.MatButtonToggleModule, typeof i8.MatButtonModule, typeof i9.MatIconModule, typeof i10.MatTooltipModule], [typeof WidgetComponent, typeof WidgetOptionsComponent, typeof DashboardWidgetsComponent, typeof i6.MatMenuModule, typeof i7.MatButtonToggleModule, typeof i8.MatButtonModule, typeof i9.MatIconModule, typeof i10.MatTooltipModule]>;
127
132
  static ɵinj: i0.ɵɵInjectorDeclaration<DashboardWidgetsSkeletonModule>;
128
133
  }
129
134
 
130
135
  export { DEFAULT_DASHBOARD_CONFIG, DashboardWidgetsComponent, DashboardWidgetsSkeletonModule, WidgetComponent, WidgetOptionsComponent, WidgetService };
131
- export type { DashboardConfig, Widget };
136
+ export type { DashboardConfig, DashboardWidgetContentComponent, DashboardWidgetContentSize, Widget };
132
137
  //# sourceMappingURL=fidurcode-dashboard-widgets-skeleton.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"fidurcode-dashboard-widgets-skeleton.d.ts","sources":["../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.ts","../../../projects/dashboard-widgets-skeleton/src/lib/models/dashboard-config.ts","../../../projects/dashboard-widgets-skeleton/src/lib/services/widget.service.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget-options/widget-options.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/dashboard-widgets/dashboard-widgets.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/dashboard-widgets-skeleton.module.ts"],"mappings":";;;;;;;;;;;UAEiB,MAAM;;;AAGrB,aAAS,IAAI;;;;;;;AAOd;;UCZgB,eAAe;AAC9B;;;;;;;AAOD;AAED,cAAa,wBAAwB,EAAE,eAQtC;;ACRD,cACa,aAAa;AACxB;AAEO,aAAS,cAAc,CAAC,MAAM;AACrC,kBAAc,cAAc,CAAC,MAAM;AACnC,YAAQ,cAAc,CAAC,eAAe;AACtC,cAAU,cAAc;AAExB,kBAAc,MAAM,CAAC,MAAM;AAK3B,iCAA6B,MAAM;sBAIjB,OAAO,CAAC,eAAe;AAIzC,sBAAkB,MAAM;AAgBxB,qCAAiC,OAAO,CAAC,MAAM;;;AAyB/C;AAIA;AAQA;AAQA;AAIA;yCAtFW,aAAa;6CAAb,aAAa;AA2FzB;;ACzFD,cAUa,eAAe;AAC1B;AAEA,UAAM,WAAW,CAAC,MAAM;AAExB,WAAK,aAAA;AAEL,uBAAmB,cAAc;AACjC,gBAAY,cAAc;AAE1B,cAAU,MAAM;AAOhB,mBAAe,MAAM;AAErB;AAKA,wBAAoB,KAAK;AAMzB;yCA9BW,eAAe;2CAAf,eAAe;AAiC3B;;AC3CD,cAMa,sBAAsB;AACjC;AACA;AAEA,YAAQ,WAAW,CAAC,MAAM;AAC1B,iBAAa,WAAW;AAExB,WAAK,aAAA;AAEL,gBAAY,MAAM;AAIlB,gBAAY,MAAM;AAIlB,qBAAiB,MAAM;AAGvB,kBAAc,MAAM;AAGpB,iBAAa,MAAM;AACnB,kBAAc,MAAM;AAKpB,wBAAoB,KAAK;AAKzB;AAIA;yCAtCW,sBAAsB;2CAAtB,sBAAsB;AAyClC;;ACpCD,cAOa,yBAA0B,YAAW,aAAa;AAC7D;AACA;AACA;AACA;AACA;;AAGA,WAAO,WAAW;AAClB,aAAS,WAAW,CAAC,MAAM;AAC3B,gBAAY,WAAW;AAEvB,aAAS,WAAW;AACpB,kBAAc,WAAW;AACzB,eAAW,WAAW;AACtB,SAAK,WAAW;AAChB,gBAAY,WAAW;AACvB,aAAS,WAAW;AACpB,mBAAe,WAAW;AAC1B,cAAU,WAAW;AAErB,kBAAc,gBAAgB,CAAC,MAAM;AAE9B,WAAK,aAAA;eAE0C,UAAU;gBAEpD,MAAM,CAAC,MAAM;;AAoDzB;AAMA;AAIA;AASA;AAIA;AAIA;yCA1GW,yBAAyB;2CAAzB,yBAAyB;AA6IrC;;AC/JD,cA4Ba,8BAA8B;yCAA9B,8BAA8B;0CAA9B,8BAA8B,UAAAA,eAAA,SAAAC,sBAAA,SAAAC,yBAAA,WAAA,EAAA,CAAA,YAAA,SAAA,EAAA,CAAA,eAAA,SAAA,EAAA,CAAA,aAAA,SAAA,EAAA,CAAA,qBAAA,SAAA,EAAA,CAAA,eAAA,SAAA,EAAA,CAAA,aAAA,SAAA,GAAA,CAAA,gBAAA,SAAA,GAAA,CAAA,gBAAA,WAAAF,eAAA,SAAAC,sBAAA,SAAAC,yBAAA,SAAA,EAAA,CAAA,aAAA,SAAA,EAAA,CAAA,qBAAA,SAAA,EAAA,CAAA,eAAA,SAAA,EAAA,CAAA,aAAA,SAAA,GAAA,CAAA,gBAAA,SAAA,GAAA,CAAA,gBAAA;0CAA9B,8BAA8B;AAAG;;;;","names":["i1.WidgetComponent","i2.WidgetOptionsComponent","i3.DashboardWidgetsComponent"]}
1
+ {"version":3,"file":"fidurcode-dashboard-widgets-skeleton.d.ts","sources":["../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.ts","../../../projects/dashboard-widgets-skeleton/src/lib/models/dashboard-config.ts","../../../projects/dashboard-widgets-skeleton/src/lib/services/widget.service.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget/widget.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/widget-options/widget-options.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/components/dashboard-widgets/dashboard-widgets.component.ts","../../../projects/dashboard-widgets-skeleton/src/lib/dashboard-widgets-skeleton.module.ts"],"mappings":";;;;;;;;;;UAEiB,0BAA0B;;;;;AAK1C;UAEgB,+BAA+B;AAC9C,gBAAY,0BAA0B,GAAG,WAAW,CAAC,0BAA0B;AAChF;UAEgB,MAAM;;;AAGrB,aAAS,IAAI,CAAC,+BAA+B;;;;;AAK9C;;UCrBgB,eAAe;AAC9B;;;;;;;;AAQD;AAED,cAAa,wBAAwB,EAAE,eAStC;;ACVD,cACa,aAAa;AACjB,aAAS,cAAc,CAAC,MAAM;AACrC,kBAAc,cAAc,CAAC,MAAM;AACnC,YAAQ,cAAc,CAAC,eAAe;AACtC,cAAU,cAAc;AAExB,kBAAc,MAAM,CAAC,MAAM;AAK3B,iCAA6B,MAAM;sBAIjB,OAAO,CAAC,eAAe;AAIzC,sBAAkB,MAAM;AAWxB,qCAAiC,OAAO,CAAC,MAAM;AAS/C;AAIA;AAQA;AAQA;AAIA;AAQA;AAIA;AAIA;yCA/EW,aAAa;6CAAb,aAAa;AAkFzB;;AC5ED,cAUa,eAAgB,YAAW,aAAa,EAAE,SAAS;AAC9D,UAAM,WAAW,CAAC,MAAM;AAExB,WAAK,aAAA;AAEqB,kBAAc,UAAU,CAAC,WAAW;AAE9D,uBAAmB,cAAc;AACjC,iBAAa,cAAc,CAAC,0BAA0B;;AAStD,cAAU,MAAM;AAUhB,mBAAe,MAAM;mBAEN,MAAM;oBAAe,0BAA0B;AAAE;AAIhE;AAQA;AAIA;AAeA;yCA5DW,eAAe;2CAAf,eAAe;AA+D3B;;AC7ED,cAMa,sBAAsB;AACjC,YAAQ,WAAW,CAAC,MAAM;AAC1B,iBAAa,WAAW;AAExB,WAAK,aAAA;AAEL,gBAAY,MAAM;AAIlB,gBAAY,MAAM;AAIlB;AAIA;yCAlBW,sBAAsB;2CAAtB,sBAAsB;AAsBlC;;ACjBD,cAOa,yBAA0B,YAAW,MAAM,EAAE,aAAa,EAAE,SAAS;AAChF;AACA;AAEA,WAAO,WAAW;AAClB,aAAS,WAAW,CAAC,MAAM;AAC3B,gBAAY,WAAW;AAEvB,aAAS,WAAW;AACpB,kBAAc,WAAW;AACzB,eAAW,WAAW;AACtB,SAAK,WAAW;AAChB,gBAAY,WAAW;AACvB,aAAS,WAAW;AACpB,mBAAe,WAAW;AAC1B,cAAU,WAAW;AAErB,kBAAc,gBAAgB,CAAC,MAAM;AAE9B,WAAK,aAAA;eAE0C,UAAU;;;;AAMhE,mBAAe,MAAM;gBAkBT,MAAM,CAAC,MAAM;;AAuCzB;AAMA;AAOA;AAIA;AAOA;AA8BA;yCA1IW,yBAAyB;2CAAzB,yBAAyB;AA6IrC;;AChKD,cA0Ba,8BAA8B;yCAA9B,8BAA8B;0CAA9B,8BAA8B,UAAAA,eAAA,SAAAC,sBAAA,SAAAC,yBAAA,WAAA,EAAA,CAAA,YAAA,SAAA,EAAA,CAAA,eAAA,SAAA,EAAA,CAAA,aAAA,SAAA,EAAA,CAAA,qBAAA,SAAA,EAAA,CAAA,eAAA,SAAA,EAAA,CAAA,aAAA,SAAA,GAAA,CAAA,gBAAA,WAAAF,eAAA,SAAAC,sBAAA,SAAAC,yBAAA,SAAA,EAAA,CAAA,aAAA,SAAA,EAAA,CAAA,qBAAA,SAAA,EAAA,CAAA,eAAA,SAAA,EAAA,CAAA,aAAA,SAAA,GAAA,CAAA,gBAAA;0CAA9B,8BAA8B;AAAG;;;;","names":["i1.WidgetComponent","i2.WidgetOptionsComponent","i3.DashboardWidgetsComponent"]}