@ogidor/dashboard 1.0.3 → 1.0.4

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.
Files changed (39) hide show
  1. package/README.md +283 -48
  2. package/app/app.module.d.ts +20 -0
  3. package/app/custom-grid.component.d.ts +93 -0
  4. package/app/dashboard-state.service.d.ts +70 -0
  5. package/app/dashboard.component.d.ts +48 -0
  6. package/app/models.d.ts +53 -0
  7. package/app/widget-renderer.component.d.ts +13 -0
  8. package/esm2020/app/app.module.mjs +57 -0
  9. package/esm2020/app/custom-grid.component.mjs +509 -0
  10. package/esm2020/app/dashboard-state.service.mjs +288 -0
  11. package/esm2020/app/dashboard.component.mjs +299 -0
  12. package/esm2020/app/models.mjs +2 -0
  13. package/esm2020/app/widget-renderer.component.mjs +83 -0
  14. package/esm2020/public-api.mjs +10 -0
  15. package/fesm2015/ogidor-dashboard.mjs +1233 -0
  16. package/fesm2015/ogidor-dashboard.mjs.map +1 -0
  17. package/fesm2020/ogidor-dashboard.mjs +1229 -0
  18. package/fesm2020/ogidor-dashboard.mjs.map +1 -0
  19. package/package.json +29 -43
  20. package/{dist-lib/public-api.d.ts → public-api.d.ts} +3 -2
  21. package/dist-lib/README.md +0 -103
  22. package/dist-lib/app/app.module.d.ts +0 -22
  23. package/dist-lib/app/dashboard-state.service.d.ts +0 -49
  24. package/dist-lib/app/dashboard.component.d.ts +0 -80
  25. package/dist-lib/app/models.d.ts +0 -73
  26. package/dist-lib/app/widget-renderer.component.d.ts +0 -40
  27. package/dist-lib/esm2020/app/app.module.mjs +0 -65
  28. package/dist-lib/esm2020/app/dashboard-state.service.mjs +0 -218
  29. package/dist-lib/esm2020/app/dashboard.component.mjs +0 -703
  30. package/dist-lib/esm2020/app/models.mjs +0 -2
  31. package/dist-lib/esm2020/app/widget-renderer.component.mjs +0 -163
  32. package/dist-lib/esm2020/public-api.mjs +0 -9
  33. package/dist-lib/fesm2015/ogidor-dashboard.mjs +0 -1154
  34. package/dist-lib/fesm2015/ogidor-dashboard.mjs.map +0 -1
  35. package/dist-lib/fesm2020/ogidor-dashboard.mjs +0 -1145
  36. package/dist-lib/fesm2020/ogidor-dashboard.mjs.map +0 -1
  37. package/dist-lib/package.json +0 -51
  38. /package/{dist-lib/esm2020 → esm2020}/ogidor-dashboard.mjs +0 -0
  39. /package/{dist-lib/index.d.ts → index.d.ts} +0 -0
@@ -0,0 +1,299 @@
1
+ import { Component, Input, Output, EventEmitter, ContentChild, TemplateRef } from '@angular/core';
2
+ import { Subscription } from 'rxjs';
3
+ import { GridCellDirective } from './custom-grid.component';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "./dashboard-state.service";
6
+ import * as i2 from "@angular/common";
7
+ import * as i3 from "./widget-renderer.component";
8
+ import * as i4 from "./custom-grid.component";
9
+ const DEFAULT_THEME = {
10
+ backgroundColor: '#000000',
11
+ panelColor: '#1c1c1e',
12
+ widgetCardColor: '#2c2c2e',
13
+ foreColor: '#8e8e93',
14
+ accentColor: '#0a84ff',
15
+ dangerColor: '#ff453a',
16
+ fontFamily: '-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif',
17
+ };
18
+ export class DashboardComponent {
19
+ constructor(stateService) {
20
+ this.stateService = stateService;
21
+ /**
22
+ * Emits when the user clicks "Add Widget".
23
+ * The consumer should open their own dialog and call `stateService.addWidget(...)`.
24
+ */
25
+ this.addWidgetRequested = new EventEmitter();
26
+ /**
27
+ * Emits the Widget when the user clicks the edit (pen) icon on a card.
28
+ * The consumer should open their own edit UI and call `stateService.updateWidget(...)`.
29
+ */
30
+ this.editWidgetRequested = new EventEmitter();
31
+ this.resolvedTheme = { ...DEFAULT_THEME };
32
+ this.wrapperStyles = {};
33
+ this.pages = [];
34
+ this.activePageId = '';
35
+ this.isPoppedOut = false;
36
+ this.subs = new Subscription();
37
+ }
38
+ ngOnChanges(changes) {
39
+ if (changes['theme'] || changes['initialLayout'])
40
+ this.applyTheme();
41
+ }
42
+ ngOnInit() {
43
+ this.applyTheme();
44
+ if (this.initialLayout) {
45
+ try {
46
+ this.stateService.loadLayout(JSON.parse(this.initialLayout));
47
+ }
48
+ catch (e) {
49
+ console.error('[Dashboard] Failed to parse initialLayout', e);
50
+ }
51
+ }
52
+ const hash = window.location.hash?.replace('#', '').trim();
53
+ if (hash)
54
+ this.isPoppedOut = true;
55
+ this.subs.add(this.stateService.pages$.subscribe(pages => {
56
+ this.pages = pages;
57
+ if (this.isPoppedOut && hash) {
58
+ const target = pages.find(p => p.id === hash);
59
+ if (target)
60
+ this.stateService.setActivePage(hash);
61
+ }
62
+ this.updateActivePage();
63
+ }));
64
+ this.subs.add(this.stateService.activePageId$.subscribe(id => {
65
+ this.activePageId = id;
66
+ this.updateActivePage();
67
+ }));
68
+ }
69
+ ngOnDestroy() { this.subs.unsubscribe(); }
70
+ onItemChanged(widget) {
71
+ this.stateService.updateWidgetPosition(this.activePageId, widget.id, widget.x, widget.y, widget.cols, widget.rows);
72
+ }
73
+ onRemoveWidget(widgetId) { this.stateService.removeWidget(widgetId); }
74
+ onSelectPage(id) { this.stateService.setActivePage(id); }
75
+ onAddPage() {
76
+ const name = prompt('Workspace name:', `Workspace ${this.pages.length + 1}`);
77
+ if (name)
78
+ this.stateService.addPage(name);
79
+ }
80
+ onRemovePage(event, id) {
81
+ event.stopPropagation();
82
+ if (confirm('Remove this workspace?'))
83
+ this.stateService.removePage(id);
84
+ }
85
+ onPopOut(event, pageId) {
86
+ event.stopPropagation();
87
+ this.stateService.popOutPage(pageId);
88
+ }
89
+ serializeLayout() { return this.stateService.serializeLayout(); }
90
+ applyTheme() {
91
+ this.resolvedTheme = { ...DEFAULT_THEME, ...(this.theme ?? {}) };
92
+ this.wrapperStyles = {
93
+ '--dash-bg': this.resolvedTheme.backgroundColor,
94
+ '--dash-panel-bg': this.resolvedTheme.panelColor,
95
+ '--dash-card-bg': this.resolvedTheme.widgetCardColor,
96
+ '--dash-fore-color': this.resolvedTheme.foreColor,
97
+ '--dash-accent-color': this.resolvedTheme.accentColor,
98
+ '--dash-danger-color': this.resolvedTheme.dangerColor,
99
+ '--dash-font-family': this.resolvedTheme.fontFamily,
100
+ };
101
+ }
102
+ updateActivePage() {
103
+ this.activePage = this.pages.find(p => p.id === this.activePageId);
104
+ }
105
+ }
106
+ DashboardComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardComponent, deps: [{ token: i1.DashboardStateService }], target: i0.ɵɵFactoryTarget.Component });
107
+ DashboardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: DashboardComponent, selector: "app-dashboard", inputs: { initialLayout: "initialLayout", theme: "theme" }, outputs: { addWidgetRequested: "addWidgetRequested", editWidgetRequested: "editWidgetRequested" }, queries: [{ propertyName: "cellTemplate", first: true, predicate: GridCellDirective, descendants: true, read: TemplateRef }], usesOnChanges: true, ngImport: i0, template: `
108
+ <!-- ══════════════════════════════════════════════════════════
109
+ POPPED-OUT MODE
110
+ ══════════════════════════════════════════════════════════ -->
111
+ <ng-container *ngIf="isPoppedOut; else normalMode">
112
+ <div class="popout-wrapper" [ngStyle]="wrapperStyles">
113
+ <header class="popout-header">
114
+ <span class="popout-title">{{ activePage?.name }}</span>
115
+ </header>
116
+ <div class="grid-container">
117
+ <app-grid [widgets]="activePage?.widgets || []" (itemChanged)="onItemChanged($event)">
118
+ <ng-template gridCell let-widget="widget">
119
+ <app-widget-renderer
120
+ [widget]="widget"
121
+ [theme]="resolvedTheme"
122
+ (editRequested)="editWidgetRequested.emit($event)"
123
+ (removeRequested)="onRemoveWidget($event)"
124
+ >
125
+ <ng-container
126
+ *ngIf="cellTemplate"
127
+ [ngTemplateOutlet]="cellTemplate"
128
+ [ngTemplateOutletContext]="{ widget: widget }">
129
+ </ng-container>
130
+ </app-widget-renderer>
131
+ </ng-template>
132
+ </app-grid>
133
+ </div>
134
+ </div>
135
+ </ng-container>
136
+
137
+ <!-- ══════════════════════════════════════════════════════════
138
+ NORMAL MODE
139
+ ══════════════════════════════════════════════════════════ -->
140
+ <ng-template #normalMode>
141
+ <div class="dashboard-wrapper" [ngStyle]="wrapperStyles">
142
+ <main class="main-content">
143
+
144
+ <header class="dashboard-header">
145
+ <div class="tabs-container">
146
+ <div
147
+ *ngFor="let page of pages"
148
+ class="tab"
149
+ [class.active]="page.id === activePageId"
150
+ (click)="onSelectPage(page.id)"
151
+ >
152
+ <span class="tab-name">{{ page.name }}</span>
153
+ <button class="tab-action tab-popout" (click)="onPopOut($event, page.id)" title="Open in new window">
154
+ <i class="la la-external-link-alt"></i>
155
+ </button>
156
+ <button class="tab-action tab-close" *ngIf="pages.length > 1" (click)="onRemovePage($event, page.id)" title="Close">
157
+ <i class="la la-times"></i>
158
+ </button>
159
+ </div>
160
+ <button class="btn-add-page" (click)="onAddPage()" title="New workspace">
161
+ <i class="la la-plus"></i>
162
+ </button>
163
+ </div>
164
+
165
+ <!-- Add Widget button — consumer decides what happens -->
166
+ <button class="btn-add-widget" (click)="addWidgetRequested.emit()" title="Add widget">
167
+ <i class="la la-plus"></i>
168
+ <span>Add Widget</span>
169
+ </button>
170
+ </header>
171
+
172
+ <div class="grid-container">
173
+ <app-grid [widgets]="activePage?.widgets || []" (itemChanged)="onItemChanged($event)">
174
+ <ng-template gridCell let-widget="widget">
175
+ <app-widget-renderer
176
+ [widget]="widget"
177
+ [theme]="resolvedTheme"
178
+ (editRequested)="editWidgetRequested.emit($event)"
179
+ (removeRequested)="onRemoveWidget($event)"
180
+ >
181
+ <!-- Stamp consumer's cell template (if provided) inside the card body -->
182
+ <ng-container
183
+ *ngIf="cellTemplate"
184
+ [ngTemplateOutlet]="cellTemplate"
185
+ [ngTemplateOutletContext]="{ widget: widget }">
186
+ </ng-container>
187
+ </app-widget-renderer>
188
+ </ng-template>
189
+ </app-grid>
190
+ </div>
191
+
192
+ </main>
193
+ </div>
194
+ </ng-template>
195
+ `, isInline: true, styles: [":host{--dash-bg: #000000;--dash-panel-bg: #1c1c1e;--dash-card-bg: #2c2c2e;--dash-fore-color:#8e8e93;--dash-accent-color: #0a84ff;--dash-danger-color: #ff453a;--dash-font-family: -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Helvetica, Arial, sans-serif}.dashboard-wrapper{display:flex;height:100vh;width:100vw;overflow:hidden;padding:16px;box-sizing:border-box;background:var(--dash-bg);color:var(--dash-fore-color);font-family:var(--dash-font-family)}.main-content{flex:1;display:flex;flex-direction:column;overflow:hidden;border-radius:40px;padding:20px;background:var(--dash-panel-bg);box-shadow:0 10px 30px #00000080;border:1px solid rgba(255,255,255,.05)}.popout-wrapper{display:flex;flex-direction:column;height:100vh;width:100vw;overflow:hidden;padding:16px;box-sizing:border-box;background:var(--dash-bg);color:var(--dash-fore-color);font-family:var(--dash-font-family)}.popout-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:16px;padding:10px 18px;background:var(--dash-panel-bg);border-radius:20px;border:1px solid rgba(255,255,255,.06);flex-shrink:0}.popout-title{font-size:16px;font-weight:700;color:#fff}.popout-wrapper .grid-container{flex:1;overflow:auto;border-radius:24px;min-height:0;background:var(--dash-panel-bg);padding:16px;border:1px solid rgba(255,255,255,.05)}.dashboard-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:20px;gap:12px}.tabs-container{display:flex;align-items:center;gap:8px;background:rgba(44,44,46,.6);backdrop-filter:blur(10px);border-radius:25px;padding:6px}.tab{display:flex;align-items:center;gap:4px;padding:8px 14px 8px 18px;border-radius:20px;cursor:pointer;font-size:14px;font-weight:500;color:var(--dash-fore-color);transition:all .3s cubic-bezier(.25,.8,.25,1)}.tab.active{background:#3a3a3c;color:#fff;box-shadow:0 2px 10px #0003}.tab:hover:not(.active){color:#e5e5ea;background:rgba(255,255,255,.05)}.tab-name{flex:1;white-space:nowrap}.tab-action{background:transparent;border:none;color:var(--dash-fore-color);padding:2px;font-size:11px;cursor:pointer;opacity:0;border-radius:50%;width:20px;height:20px;display:flex;align-items:center;justify-content:center;transition:all .2s;flex-shrink:0}.tab:hover .tab-action{opacity:1}.tab-popout:hover{color:var(--dash-accent-color);background:rgba(10,132,255,.18)}.tab-close:hover{color:var(--dash-danger-color);background:rgba(255,69,58,.2)}.btn-add-page{background:transparent;border:none;color:var(--dash-fore-color);padding:8px 16px;cursor:pointer;border-radius:20px;transition:all .2s}.btn-add-page:hover{color:var(--dash-accent-color);background:rgba(10,132,255,.1)}.btn-add-widget{display:flex;align-items:center;gap:6px;background:var(--dash-accent-color);border:none;color:#fff;font-size:14px;font-weight:600;padding:10px 20px;border-radius:22px;cursor:pointer;white-space:nowrap;transition:opacity .2s,transform .15s;box-shadow:0 4px 14px #0a84ff66}.btn-add-widget:hover{opacity:.9;transform:translateY(-1px)}.btn-add-widget:active{transform:translateY(0)}.main-content .grid-container{flex:1;overflow:auto;border-radius:24px;min-height:0}app-grid{background:transparent;display:block;height:100%;width:100%}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i3.WidgetRendererComponent, selector: "app-widget-renderer", inputs: ["widget", "theme"], outputs: ["editRequested", "removeRequested"] }, { kind: "component", type: i4.CustomGridComponent, selector: "app-grid", inputs: ["widgets", "columns", "gap", "rowHeight", "minItemCols", "minItemRows"], outputs: ["itemChanged", "layoutChanged"] }, { kind: "directive", type: i4.GridCellDirective, selector: "[gridCell]" }] });
196
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardComponent, decorators: [{
197
+ type: Component,
198
+ args: [{ selector: 'app-dashboard', template: `
199
+ <!-- ══════════════════════════════════════════════════════════
200
+ POPPED-OUT MODE
201
+ ══════════════════════════════════════════════════════════ -->
202
+ <ng-container *ngIf="isPoppedOut; else normalMode">
203
+ <div class="popout-wrapper" [ngStyle]="wrapperStyles">
204
+ <header class="popout-header">
205
+ <span class="popout-title">{{ activePage?.name }}</span>
206
+ </header>
207
+ <div class="grid-container">
208
+ <app-grid [widgets]="activePage?.widgets || []" (itemChanged)="onItemChanged($event)">
209
+ <ng-template gridCell let-widget="widget">
210
+ <app-widget-renderer
211
+ [widget]="widget"
212
+ [theme]="resolvedTheme"
213
+ (editRequested)="editWidgetRequested.emit($event)"
214
+ (removeRequested)="onRemoveWidget($event)"
215
+ >
216
+ <ng-container
217
+ *ngIf="cellTemplate"
218
+ [ngTemplateOutlet]="cellTemplate"
219
+ [ngTemplateOutletContext]="{ widget: widget }">
220
+ </ng-container>
221
+ </app-widget-renderer>
222
+ </ng-template>
223
+ </app-grid>
224
+ </div>
225
+ </div>
226
+ </ng-container>
227
+
228
+ <!-- ══════════════════════════════════════════════════════════
229
+ NORMAL MODE
230
+ ══════════════════════════════════════════════════════════ -->
231
+ <ng-template #normalMode>
232
+ <div class="dashboard-wrapper" [ngStyle]="wrapperStyles">
233
+ <main class="main-content">
234
+
235
+ <header class="dashboard-header">
236
+ <div class="tabs-container">
237
+ <div
238
+ *ngFor="let page of pages"
239
+ class="tab"
240
+ [class.active]="page.id === activePageId"
241
+ (click)="onSelectPage(page.id)"
242
+ >
243
+ <span class="tab-name">{{ page.name }}</span>
244
+ <button class="tab-action tab-popout" (click)="onPopOut($event, page.id)" title="Open in new window">
245
+ <i class="la la-external-link-alt"></i>
246
+ </button>
247
+ <button class="tab-action tab-close" *ngIf="pages.length > 1" (click)="onRemovePage($event, page.id)" title="Close">
248
+ <i class="la la-times"></i>
249
+ </button>
250
+ </div>
251
+ <button class="btn-add-page" (click)="onAddPage()" title="New workspace">
252
+ <i class="la la-plus"></i>
253
+ </button>
254
+ </div>
255
+
256
+ <!-- Add Widget button — consumer decides what happens -->
257
+ <button class="btn-add-widget" (click)="addWidgetRequested.emit()" title="Add widget">
258
+ <i class="la la-plus"></i>
259
+ <span>Add Widget</span>
260
+ </button>
261
+ </header>
262
+
263
+ <div class="grid-container">
264
+ <app-grid [widgets]="activePage?.widgets || []" (itemChanged)="onItemChanged($event)">
265
+ <ng-template gridCell let-widget="widget">
266
+ <app-widget-renderer
267
+ [widget]="widget"
268
+ [theme]="resolvedTheme"
269
+ (editRequested)="editWidgetRequested.emit($event)"
270
+ (removeRequested)="onRemoveWidget($event)"
271
+ >
272
+ <!-- Stamp consumer's cell template (if provided) inside the card body -->
273
+ <ng-container
274
+ *ngIf="cellTemplate"
275
+ [ngTemplateOutlet]="cellTemplate"
276
+ [ngTemplateOutletContext]="{ widget: widget }">
277
+ </ng-container>
278
+ </app-widget-renderer>
279
+ </ng-template>
280
+ </app-grid>
281
+ </div>
282
+
283
+ </main>
284
+ </div>
285
+ </ng-template>
286
+ `, styles: [":host{--dash-bg: #000000;--dash-panel-bg: #1c1c1e;--dash-card-bg: #2c2c2e;--dash-fore-color:#8e8e93;--dash-accent-color: #0a84ff;--dash-danger-color: #ff453a;--dash-font-family: -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Helvetica, Arial, sans-serif}.dashboard-wrapper{display:flex;height:100vh;width:100vw;overflow:hidden;padding:16px;box-sizing:border-box;background:var(--dash-bg);color:var(--dash-fore-color);font-family:var(--dash-font-family)}.main-content{flex:1;display:flex;flex-direction:column;overflow:hidden;border-radius:40px;padding:20px;background:var(--dash-panel-bg);box-shadow:0 10px 30px #00000080;border:1px solid rgba(255,255,255,.05)}.popout-wrapper{display:flex;flex-direction:column;height:100vh;width:100vw;overflow:hidden;padding:16px;box-sizing:border-box;background:var(--dash-bg);color:var(--dash-fore-color);font-family:var(--dash-font-family)}.popout-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:16px;padding:10px 18px;background:var(--dash-panel-bg);border-radius:20px;border:1px solid rgba(255,255,255,.06);flex-shrink:0}.popout-title{font-size:16px;font-weight:700;color:#fff}.popout-wrapper .grid-container{flex:1;overflow:auto;border-radius:24px;min-height:0;background:var(--dash-panel-bg);padding:16px;border:1px solid rgba(255,255,255,.05)}.dashboard-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:20px;gap:12px}.tabs-container{display:flex;align-items:center;gap:8px;background:rgba(44,44,46,.6);backdrop-filter:blur(10px);border-radius:25px;padding:6px}.tab{display:flex;align-items:center;gap:4px;padding:8px 14px 8px 18px;border-radius:20px;cursor:pointer;font-size:14px;font-weight:500;color:var(--dash-fore-color);transition:all .3s cubic-bezier(.25,.8,.25,1)}.tab.active{background:#3a3a3c;color:#fff;box-shadow:0 2px 10px #0003}.tab:hover:not(.active){color:#e5e5ea;background:rgba(255,255,255,.05)}.tab-name{flex:1;white-space:nowrap}.tab-action{background:transparent;border:none;color:var(--dash-fore-color);padding:2px;font-size:11px;cursor:pointer;opacity:0;border-radius:50%;width:20px;height:20px;display:flex;align-items:center;justify-content:center;transition:all .2s;flex-shrink:0}.tab:hover .tab-action{opacity:1}.tab-popout:hover{color:var(--dash-accent-color);background:rgba(10,132,255,.18)}.tab-close:hover{color:var(--dash-danger-color);background:rgba(255,69,58,.2)}.btn-add-page{background:transparent;border:none;color:var(--dash-fore-color);padding:8px 16px;cursor:pointer;border-radius:20px;transition:all .2s}.btn-add-page:hover{color:var(--dash-accent-color);background:rgba(10,132,255,.1)}.btn-add-widget{display:flex;align-items:center;gap:6px;background:var(--dash-accent-color);border:none;color:#fff;font-size:14px;font-weight:600;padding:10px 20px;border-radius:22px;cursor:pointer;white-space:nowrap;transition:opacity .2s,transform .15s;box-shadow:0 4px 14px #0a84ff66}.btn-add-widget:hover{opacity:.9;transform:translateY(-1px)}.btn-add-widget:active{transform:translateY(0)}.main-content .grid-container{flex:1;overflow:auto;border-radius:24px;min-height:0}app-grid{background:transparent;display:block;height:100%;width:100%}\n"] }]
287
+ }], ctorParameters: function () { return [{ type: i1.DashboardStateService }]; }, propDecorators: { initialLayout: [{
288
+ type: Input
289
+ }], theme: [{
290
+ type: Input
291
+ }], addWidgetRequested: [{
292
+ type: Output
293
+ }], editWidgetRequested: [{
294
+ type: Output
295
+ }], cellTemplate: [{
296
+ type: ContentChild,
297
+ args: [GridCellDirective, { read: TemplateRef }]
298
+ }] } });
299
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dashboard.component.js","sourceRoot":"","sources":["../../../src/app/dashboard.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, OnInit, OnDestroy, Input, Output, OnChanges,\n  SimpleChanges, EventEmitter, ContentChild, TemplateRef\n} from '@angular/core';\nimport { DashboardStateService } from './dashboard-state.service';\nimport { Page, Widget, DashboardTheme } from './models';\nimport { Subscription } from 'rxjs';\nimport { GridCellDirective } from './custom-grid.component';\n\nconst DEFAULT_THEME: Required<DashboardTheme> = {\n  backgroundColor: '#000000',\n  panelColor:      '#1c1c1e',\n  widgetCardColor: '#2c2c2e',\n  foreColor:       '#8e8e93',\n  accentColor:     '#0a84ff',\n  dangerColor:     '#ff453a',\n  fontFamily:      '-apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Helvetica, Arial, sans-serif',\n};\n\n@Component({\n  selector: 'app-dashboard',\n  template: `\n    <!-- ══════════════════════════════════════════════════════════\n         POPPED-OUT MODE\n         ══════════════════════════════════════════════════════════ -->\n    <ng-container *ngIf=\"isPoppedOut; else normalMode\">\n      <div class=\"popout-wrapper\" [ngStyle]=\"wrapperStyles\">\n        <header class=\"popout-header\">\n          <span class=\"popout-title\">{{ activePage?.name }}</span>\n        </header>\n        <div class=\"grid-container\">\n          <app-grid [widgets]=\"activePage?.widgets || []\" (itemChanged)=\"onItemChanged($event)\">\n            <ng-template gridCell let-widget=\"widget\">\n              <app-widget-renderer\n                [widget]=\"widget\"\n                [theme]=\"resolvedTheme\"\n                (editRequested)=\"editWidgetRequested.emit($event)\"\n                (removeRequested)=\"onRemoveWidget($event)\"\n              >\n                <ng-container\n                  *ngIf=\"cellTemplate\"\n                  [ngTemplateOutlet]=\"cellTemplate\"\n                  [ngTemplateOutletContext]=\"{ widget: widget }\">\n                </ng-container>\n              </app-widget-renderer>\n            </ng-template>\n          </app-grid>\n        </div>\n      </div>\n    </ng-container>\n\n    <!-- ══════════════════════════════════════════════════════════\n         NORMAL MODE\n         ══════════════════════════════════════════════════════════ -->\n    <ng-template #normalMode>\n      <div class=\"dashboard-wrapper\" [ngStyle]=\"wrapperStyles\">\n        <main class=\"main-content\">\n\n          <header class=\"dashboard-header\">\n            <div class=\"tabs-container\">\n              <div\n                *ngFor=\"let page of pages\"\n                class=\"tab\"\n                [class.active]=\"page.id === activePageId\"\n                (click)=\"onSelectPage(page.id)\"\n              >\n                <span class=\"tab-name\">{{ page.name }}</span>\n                <button class=\"tab-action tab-popout\" (click)=\"onPopOut($event, page.id)\" title=\"Open in new window\">\n                  <i class=\"la la-external-link-alt\"></i>\n                </button>\n                <button class=\"tab-action tab-close\" *ngIf=\"pages.length > 1\" (click)=\"onRemovePage($event, page.id)\" title=\"Close\">\n                  <i class=\"la la-times\"></i>\n                </button>\n              </div>\n              <button class=\"btn-add-page\" (click)=\"onAddPage()\" title=\"New workspace\">\n                <i class=\"la la-plus\"></i>\n              </button>\n            </div>\n\n            <!-- Add Widget button — consumer decides what happens -->\n            <button class=\"btn-add-widget\" (click)=\"addWidgetRequested.emit()\" title=\"Add widget\">\n              <i class=\"la la-plus\"></i>\n              <span>Add Widget</span>\n            </button>\n          </header>\n\n          <div class=\"grid-container\">\n            <app-grid [widgets]=\"activePage?.widgets || []\" (itemChanged)=\"onItemChanged($event)\">\n              <ng-template gridCell let-widget=\"widget\">\n                <app-widget-renderer\n                  [widget]=\"widget\"\n                  [theme]=\"resolvedTheme\"\n                  (editRequested)=\"editWidgetRequested.emit($event)\"\n                  (removeRequested)=\"onRemoveWidget($event)\"\n                >\n                  <!-- Stamp consumer's cell template (if provided) inside the card body -->\n                  <ng-container\n                    *ngIf=\"cellTemplate\"\n                    [ngTemplateOutlet]=\"cellTemplate\"\n                    [ngTemplateOutletContext]=\"{ widget: widget }\">\n                  </ng-container>\n                </app-widget-renderer>\n              </ng-template>\n            </app-grid>\n          </div>\n\n        </main>\n      </div>\n    </ng-template>\n  `,\n  styles: [`\n    :host {\n      --dash-bg:        #000000;\n      --dash-panel-bg:  #1c1c1e;\n      --dash-card-bg:   #2c2c2e;\n      --dash-fore-color:#8e8e93;\n      --dash-accent-color: #0a84ff;\n      --dash-danger-color: #ff453a;\n      --dash-font-family: -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Helvetica, Arial, sans-serif;\n    }\n\n    /* ── Normal layout ── */\n    .dashboard-wrapper {\n      display: flex; height: 100vh; width: 100vw; overflow: hidden;\n      padding: 16px; box-sizing: border-box;\n      background: var(--dash-bg);\n      color: var(--dash-fore-color);\n      font-family: var(--dash-font-family);\n    }\n    .main-content {\n      flex: 1; display: flex; flex-direction: column; overflow: hidden;\n      border-radius: 40px; padding: 20px;\n      background: var(--dash-panel-bg);\n      box-shadow: 0 10px 30px rgba(0,0,0,0.5);\n      border: 1px solid rgba(255,255,255,0.05);\n    }\n\n    /* ── Pop-out layout ── */\n    .popout-wrapper {\n      display: flex; flex-direction: column; height: 100vh; width: 100vw;\n      overflow: hidden; padding: 16px; box-sizing: border-box;\n      background: var(--dash-bg);\n      color: var(--dash-fore-color);\n      font-family: var(--dash-font-family);\n    }\n    .popout-header {\n      display: flex; align-items: center; justify-content: space-between;\n      margin-bottom: 16px; padding: 10px 18px;\n      background: var(--dash-panel-bg); border-radius: 20px;\n      border: 1px solid rgba(255,255,255,0.06); flex-shrink: 0;\n    }\n    .popout-title { font-size: 16px; font-weight: 700; color: #fff; }\n    .popout-wrapper .grid-container {\n      flex: 1; overflow: auto; border-radius: 24px; min-height: 0;\n      background: var(--dash-panel-bg); padding: 16px;\n      border: 1px solid rgba(255,255,255,0.05);\n    }\n\n    /* ── Header ── */\n    .dashboard-header {\n      display: flex; align-items: center; justify-content: space-between;\n      margin-bottom: 20px; gap: 12px;\n    }\n    .tabs-container {\n      display: flex; align-items: center; gap: 8px;\n      background: rgba(44,44,46,0.6); backdrop-filter: blur(10px);\n      border-radius: 25px; padding: 6px;\n    }\n    .tab {\n      display: flex; align-items: center; gap: 4px;\n      padding: 8px 14px 8px 18px; border-radius: 20px;\n      cursor: pointer; font-size: 14px; font-weight: 500;\n      color: var(--dash-fore-color);\n      transition: all 0.3s cubic-bezier(0.25,0.8,0.25,1);\n    }\n    .tab.active { background: #3a3a3c; color: #fff; box-shadow: 0 2px 10px rgba(0,0,0,0.2); }\n    .tab:hover:not(.active) { color: #e5e5ea; background: rgba(255,255,255,0.05); }\n    .tab-name { flex: 1; white-space: nowrap; }\n    .tab-action {\n      background: transparent; border: none; color: var(--dash-fore-color);\n      padding: 2px; font-size: 11px; cursor: pointer; opacity: 0; border-radius: 50%;\n      width: 20px; height: 20px; display: flex; align-items: center; justify-content: center;\n      transition: all 0.2s; flex-shrink: 0;\n    }\n    .tab:hover .tab-action { opacity: 1; }\n    .tab-popout:hover { color: var(--dash-accent-color); background: rgba(10,132,255,0.18); }\n    .tab-close:hover  { color: var(--dash-danger-color); background: rgba(255,69,58,0.2); }\n    .btn-add-page {\n      background: transparent; border: none; color: var(--dash-fore-color);\n      padding: 8px 16px; cursor: pointer; border-radius: 20px; transition: all 0.2s;\n    }\n    .btn-add-page:hover { color: var(--dash-accent-color); background: rgba(10,132,255,0.1); }\n    .btn-add-widget {\n      display: flex; align-items: center; gap: 6px;\n      background: var(--dash-accent-color); border: none; color: #fff;\n      font-size: 14px; font-weight: 600; padding: 10px 20px; border-radius: 22px;\n      cursor: pointer; white-space: nowrap;\n      transition: opacity 0.2s, transform 0.15s;\n      box-shadow: 0 4px 14px rgba(10,132,255,0.4);\n    }\n    .btn-add-widget:hover  { opacity: 0.9; transform: translateY(-1px); }\n    .btn-add-widget:active { transform: translateY(0); }\n\n    /* ── Grid ── */\n    .main-content .grid-container { flex: 1; overflow: auto; border-radius: 24px; min-height: 0; }\n    app-grid { background: transparent; display: block; height: 100%; width: 100%; }\n  `]\n})\nexport class DashboardComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() initialLayout?: string;\n  @Input() theme?: DashboardTheme;\n\n  /**\n   * Emits when the user clicks \"Add Widget\".\n   * The consumer should open their own dialog and call `stateService.addWidget(...)`.\n   */\n  @Output() addWidgetRequested = new EventEmitter<void>();\n\n  /**\n   * Emits the Widget when the user clicks the edit (pen) icon on a card.\n   * The consumer should open their own edit UI and call `stateService.updateWidget(...)`.\n   */\n  @Output() editWidgetRequested = new EventEmitter<Widget>();\n\n  /**\n   * Optional: consumer can provide an `<ng-template widgetCell let-widget=\"widget\">` child\n   * to render custom content inside every card body.\n   */\n  @ContentChild(GridCellDirective, { read: TemplateRef })\n  cellTemplate?: TemplateRef<{ widget: Widget }>;\n\n  resolvedTheme: Required<DashboardTheme> = { ...DEFAULT_THEME };\n  wrapperStyles: Record<string, string> = {};\n\n  pages: Page[] = [];\n  activePageId = '';\n  activePage?: Page;\n  isPoppedOut = false;\n\n  private subs = new Subscription();\n\n  constructor(public stateService: DashboardStateService) {}\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['theme'] || changes['initialLayout']) this.applyTheme();\n  }\n\n  ngOnInit() {\n    this.applyTheme();\n\n    if (this.initialLayout) {\n      try { this.stateService.loadLayout(JSON.parse(this.initialLayout)); }\n      catch (e) { console.error('[Dashboard] Failed to parse initialLayout', e); }\n    }\n\n    const hash = window.location.hash?.replace('#', '').trim();\n    if (hash) this.isPoppedOut = true;\n\n    this.subs.add(this.stateService.pages$.subscribe(pages => {\n      this.pages = pages;\n      if (this.isPoppedOut && hash) {\n        const target = pages.find(p => p.id === hash);\n        if (target) this.stateService.setActivePage(hash);\n      }\n      this.updateActivePage();\n    }));\n    this.subs.add(this.stateService.activePageId$.subscribe(id => {\n      this.activePageId = id;\n      this.updateActivePage();\n    }));\n  }\n\n  ngOnDestroy() { this.subs.unsubscribe(); }\n\n  onItemChanged(widget: Widget) {\n    this.stateService.updateWidgetPosition(\n      this.activePageId, widget.id, widget.x, widget.y, widget.cols, widget.rows\n    );\n  }\n\n  onRemoveWidget(widgetId: string) { this.stateService.removeWidget(widgetId); }\n\n  onSelectPage(id: string) { this.stateService.setActivePage(id); }\n\n  onAddPage() {\n    const name = prompt('Workspace name:', `Workspace ${this.pages.length + 1}`);\n    if (name) this.stateService.addPage(name);\n  }\n\n  onRemovePage(event: Event, id: string) {\n    event.stopPropagation();\n    if (confirm('Remove this workspace?')) this.stateService.removePage(id);\n  }\n\n  onPopOut(event: Event, pageId: string) {\n    event.stopPropagation();\n    this.stateService.popOutPage(pageId);\n  }\n\n  serializeLayout(): string { return this.stateService.serializeLayout(); }\n\n  private applyTheme() {\n    this.resolvedTheme = { ...DEFAULT_THEME, ...(this.theme ?? {}) };\n    this.wrapperStyles = {\n      '--dash-bg':           this.resolvedTheme.backgroundColor,\n      '--dash-panel-bg':     this.resolvedTheme.panelColor,\n      '--dash-card-bg':      this.resolvedTheme.widgetCardColor,\n      '--dash-fore-color':   this.resolvedTheme.foreColor,\n      '--dash-accent-color': this.resolvedTheme.accentColor,\n      '--dash-danger-color': this.resolvedTheme.dangerColor,\n      '--dash-font-family':  this.resolvedTheme.fontFamily,\n    };\n  }\n\n  private updateActivePage() {\n    this.activePage = this.pages.find(p => p.id === this.activePageId);\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kZWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2FwcC9tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQSBzaW5nbGUgd2lkZ2V0IG9uIHRoZSBncmlkLlxuICogVGhlIGxpYiBpcyBjb250ZW50LWFnbm9zdGljIOKAlCBgZGF0YWAgY2FuIGhvbGQgYW55dGhpbmcgdGhlIGNvbnN1bWVyIG5lZWRzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFdpZGdldCB7XG4gIGlkOiBzdHJpbmc7XG4gIC8qKiBHcmlkIGNvbHVtbiAoMC1iYXNlZCkgKi9cbiAgeDogbnVtYmVyO1xuICAvKiogR3JpZCByb3cgKDAtYmFzZWQpICovXG4gIHk6IG51bWJlcjtcbiAgLyoqIE51bWJlciBvZiBjb2x1bW5zIHRoaXMgd2lkZ2V0IHNwYW5zICovXG4gIGNvbHM6IG51bWJlcjtcbiAgLyoqIE51bWJlciBvZiByb3dzIHRoaXMgd2lkZ2V0IHNwYW5zICovXG4gIHJvd3M6IG51bWJlcjtcbiAgLyoqIERpc3BsYXkgdGl0bGUgc2hvd24gaW4gdGhlIGNhcmQgaGVhZGVyICovXG4gIHRpdGxlOiBzdHJpbmc7XG4gIC8qKiBPcHRpb25hbCBwZXItd2lkZ2V0IGNhcmQgYmFja2dyb3VuZCBjb2xvci4gT3ZlcnJpZGVzIHRoZSBnbG9iYWwgdGhlbWUgd2lkZ2V0Q2FyZENvbG9yLiAqL1xuICBjYXJkQ29sb3I/OiBzdHJpbmc7XG4gIC8qKlxuICAgKiBBcmJpdHJhcnkgY29uc3VtZXIgZGF0YS5cbiAgICogVGhlIGxpYiBzdG9yZXMgYW5kIHN5bmNzIHRoaXMgYnV0IG5ldmVyIGluc3BlY3RzIGl0LlxuICAgKi9cbiAgZGF0YT86IGFueTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQYWdlIHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICB3aWRnZXRzOiBXaWRnZXRbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBEYXNoYm9hcmRDb25maWcge1xuICBwYWdlczogUGFnZVtdO1xuICBhY3RpdmVQYWdlSWQ6IHN0cmluZztcbn1cblxuLyoqXG4gKiBUaGVtZSBjb25maWd1cmF0aW9uIGZvciB0aGUgZGFzaGJvYXJkLlxuICogQWxsIHByb3BlcnRpZXMgYXJlIG9wdGlvbmFsIOKAlCBhbnkgdmFsdWVzIG5vdCBwcm92aWRlZCB3aWxsIGZhbGwgYmFjayB0byBkZWZhdWx0cy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBEYXNoYm9hcmRUaGVtZSB7XG4gIC8qKiBCYWNrZ3JvdW5kIGNvbG9yIG9mIHRoZSBvdmVyYWxsIGRhc2hib2FyZCB3cmFwcGVyLiBEZWZhdWx0OiBgIzAwMDAwMGAgKi9cbiAgYmFja2dyb3VuZENvbG9yPzogc3RyaW5nO1xuICAvKiogQmFja2dyb3VuZCBjb2xvciBvZiB0aGUgbWFpbiBjb250ZW50IHBhbmVsLiBEZWZhdWx0OiBgIzFjMWMxZWAgKi9cbiAgcGFuZWxDb2xvcj86IHN0cmluZztcbiAgLyoqIEJhY2tncm91bmQgY29sb3Igb2YgaW5kaXZpZHVhbCB3aWRnZXQgY2FyZHMuIERlZmF1bHQ6IGAjMmMyYzJlYCAqL1xuICB3aWRnZXRDYXJkQ29sb3I/OiBzdHJpbmc7XG4gIC8qKiBQcmltYXJ5IGZvcmVncm91bmQvdGV4dCBjb2xvci4gRGVmYXVsdDogYCM4ZThlOTNgICovXG4gIGZvcmVDb2xvcj86IHN0cmluZztcbiAgLyoqIFByaW1hcnkgYWNjZW50IGNvbG9yIChhY3RpdmUgdGFicywgYnV0dG9ucywgZXRjLikuIERlZmF1bHQ6IGAjMGE4NGZmYCAqL1xuICBhY2NlbnRDb2xvcj86IHN0cmluZztcbiAgLyoqIERhbmdlciBjb2xvciB1c2VkIG9uIHJlbW92ZS13aWRnZXQgYnV0dG9uIGhvdmVyLiBEZWZhdWx0OiBgI2ZmNDUzYWAgKi9cbiAgZGFuZ2VyQ29sb3I/OiBzdHJpbmc7XG4gIC8qKiBGb250IGZhbWlseSBhcHBsaWVkIHRvIHRoZSB3aG9sZSBkYXNoYm9hcmQuIERlZmF1bHQ6IHN5c3RlbS11aSBzdGFjayAqL1xuICBmb250RmFtaWx5Pzogc3RyaW5nO1xufVxuIl19
@@ -0,0 +1,83 @@
1
+ import { Component, Input, Output, EventEmitter } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class WidgetRendererComponent {
4
+ constructor() {
5
+ this.editRequested = new EventEmitter();
6
+ this.removeRequested = new EventEmitter();
7
+ this.cardBg = 'var(--dash-card-bg, #2c2c2e)';
8
+ }
9
+ ngOnChanges(changes) {
10
+ if (changes['widget'] || changes['theme']) {
11
+ this.cardBg = this.widget?.cardColor ?? 'var(--dash-card-bg, #2c2c2e)';
12
+ }
13
+ }
14
+ }
15
+ WidgetRendererComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: WidgetRendererComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
16
+ WidgetRendererComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: WidgetRendererComponent, selector: "app-widget-renderer", inputs: { widget: "widget", theme: "theme" }, outputs: { editRequested: "editRequested", removeRequested: "removeRequested" }, usesOnChanges: true, ngImport: i0, template: `
17
+ <div class="widget-card" [style.background]="cardBg">
18
+ <!-- Header: drag handle + title + actions -->
19
+ <div class="widget-header">
20
+ <div class="drag-handle" title="Drag to move">
21
+ <svg width="14" height="14" viewBox="0 0 14 14" fill="currentColor">
22
+ <circle cx="4" cy="3" r="1.2"/><circle cx="10" cy="3" r="1.2"/>
23
+ <circle cx="4" cy="7" r="1.2"/><circle cx="10" cy="7" r="1.2"/>
24
+ <circle cx="4" cy="11" r="1.2"/><circle cx="10" cy="11" r="1.2"/>
25
+ </svg>
26
+ </div>
27
+ <span class="widget-title">{{ widget.title }}</span>
28
+ <div class="header-actions">
29
+ <button class="btn-icon btn-edit" (click)="editRequested.emit(widget)" title="Edit">
30
+ <i class="la la-pen"></i>
31
+ </button>
32
+ <button class="btn-icon btn-remove" (click)="removeRequested.emit(widget.id)" title="Remove">
33
+ <i class="la la-times"></i>
34
+ </button>
35
+ </div>
36
+ </div>
37
+
38
+ <!-- Consumer content slot -->
39
+ <div class="widget-body">
40
+ <ng-content></ng-content>
41
+ </div>
42
+ </div>
43
+ `, isInline: true, styles: [":host{display:flex;flex-direction:column;flex:1;min-height:0;width:100%}.widget-card{flex:1;min-height:0;width:100%;display:flex;flex-direction:column;border-radius:24px;overflow:hidden;box-shadow:0 8px 24px #00000040;border:1px solid rgba(255,255,255,.07);transition:box-shadow .2s ease;box-sizing:border-box;background:var(--dash-card-bg, #2c2c2e)}.widget-card:hover{box-shadow:0 12px 32px #00000059}.widget-header{display:flex;align-items:center;gap:8px;padding:12px 14px 8px;flex-shrink:0}.drag-handle{color:#fff3;cursor:grab;display:flex;align-items:center;flex-shrink:0;transition:color .2s}.drag-handle:hover{color:#ffffff80}.widget-title{flex:1;color:#fff;font-size:14px;font-weight:600;letter-spacing:.2px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.header-actions{display:flex;gap:4px;align-items:center;opacity:0;transition:opacity .2s}.widget-card:hover .header-actions{opacity:1}.btn-icon{background:rgba(255,255,255,.07);border:none;color:#ffffff80;cursor:pointer;font-size:11px;width:24px;height:24px;border-radius:50%;display:flex;align-items:center;justify-content:center;transition:all .2s;flex-shrink:0}.btn-edit:hover{background:rgba(10,132,255,.22);color:var(--dash-accent-color, #0a84ff)}.btn-remove:hover{background:rgba(255,69,58,.22);color:var(--dash-danger-color, #ff453a)}.widget-body{flex:1;min-height:0;display:flex;flex-direction:column;padding:0 14px 14px;overflow:hidden}\n"] });
44
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: WidgetRendererComponent, decorators: [{
45
+ type: Component,
46
+ args: [{ selector: 'app-widget-renderer', template: `
47
+ <div class="widget-card" [style.background]="cardBg">
48
+ <!-- Header: drag handle + title + actions -->
49
+ <div class="widget-header">
50
+ <div class="drag-handle" title="Drag to move">
51
+ <svg width="14" height="14" viewBox="0 0 14 14" fill="currentColor">
52
+ <circle cx="4" cy="3" r="1.2"/><circle cx="10" cy="3" r="1.2"/>
53
+ <circle cx="4" cy="7" r="1.2"/><circle cx="10" cy="7" r="1.2"/>
54
+ <circle cx="4" cy="11" r="1.2"/><circle cx="10" cy="11" r="1.2"/>
55
+ </svg>
56
+ </div>
57
+ <span class="widget-title">{{ widget.title }}</span>
58
+ <div class="header-actions">
59
+ <button class="btn-icon btn-edit" (click)="editRequested.emit(widget)" title="Edit">
60
+ <i class="la la-pen"></i>
61
+ </button>
62
+ <button class="btn-icon btn-remove" (click)="removeRequested.emit(widget.id)" title="Remove">
63
+ <i class="la la-times"></i>
64
+ </button>
65
+ </div>
66
+ </div>
67
+
68
+ <!-- Consumer content slot -->
69
+ <div class="widget-body">
70
+ <ng-content></ng-content>
71
+ </div>
72
+ </div>
73
+ `, styles: [":host{display:flex;flex-direction:column;flex:1;min-height:0;width:100%}.widget-card{flex:1;min-height:0;width:100%;display:flex;flex-direction:column;border-radius:24px;overflow:hidden;box-shadow:0 8px 24px #00000040;border:1px solid rgba(255,255,255,.07);transition:box-shadow .2s ease;box-sizing:border-box;background:var(--dash-card-bg, #2c2c2e)}.widget-card:hover{box-shadow:0 12px 32px #00000059}.widget-header{display:flex;align-items:center;gap:8px;padding:12px 14px 8px;flex-shrink:0}.drag-handle{color:#fff3;cursor:grab;display:flex;align-items:center;flex-shrink:0;transition:color .2s}.drag-handle:hover{color:#ffffff80}.widget-title{flex:1;color:#fff;font-size:14px;font-weight:600;letter-spacing:.2px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.header-actions{display:flex;gap:4px;align-items:center;opacity:0;transition:opacity .2s}.widget-card:hover .header-actions{opacity:1}.btn-icon{background:rgba(255,255,255,.07);border:none;color:#ffffff80;cursor:pointer;font-size:11px;width:24px;height:24px;border-radius:50%;display:flex;align-items:center;justify-content:center;transition:all .2s;flex-shrink:0}.btn-edit:hover{background:rgba(10,132,255,.22);color:var(--dash-accent-color, #0a84ff)}.btn-remove:hover{background:rgba(255,69,58,.22);color:var(--dash-danger-color, #ff453a)}.widget-body{flex:1;min-height:0;display:flex;flex-direction:column;padding:0 14px 14px;overflow:hidden}\n"] }]
74
+ }], propDecorators: { widget: [{
75
+ type: Input
76
+ }], theme: [{
77
+ type: Input
78
+ }], editRequested: [{
79
+ type: Output
80
+ }], removeRequested: [{
81
+ type: Output
82
+ }] } });
83
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2lkZ2V0LXJlbmRlcmVyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hcHAvd2lkZ2V0LXJlbmRlcmVyLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsWUFBWSxFQUN2QyxNQUFNLGVBQWUsQ0FBQzs7QUEwSHZCLE1BQU0sT0FBTyx1QkFBdUI7SUF2SHBDO1FBMkhZLGtCQUFhLEdBQUssSUFBSSxZQUFZLEVBQVUsQ0FBQztRQUM3QyxvQkFBZSxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7UUFFdkQsV0FBTSxHQUFHLDhCQUE4QixDQUFDO0tBT3pDO0lBTEMsV0FBVyxDQUFDLE9BQXNCO1FBQ2hDLElBQUksT0FBTyxDQUFDLFFBQVEsQ0FBQyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsRUFBRTtZQUN6QyxJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxNQUFNLEVBQUUsU0FBUyxJQUFJLDhCQUE4QixDQUFDO1NBQ3hFO0lBQ0gsQ0FBQzs7cUhBYlUsdUJBQXVCO3lHQUF2Qix1QkFBdUIsK01Bckh4Qjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBMkJUOzRGQTBGVSx1QkFBdUI7a0JBdkhuQyxTQUFTOytCQUNFLHFCQUFxQixZQUNyQjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBMkJUOzhCQTJGUSxNQUFNO3NCQUFkLEtBQUs7Z0JBQ0csS0FBSztzQkFBYixLQUFLO2dCQUVJLGFBQWE7c0JBQXRCLE1BQU07Z0JBQ0csZUFBZTtzQkFBeEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIENvbXBvbmVudCwgSW5wdXQsIE91dHB1dCwgRXZlbnRFbWl0dGVyLCBPbkNoYW5nZXMsIFNpbXBsZUNoYW5nZXNcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBXaWRnZXQsIERhc2hib2FyZFRoZW1lIH0gZnJvbSAnLi9tb2RlbHMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdhcHAtd2lkZ2V0LXJlbmRlcmVyJyxcbiAgdGVtcGxhdGU6IGBcbiAgICA8ZGl2IGNsYXNzPVwid2lkZ2V0LWNhcmRcIiBbc3R5bGUuYmFja2dyb3VuZF09XCJjYXJkQmdcIj5cbiAgICAgIDwhLS0gSGVhZGVyOiBkcmFnIGhhbmRsZSArIHRpdGxlICsgYWN0aW9ucyAtLT5cbiAgICAgIDxkaXYgY2xhc3M9XCJ3aWRnZXQtaGVhZGVyXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJkcmFnLWhhbmRsZVwiIHRpdGxlPVwiRHJhZyB0byBtb3ZlXCI+XG4gICAgICAgICAgPHN2ZyB3aWR0aD1cIjE0XCIgaGVpZ2h0PVwiMTRcIiB2aWV3Qm94PVwiMCAwIDE0IDE0XCIgZmlsbD1cImN1cnJlbnRDb2xvclwiPlxuICAgICAgICAgICAgPGNpcmNsZSBjeD1cIjRcIiBjeT1cIjNcIiAgcj1cIjEuMlwiLz48Y2lyY2xlIGN4PVwiMTBcIiBjeT1cIjNcIiAgcj1cIjEuMlwiLz5cbiAgICAgICAgICAgIDxjaXJjbGUgY3g9XCI0XCIgY3k9XCI3XCIgIHI9XCIxLjJcIi8+PGNpcmNsZSBjeD1cIjEwXCIgY3k9XCI3XCIgIHI9XCIxLjJcIi8+XG4gICAgICAgICAgICA8Y2lyY2xlIGN4PVwiNFwiIGN5PVwiMTFcIiByPVwiMS4yXCIvPjxjaXJjbGUgY3g9XCIxMFwiIGN5PVwiMTFcIiByPVwiMS4yXCIvPlxuICAgICAgICAgIDwvc3ZnPlxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgPHNwYW4gY2xhc3M9XCJ3aWRnZXQtdGl0bGVcIj57eyB3aWRnZXQudGl0bGUgfX08L3NwYW4+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJoZWFkZXItYWN0aW9uc1wiPlxuICAgICAgICAgIDxidXR0b24gY2xhc3M9XCJidG4taWNvbiBidG4tZWRpdFwiICAgKGNsaWNrKT1cImVkaXRSZXF1ZXN0ZWQuZW1pdCh3aWRnZXQpXCIgIHRpdGxlPVwiRWRpdFwiPlxuICAgICAgICAgICAgPGkgY2xhc3M9XCJsYSBsYS1wZW5cIj48L2k+XG4gICAgICAgICAgPC9idXR0b24+XG4gICAgICAgICAgPGJ1dHRvbiBjbGFzcz1cImJ0bi1pY29uIGJ0bi1yZW1vdmVcIiAoY2xpY2spPVwicmVtb3ZlUmVxdWVzdGVkLmVtaXQod2lkZ2V0LmlkKVwiIHRpdGxlPVwiUmVtb3ZlXCI+XG4gICAgICAgICAgICA8aSBjbGFzcz1cImxhIGxhLXRpbWVzXCI+PC9pPlxuICAgICAgICAgIDwvYnV0dG9uPlxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuXG4gICAgICA8IS0tIENvbnN1bWVyIGNvbnRlbnQgc2xvdCAtLT5cbiAgICAgIDxkaXYgY2xhc3M9XCJ3aWRnZXQtYm9keVwiPlxuICAgICAgICA8bmctY29udGVudD48L25nLWNvbnRlbnQ+XG4gICAgICA8L2Rpdj5cbiAgICA8L2Rpdj5cbiAgYCxcbiAgc3R5bGVzOiBbYFxuICAgIDpob3N0IHtcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xuICAgICAgZmxleDogMTtcbiAgICAgIG1pbi1oZWlnaHQ6IDA7XG4gICAgICB3aWR0aDogMTAwJTtcbiAgICB9XG5cbiAgICAud2lkZ2V0LWNhcmQge1xuICAgICAgZmxleDogMTtcbiAgICAgIG1pbi1oZWlnaHQ6IDA7XG4gICAgICB3aWR0aDogMTAwJTtcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xuICAgICAgYm9yZGVyLXJhZGl1czogMjRweDtcbiAgICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgICBib3gtc2hhZG93OiAwIDhweCAyNHB4IHJnYmEoMCwwLDAsMC4yNSk7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCByZ2JhKDI1NSwyNTUsMjU1LDAuMDcpO1xuICAgICAgdHJhbnNpdGlvbjogYm94LXNoYWRvdyAwLjJzIGVhc2U7XG4gICAgICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tZGFzaC1jYXJkLWJnLCAjMmMyYzJlKTtcbiAgICB9XG4gICAgLndpZGdldC1jYXJkOmhvdmVyIHtcbiAgICAgIGJveC1zaGFkb3c6IDAgMTJweCAzMnB4IHJnYmEoMCwwLDAsMC4zNSk7XG4gICAgfVxuXG4gICAgLyog4pSA4pSAIEhlYWRlciDilIDilIAgKi9cbiAgICAud2lkZ2V0LWhlYWRlciB7XG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICAgIGdhcDogOHB4O1xuICAgICAgcGFkZGluZzogMTJweCAxNHB4IDhweDtcbiAgICAgIGZsZXgtc2hyaW5rOiAwO1xuICAgIH1cbiAgICAuZHJhZy1oYW5kbGUge1xuICAgICAgY29sb3I6IHJnYmEoMjU1LDI1NSwyNTUsMC4yKTtcbiAgICAgIGN1cnNvcjogZ3JhYjtcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBhbGlnbi1pdGVtczogY2VudGVyO1xuICAgICAgZmxleC1zaHJpbms6IDA7XG4gICAgICB0cmFuc2l0aW9uOiBjb2xvciAwLjJzO1xuICAgIH1cbiAgICAuZHJhZy1oYW5kbGU6aG92ZXIgeyBjb2xvcjogcmdiYSgyNTUsMjU1LDI1NSwwLjUpOyB9XG4gICAgLndpZGdldC10aXRsZSB7XG4gICAgICBmbGV4OiAxO1xuICAgICAgY29sb3I6ICNmZmZmZmY7XG4gICAgICBmb250LXNpemU6IDE0cHg7XG4gICAgICBmb250LXdlaWdodDogNjAwO1xuICAgICAgbGV0dGVyLXNwYWNpbmc6IDAuMnB4O1xuICAgICAgd2hpdGUtc3BhY2U6IG5vd3JhcDtcbiAgICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgICB0ZXh0LW92ZXJmbG93OiBlbGxpcHNpcztcbiAgICB9XG4gICAgLmhlYWRlci1hY3Rpb25zIHtcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBnYXA6IDRweDtcbiAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgICBvcGFjaXR5OiAwO1xuICAgICAgdHJhbnNpdGlvbjogb3BhY2l0eSAwLjJzO1xuICAgIH1cbiAgICAud2lkZ2V0LWNhcmQ6aG92ZXIgLmhlYWRlci1hY3Rpb25zIHsgb3BhY2l0eTogMTsgfVxuXG4gICAgLmJ0bi1pY29uIHtcbiAgICAgIGJhY2tncm91bmQ6IHJnYmEoMjU1LDI1NSwyNTUsMC4wNyk7XG4gICAgICBib3JkZXI6IG5vbmU7XG4gICAgICBjb2xvcjogcmdiYSgyNTUsMjU1LDI1NSwwLjUpO1xuICAgICAgY3Vyc29yOiBwb2ludGVyO1xuICAgICAgZm9udC1zaXplOiAxMXB4O1xuICAgICAgd2lkdGg6IDI0cHg7IGhlaWdodDogMjRweDtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDUwJTtcbiAgICAgIGRpc3BsYXk6IGZsZXg7IGFsaWduLWl0ZW1zOiBjZW50ZXI7IGp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgICAgdHJhbnNpdGlvbjogYWxsIDAuMnM7XG4gICAgICBmbGV4LXNocmluazogMDtcbiAgICB9XG4gICAgLmJ0bi1lZGl0OmhvdmVyICAgeyBiYWNrZ3JvdW5kOiByZ2JhKDEwLDEzMiwyNTUsMC4yMik7ICBjb2xvcjogdmFyKC0tZGFzaC1hY2NlbnQtY29sb3IsICMwYTg0ZmYpOyB9XG4gICAgLmJ0bi1yZW1vdmU6aG92ZXIgeyBiYWNrZ3JvdW5kOiByZ2JhKDI1NSw2OSw1OCwwLjIyKTsgICBjb2xvcjogdmFyKC0tZGFzaC1kYW5nZXItY29sb3IsICNmZjQ1M2EpOyB9XG5cbiAgICAvKiDilIDilIAgQm9keSDilIDilIAgKi9cbiAgICAud2lkZ2V0LWJvZHkge1xuICAgICAgZmxleDogMTtcbiAgICAgIG1pbi1oZWlnaHQ6IDA7XG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgICAgIHBhZGRpbmc6IDAgMTRweCAxNHB4O1xuICAgICAgb3ZlcmZsb3c6IGhpZGRlbjtcbiAgICB9XG4gIGBdXG59KVxuZXhwb3J0IGNsYXNzIFdpZGdldFJlbmRlcmVyQ29tcG9uZW50IGltcGxlbWVudHMgT25DaGFuZ2VzIHtcbiAgQElucHV0KCkgd2lkZ2V0ITogV2lkZ2V0O1xuICBASW5wdXQoKSB0aGVtZT86IFJlcXVpcmVkPERhc2hib2FyZFRoZW1lPjtcblxuICBAT3V0cHV0KCkgZWRpdFJlcXVlc3RlZCAgID0gbmV3IEV2ZW50RW1pdHRlcjxXaWRnZXQ+KCk7XG4gIEBPdXRwdXQoKSByZW1vdmVSZXF1ZXN0ZWQgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZz4oKTtcblxuICBjYXJkQmcgPSAndmFyKC0tZGFzaC1jYXJkLWJnLCAjMmMyYzJlKSc7XG5cbiAgbmdPbkNoYW5nZXMoY2hhbmdlczogU2ltcGxlQ2hhbmdlcykge1xuICAgIGlmIChjaGFuZ2VzWyd3aWRnZXQnXSB8fCBjaGFuZ2VzWyd0aGVtZSddKSB7XG4gICAgICB0aGlzLmNhcmRCZyA9IHRoaXMud2lkZ2V0Py5jYXJkQ29sb3IgPz8gJ3ZhcigtLWRhc2gtY2FyZC1iZywgIzJjMmMyZSknO1xuICAgIH1cbiAgfVxufVxuIl19
@@ -0,0 +1,10 @@
1
+ /*
2
+ * Public API Surface of @ogidor/dashboard
3
+ */
4
+ export * from './app/app.module';
5
+ export * from './app/dashboard.component';
6
+ export * from './app/widget-renderer.component';
7
+ export * from './app/custom-grid.component';
8
+ export * from './app/dashboard-state.service';
9
+ export * from './app/models';
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxrQkFBa0IsQ0FBQztBQUNqQyxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYyw2QkFBNkIsQ0FBQztBQUM1QyxjQUFjLCtCQUErQixDQUFDO0FBQzlDLGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiBAb2dpZG9yL2Rhc2hib2FyZFxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vYXBwL2FwcC5tb2R1bGUnO1xuZXhwb3J0ICogZnJvbSAnLi9hcHAvZGFzaGJvYXJkLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2FwcC93aWRnZXQtcmVuZGVyZXIuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vYXBwL2N1c3RvbS1ncmlkLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2FwcC9kYXNoYm9hcmQtc3RhdGUuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2FwcC9tb2RlbHMnO1xuIl19