@ogidor/dashboard 1.0.2 → 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.
@@ -1,218 +1,288 @@
1
1
  import { Injectable } from '@angular/core';
2
- import { BehaviorSubject, Subject } from 'rxjs';
2
+ import { BehaviorSubject } from 'rxjs';
3
3
  import * as i0 from "@angular/core";
4
4
  export class DashboardStateService {
5
- constructor() {
6
- this.STORAGE_KEY = 'xtb_dashboard_layout';
5
+ constructor(zone) {
6
+ this.zone = zone;
7
+ /**
8
+ * Shared key — stores page list + widget metadata (no positions).
9
+ * Read and written by every window.
10
+ */
11
+ this.SHARED_KEY = 'ogidor_shared';
12
+ this.CHANNEL_NAME = 'ogidor_dashboard_sync';
13
+ this.channel = typeof BroadcastChannel !== 'undefined'
14
+ ? new BroadcastChannel(this.CHANNEL_NAME) : null;
7
15
  this.initialPages = [
8
- {
9
- id: 'page-1',
10
- name: 'Default Workspace',
11
- widgets: [
12
- {
13
- id: 'w-1',
14
- type: 'LINE',
15
- x: 0,
16
- y: 0,
17
- cols: 4,
18
- rows: 4,
19
- title: 'Market Trend',
20
- data: { series: [{ name: 'Price', data: [30, 40, 35, 50, 49, 60, 70, 91, 125] }] }
21
- },
22
- {
23
- id: 'w-2',
24
- type: 'DONUT',
25
- x: 4,
26
- y: 0,
27
- cols: 2,
28
- rows: 4,
29
- title: 'Asset Allocation',
30
- data: { series: [44, 55, 41, 17, 15] }
31
- }
32
- ]
33
- }
16
+ { id: 'page-1', name: 'Default Workspace', widgets: [] }
34
17
  ];
35
18
  this.pagesSubject = new BehaviorSubject(this.initialPages);
36
19
  this.activePageIdSubject = new BehaviorSubject(this.initialPages[0].id);
37
- /**
38
- * Emits whenever a widget's data is updated programmatically.
39
- * Key = widget id, value = new data payload.
40
- */
41
- this.widgetDataSubject = new Subject();
42
- this.widgetData$ = this.widgetDataSubject.asObservable();
43
20
  this.pages$ = this.pagesSubject.asObservable();
44
21
  this.activePageId$ = this.activePageIdSubject.asObservable();
45
- const saved = localStorage.getItem(this.STORAGE_KEY);
46
- if (saved) {
22
+ // Determine whether we are a pop-out window
23
+ const hash = typeof window !== 'undefined'
24
+ ? window.location.hash.replace('#', '').trim() : '';
25
+ this.positionsKey = hash ? `ogidor_positions_${hash}` : 'ogidor_positions_main';
26
+ // 1. Load the shared structural state
27
+ const shared = localStorage.getItem(this.SHARED_KEY);
28
+ if (shared) {
47
29
  try {
48
- this.loadLayout(JSON.parse(saved));
30
+ this._applyShared(JSON.parse(shared), false);
49
31
  }
50
32
  catch (e) {
51
- console.error('Failed to load saved layout', e);
33
+ console.error('[Dashboard] bad shared state', e);
52
34
  }
53
35
  }
36
+ // 2. Overlay this window's saved positions on top
37
+ this._restorePositions();
38
+ // 3. Listen for structural changes broadcast by other windows
39
+ if (this.channel) {
40
+ this.channel.onmessage = (ev) => {
41
+ this.zone.run(() => this._onSyncEvent(ev.data));
42
+ };
43
+ }
54
44
  }
45
+ ngOnDestroy() { this.channel?.close(); }
46
+ // ── Page actions ──────────────────────────────────────────────
55
47
  getActivePage() {
56
48
  return this.pagesSubject.value.find(p => p.id === this.activePageIdSubject.value);
57
49
  }
58
- setActivePage(id) {
59
- this.activePageIdSubject.next(id);
60
- }
50
+ setActivePage(id) { this.activePageIdSubject.next(id); }
61
51
  addPage(name) {
62
- const newPage = {
63
- id: `page-${Date.now()}`,
64
- name,
65
- widgets: []
66
- };
67
- const updatedPages = [...this.pagesSubject.value, newPage];
68
- this.pagesSubject.next(updatedPages);
69
- this.activePageIdSubject.next(newPage.id);
70
- this.saveToLocalStorage();
52
+ const page = { id: `page-${Date.now()}`, name, widgets: [] };
53
+ this.pagesSubject.next([...this.pagesSubject.value, page]);
54
+ this.activePageIdSubject.next(page.id);
55
+ this._saveShared();
56
+ this._broadcast({ type: 'PAGE_ADDED', page });
71
57
  }
72
58
  removePage(id) {
73
59
  const pages = this.pagesSubject.value;
74
60
  if (pages.length <= 1)
75
61
  return;
76
- const updatedPages = pages.filter(p => p.id !== id);
77
- this.pagesSubject.next(updatedPages);
62
+ const updated = pages.filter(p => p.id !== id);
63
+ this.pagesSubject.next(updated);
78
64
  if (this.activePageIdSubject.value === id) {
79
- this.activePageIdSubject.next(updatedPages[0].id);
65
+ this.activePageIdSubject.next(updated[0].id);
80
66
  }
81
- this.saveToLocalStorage();
67
+ this._saveShared();
68
+ this._broadcast({ type: 'PAGE_REMOVED', pageId: id });
82
69
  }
83
- addWidget(type) {
70
+ // ── Widget structural actions (synced to all windows) ─────────
71
+ addWidget(widget) {
84
72
  const activePage = this.getActivePage();
85
73
  if (!activePage)
86
- return;
74
+ throw new Error('No active page');
87
75
  const newWidget = {
88
76
  id: `widget-${Date.now()}`,
89
- type,
90
- x: 0, y: 0,
91
- cols: type === 'DONUT' ? 2 : 4,
92
- rows: 4,
93
- title: `${type.charAt(0) + type.slice(1).toLowerCase()} Chart`,
94
- data: this.getDefaultDataForType(type)
77
+ x: 0, y: 0, cols: 4, rows: 3,
78
+ ...widget,
95
79
  };
96
- activePage.widgets.push(newWidget);
97
- this.updatePages(this.pagesSubject.value);
80
+ activePage.widgets = [...activePage.widgets, newWidget];
81
+ this.pagesSubject.next([...this.pagesSubject.value]);
82
+ this._saveShared();
83
+ this._savePositions(); // register default position locally
84
+ this._broadcast({ type: 'WIDGET_ADDED', pageId: activePage.id, widget: newWidget });
85
+ return newWidget;
98
86
  }
99
- addWidgetWithData(type, title, data) {
87
+ updateWidget(widgetId, patch) {
88
+ const pages = this.pagesSubject.value;
89
+ for (const page of pages) {
90
+ const w = page.widgets.find(w => w.id === widgetId);
91
+ if (w) {
92
+ Object.assign(w, patch);
93
+ this.pagesSubject.next([...pages]);
94
+ this._saveShared();
95
+ this._broadcast({ type: 'WIDGET_META', widgetId, patch });
96
+ return;
97
+ }
98
+ }
99
+ }
100
+ removeWidget(widgetId) {
100
101
  const activePage = this.getActivePage();
101
102
  if (!activePage)
102
103
  return;
103
- const newWidget = {
104
- id: `widget-${Date.now()}`,
105
- type,
106
- x: 0, y: 0,
107
- cols: type === 'DONUT' ? 3 : 6,
108
- rows: 4,
109
- title,
110
- data
111
- };
112
- activePage.widgets.push(newWidget);
113
- this.updatePages(this.pagesSubject.value);
114
- }
115
- getDefaultDataForType(type) {
116
- if (type === 'DONUT') {
117
- return { series: [30, 20, 50] };
118
- }
119
- return {
120
- series: [{
121
- name: 'Sample Data',
122
- data: Array.from({ length: 10 }, () => Math.floor(Math.random() * 100))
123
- }]
124
- };
104
+ const pageId = activePage.id;
105
+ activePage.widgets = activePage.widgets.filter(w => w.id !== widgetId);
106
+ this.pagesSubject.next([...this.pagesSubject.value]);
107
+ this._saveShared();
108
+ this._savePositions();
109
+ this._broadcast({ type: 'WIDGET_REMOVED', pageId, widgetId });
125
110
  }
111
+ // ── Position actions (local to THIS window only) ──────────────
112
+ /**
113
+ * Update a widget's grid position/size.
114
+ * Saved only for this window — never broadcast to others.
115
+ */
126
116
  updateWidgetPosition(pageId, widgetId, x, y, cols, rows) {
127
117
  const pages = this.pagesSubject.value;
128
- const page = pages.find(p => p.id === pageId);
129
- if (page) {
130
- const widget = page.widgets.find(w => w.id === widgetId);
131
- if (widget) {
132
- widget.x = x;
133
- widget.y = y;
134
- widget.cols = cols;
135
- widget.rows = rows;
136
- this.updatePages(pages);
137
- }
118
+ const widget = pages.find(p => p.id === pageId)?.widgets.find(w => w.id === widgetId);
119
+ if (widget) {
120
+ widget.x = x;
121
+ widget.y = y;
122
+ widget.cols = cols;
123
+ widget.rows = rows;
124
+ this.pagesSubject.next([...pages]);
125
+ this._savePositions(); // local only — intentionally no _saveShared / no broadcast
138
126
  }
139
127
  }
140
- removeWidget(widgetId) {
141
- const activePage = this.getActivePage();
142
- if (activePage) {
143
- activePage.widgets = activePage.widgets.filter(w => w.id !== widgetId);
144
- this.updatePages(this.pagesSubject.value);
145
- }
128
+ // ── Serialization ─────────────────────────────────────────────
129
+ serializeLayout() {
130
+ return JSON.stringify({
131
+ pages: this.pagesSubject.value,
132
+ activePageId: this.activePageIdSubject.value,
133
+ });
134
+ }
135
+ loadLayout(config) {
136
+ if (!config?.pages)
137
+ return;
138
+ this._applyShared(config, true);
139
+ this._saveShared();
140
+ this._savePositions();
141
+ }
142
+ popOutPage(pageId) {
143
+ const url = `${window.location.origin}${window.location.pathname}#${pageId}`;
144
+ window.open(url, `workspace_${pageId}`, 'width=1280,height=800,menubar=no,toolbar=no,location=no,status=no');
146
145
  }
146
+ // ── Private helpers ───────────────────────────────────────────
147
147
  /**
148
- * Push new data into a widget by its id.
149
- * The widget's chart will re-render immediately.
150
- *
151
- * @param widgetId The `id` of the target widget.
152
- * @param data New data — `LineBarData` for LINE/BAR, `DonutData` for DONUT.
148
+ * Save page list + widget metadata (titles, cardColor, data) — no positions.
153
149
  */
154
- updateWidgetData(widgetId, data) {
155
- const pages = this.pagesSubject.value;
156
- for (const page of pages) {
157
- const widget = page.widgets.find(w => w.id === widgetId);
158
- if (widget) {
159
- widget.data = data;
160
- this.widgetDataSubject.next({ widgetId, data });
161
- this.updatePages(pages);
162
- return;
150
+ _saveShared() {
151
+ const stripped = this.pagesSubject.value.map(p => ({
152
+ ...p,
153
+ widgets: p.widgets.map(({ id, title, cardColor, data, cols, rows }) =>
154
+ // Keep default cols/rows so new windows get a sensible first-open layout.
155
+ // x/y are intentionally omitted from shared state.
156
+ ({ id, title, cardColor, data, x: 0, y: 0, cols, rows })),
157
+ }));
158
+ localStorage.setItem(this.SHARED_KEY, JSON.stringify({
159
+ pages: stripped,
160
+ activePageId: this.activePageIdSubject.value,
161
+ }));
162
+ }
163
+ /**
164
+ * Save this window's grid positions (x, y, cols, rows) per widget.
165
+ */
166
+ _savePositions() {
167
+ const positions = {};
168
+ for (const page of this.pagesSubject.value) {
169
+ for (const w of page.widgets) {
170
+ positions[`${page.id}:${w.id}`] = { x: w.x, y: w.y, cols: w.cols, rows: w.rows };
163
171
  }
164
172
  }
165
- console.warn(`[Dashboard] updateWidgetData: widget "${widgetId}" not found.`);
173
+ localStorage.setItem(this.positionsKey, JSON.stringify(positions));
166
174
  }
167
- updateWidgetMeta(widgetId, meta) {
168
- const pages = this.pagesSubject.value;
169
- for (const page of pages) {
170
- const widget = page.widgets.find(w => w.id === widgetId);
171
- if (widget) {
172
- if (meta.title !== undefined)
173
- widget.title = meta.title;
174
- if (meta.colors !== undefined)
175
- widget.colors = meta.colors;
176
- if (meta.cardColor !== undefined)
177
- widget.cardColor = meta.cardColor;
178
- if (meta.data !== undefined) {
179
- widget.data = meta.data;
180
- this.widgetDataSubject.next({ widgetId, data: meta.data });
175
+ /**
176
+ * Overlay the positions saved for THIS window on top of the current pages.
177
+ */
178
+ _restorePositions() {
179
+ const raw = localStorage.getItem(this.positionsKey);
180
+ if (!raw)
181
+ return;
182
+ try {
183
+ const positions = JSON.parse(raw);
184
+ const pages = this.pagesSubject.value;
185
+ for (const page of pages) {
186
+ for (const w of page.widgets) {
187
+ const pos = positions[`${page.id}:${w.id}`];
188
+ if (pos) {
189
+ w.x = pos.x;
190
+ w.y = pos.y;
191
+ w.cols = pos.cols;
192
+ w.rows = pos.rows;
193
+ }
181
194
  }
182
- this.updatePages(pages);
183
- return;
184
195
  }
196
+ this.pagesSubject.next([...pages]);
197
+ }
198
+ catch (e) {
199
+ console.error('[Dashboard] bad positions state', e);
185
200
  }
186
201
  }
187
- updatePages(pages) {
188
- this.pagesSubject.next([...pages]);
189
- this.saveToLocalStorage();
190
- }
191
- serializeLayout() {
192
- const config = {
193
- pages: this.pagesSubject.value,
194
- activePageId: this.activePageIdSubject.value
195
- };
196
- return JSON.stringify(config);
202
+ /**
203
+ * Apply a shared config object (page list + metadata) without touching
204
+ * this window's saved positions.
205
+ */
206
+ _applyShared(config, overwritePositions) {
207
+ if (!config?.pages)
208
+ return;
209
+ this.pagesSubject.next(config.pages);
210
+ if (config.activePageId)
211
+ this.activePageIdSubject.next(config.activePageId);
212
+ if (!overwritePositions)
213
+ this._restorePositions();
197
214
  }
198
- loadLayout(config) {
199
- if (config && config.pages) {
200
- this.pagesSubject.next(config.pages);
201
- if (config.activePageId) {
202
- this.activePageIdSubject.next(config.activePageId);
215
+ /**
216
+ * Handle a structural sync event arriving from another window.
217
+ * Position changes are never sent so we never receive them here.
218
+ */
219
+ _onSyncEvent(event) {
220
+ // Work on a shallow clone of pages so Angular detects the change
221
+ const pages = this.pagesSubject.value.map(p => ({
222
+ ...p, widgets: [...p.widgets]
223
+ }));
224
+ switch (event.type) {
225
+ case 'PAGE_ADDED':
226
+ if (!pages.find(p => p.id === event.page.id)) {
227
+ pages.push({ ...event.page, widgets: [...event.page.widgets] });
228
+ this.pagesSubject.next(pages);
229
+ this._saveShared();
230
+ }
231
+ break;
232
+ case 'PAGE_REMOVED': {
233
+ const updated = pages.filter(p => p.id !== event.pageId);
234
+ if (updated.length !== pages.length) {
235
+ this.pagesSubject.next(updated);
236
+ if (this.activePageIdSubject.value === event.pageId) {
237
+ this.activePageIdSubject.next(updated[0]?.id ?? '');
238
+ }
239
+ this._saveShared();
240
+ }
241
+ break;
242
+ }
243
+ case 'WIDGET_ADDED': {
244
+ const page = pages.find(p => p.id === event.pageId);
245
+ if (page && !page.widgets.find(w => w.id === event.widget.id)) {
246
+ page.widgets = [...page.widgets, { ...event.widget }];
247
+ this.pagesSubject.next(pages);
248
+ this._saveShared();
249
+ // Don't touch positions — this window will keep its own layout for
250
+ // existing widgets; the new one starts at its default position.
251
+ }
252
+ break;
253
+ }
254
+ case 'WIDGET_REMOVED': {
255
+ const page = pages.find(p => p.id === event.pageId);
256
+ if (page) {
257
+ page.widgets = page.widgets.filter(w => w.id !== event.widgetId);
258
+ this.pagesSubject.next(pages);
259
+ this._saveShared();
260
+ this._savePositions();
261
+ }
262
+ break;
263
+ }
264
+ case 'WIDGET_META': {
265
+ for (const page of pages) {
266
+ const w = page.widgets.find(w => w.id === event.widgetId);
267
+ if (w) {
268
+ Object.assign(w, event.patch);
269
+ this.pagesSubject.next(pages);
270
+ this._saveShared();
271
+ break;
272
+ }
273
+ }
274
+ break;
203
275
  }
204
276
  }
205
277
  }
206
- saveToLocalStorage() {
207
- localStorage.setItem(this.STORAGE_KEY, this.serializeLayout());
278
+ _broadcast(event) {
279
+ this.channel?.postMessage(event);
208
280
  }
209
281
  }
210
- DashboardStateService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
282
+ DashboardStateService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardStateService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
211
283
  DashboardStateService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardStateService, providedIn: 'root' });
212
284
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardStateService, decorators: [{
213
285
  type: Injectable,
214
- args: [{
215
- providedIn: 'root'
216
- }]
217
- }], ctorParameters: function () { return []; } });
218
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dashboard-state.service.js","sourceRoot":"","sources":["../../../src/app/dashboard-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Subject } from 'rxjs';\nimport { Page, Widget, DashboardConfig, WidgetType, LineBarData, DonutData } from './models';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class DashboardStateService {\n  private readonly STORAGE_KEY = 'xtb_dashboard_layout';\n  \n  private initialPages: Page[] = [\n    {\n      id: 'page-1',\n      name: 'Default Workspace',\n      widgets: [\n        {\n          id: 'w-1',\n          type: 'LINE',\n          x: 0,\n          y: 0,\n          cols: 4,\n          rows: 4,\n          title: 'Market Trend',\n          data: { series: [{ name: 'Price', data: [30, 40, 35, 50, 49, 60, 70, 91, 125] }] }\n        },\n        {\n          id: 'w-2',\n          type: 'DONUT',\n          x: 4,\n          y: 0,\n          cols: 2,\n          rows: 4,\n          title: 'Asset Allocation',\n          data: { series: [44, 55, 41, 17, 15] }\n        }\n      ]\n    }\n  ];\n\n  private pagesSubject = new BehaviorSubject<Page[]>(this.initialPages);\n  private activePageIdSubject = new BehaviorSubject<string>(this.initialPages[0].id);\n\n  /**\n   * Emits whenever a widget's data is updated programmatically.\n   * Key = widget id, value = new data payload.\n   */\n  private widgetDataSubject = new Subject<{ widgetId: string; data: LineBarData | DonutData }>();\n  widgetData$ = this.widgetDataSubject.asObservable();\n\n  pages$ = this.pagesSubject.asObservable();\n  activePageId$ = this.activePageIdSubject.asObservable();\n\n  constructor() {\n    const saved = localStorage.getItem(this.STORAGE_KEY);\n    if (saved) {\n      try {\n        this.loadLayout(JSON.parse(saved));\n      } catch (e) {\n        console.error('Failed to load saved layout', e);\n      }\n    }\n  }\n\n  getActivePage(): Page | undefined {\n    return this.pagesSubject.value.find(p => p.id === this.activePageIdSubject.value);\n  }\n\n  setActivePage(id: string) {\n    this.activePageIdSubject.next(id);\n  }\n\n  addPage(name: string) {\n    const newPage: Page = {\n      id: `page-${Date.now()}`,\n      name,\n      widgets: []\n    };\n    const updatedPages = [...this.pagesSubject.value, newPage];\n    this.pagesSubject.next(updatedPages);\n    this.activePageIdSubject.next(newPage.id);\n    this.saveToLocalStorage();\n  }\n\n  removePage(id: string) {\n    const pages = this.pagesSubject.value;\n    if (pages.length <= 1) return;\n\n    const updatedPages = pages.filter(p => p.id !== id);\n    this.pagesSubject.next(updatedPages);\n    \n    if (this.activePageIdSubject.value === id) {\n      this.activePageIdSubject.next(updatedPages[0].id);\n    }\n    this.saveToLocalStorage();\n  }\n\n  addWidget(type: WidgetType) {\n    const activePage = this.getActivePage();\n    if (!activePage) return;\n\n    const newWidget: Widget = {\n      id: `widget-${Date.now()}`,\n      type,\n      x: 0, y: 0,\n      cols: type === 'DONUT' ? 2 : 4,\n      rows: 4,\n      title: `${type.charAt(0) + type.slice(1).toLowerCase()} Chart`,\n      data: this.getDefaultDataForType(type)\n    };\n\n    activePage.widgets.push(newWidget);\n    this.updatePages(this.pagesSubject.value);\n  }\n\n  addWidgetWithData(type: WidgetType, title: string, data: LineBarData | DonutData) {\n    const activePage = this.getActivePage();\n    if (!activePage) return;\n\n    const newWidget: Widget = {\n      id: `widget-${Date.now()}`,\n      type,\n      x: 0, y: 0,\n      cols: type === 'DONUT' ? 3 : 6,\n      rows: 4,\n      title,\n      data\n    };\n\n    activePage.widgets.push(newWidget);\n    this.updatePages(this.pagesSubject.value);\n  }\n\n  private getDefaultDataForType(type: WidgetType) {\n    if (type === 'DONUT') {\n      return { series: [30, 20, 50] };\n    }\n    return {\n      series: [{\n        name: 'Sample Data',\n        data: Array.from({ length: 10 }, () => Math.floor(Math.random() * 100))\n      }]\n    };\n  }\n\n  updateWidgetPosition(pageId: string, widgetId: string, x: number, y: number, cols: number, rows: number) {\n    const pages = this.pagesSubject.value;\n    const page = pages.find(p => p.id === pageId);\n    if (page) {\n      const widget = page.widgets.find(w => w.id === widgetId);\n      if (widget) {\n        widget.x = x;\n        widget.y = y;\n        widget.cols = cols;\n        widget.rows = rows;\n        this.updatePages(pages);\n      }\n    }\n  }\n\n  removeWidget(widgetId: string) {\n    const activePage = this.getActivePage();\n    if (activePage) {\n      activePage.widgets = activePage.widgets.filter(w => w.id !== widgetId);\n      this.updatePages(this.pagesSubject.value);\n    }\n  }\n\n  /**\n   * Push new data into a widget by its id.\n   * The widget's chart will re-render immediately.\n   *\n   * @param widgetId  The `id` of the target widget.\n   * @param data      New data — `LineBarData` for LINE/BAR, `DonutData` for DONUT.\n   */\n  updateWidgetData(widgetId: string, data: LineBarData | DonutData) {\n    const pages = this.pagesSubject.value;\n    for (const page of pages) {\n      const widget = page.widgets.find(w => w.id === widgetId);\n      if (widget) {\n        widget.data = data;\n        this.widgetDataSubject.next({ widgetId, data });\n        this.updatePages(pages);\n        return;\n      }\n    }\n    console.warn(`[Dashboard] updateWidgetData: widget \"${widgetId}\" not found.`);\n  }\n\n  updateWidgetMeta(widgetId: string, meta: { title?: string; colors?: string[]; cardColor?: string; data?: LineBarData | DonutData }) {\n    const pages = this.pagesSubject.value;\n    for (const page of pages) {\n      const widget = page.widgets.find(w => w.id === widgetId);\n      if (widget) {\n        if (meta.title     !== undefined) widget.title     = meta.title;\n        if (meta.colors    !== undefined) widget.colors    = meta.colors;\n        if (meta.cardColor !== undefined) widget.cardColor = meta.cardColor;\n        if (meta.data      !== undefined) {\n          widget.data = meta.data;\n          this.widgetDataSubject.next({ widgetId, data: meta.data });\n        }\n        this.updatePages(pages);\n        return;\n      }\n    }\n  }\n\n  private updatePages(pages: Page[]) {\n    this.pagesSubject.next([...pages]);\n    this.saveToLocalStorage();\n  }\n\n  serializeLayout(): string {\n    const config: DashboardConfig = {\n      pages: this.pagesSubject.value,\n      activePageId: this.activePageIdSubject.value\n    };\n    return JSON.stringify(config);\n  }\n\n  loadLayout(config: any) {\n    if (config && config.pages) {\n      this.pagesSubject.next(config.pages);\n      if (config.activePageId) {\n        this.activePageIdSubject.next(config.activePageId);\n      }\n    }\n  }\n\n  private saveToLocalStorage() {\n    localStorage.setItem(this.STORAGE_KEY, this.serializeLayout());\n  }\n}\n"]}
286
+ args: [{ providedIn: 'root' }]
287
+ }], ctorParameters: function () { return [{ type: i0.NgZone }]; } });
288
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dashboard-state.service.js","sourceRoot":"","sources":["../../../src/app/dashboard-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, OnDestroy } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { Page, Widget, DashboardConfig } from './models';\n\ntype PositionMap = Record<string, { x: number; y: number; cols: number; rows: number }>;\n\ntype SyncEvent =\n  | { type: 'PAGE_ADDED';    page: Page }\n  | { type: 'PAGE_REMOVED';  pageId: string }\n  | { type: 'PAGE_RENAMED';  pageId: string; name: string }\n  | { type: 'WIDGET_ADDED';  pageId: string; widget: Widget }\n  | { type: 'WIDGET_REMOVED';pageId: string; widgetId: string }\n  | { type: 'WIDGET_META';   widgetId: string; patch: Partial<Omit<Widget,'id'|'x'|'y'|'cols'|'rows'>> };\n\n@Injectable({ providedIn: 'root' })\nexport class DashboardStateService implements OnDestroy {\n  /**\n   * Shared key — stores page list + widget metadata (no positions).\n   * Read and written by every window.\n   */\n  private readonly SHARED_KEY = 'ogidor_shared';\n\n  /**\n   * Per-window positions key — stores {pageId:widgetId → {x,y,cols,rows}}.\n   * Pop-out windows get a key scoped to their pageId so they never overwrite\n   * the main window or each other.\n   *   main window  → ogidor_positions_main\n   *   pop-out #abc → ogidor_positions_abc\n   */\n  private readonly positionsKey: string;\n\n  private readonly CHANNEL_NAME = 'ogidor_dashboard_sync';\n  private channel = typeof BroadcastChannel !== 'undefined'\n    ? new BroadcastChannel(this.CHANNEL_NAME) : null;\n\n  private readonly initialPages: Page[] = [\n    { id: 'page-1', name: 'Default Workspace', widgets: [] }\n  ];\n\n  private pagesSubject       = new BehaviorSubject<Page[]>(this.initialPages);\n  private activePageIdSubject = new BehaviorSubject<string>(this.initialPages[0].id);\n\n  pages$        = this.pagesSubject.asObservable();\n  activePageId$ = this.activePageIdSubject.asObservable();\n\n  constructor(private zone: NgZone) {\n    // Determine whether we are a pop-out window\n    const hash = typeof window !== 'undefined'\n      ? window.location.hash.replace('#', '').trim() : '';\n    this.positionsKey = hash ? `ogidor_positions_${hash}` : 'ogidor_positions_main';\n\n    // 1. Load the shared structural state\n    const shared = localStorage.getItem(this.SHARED_KEY);\n    if (shared) {\n      try { this._applyShared(JSON.parse(shared), false); }\n      catch (e) { console.error('[Dashboard] bad shared state', e); }\n    }\n\n    // 2. Overlay this window's saved positions on top\n    this._restorePositions();\n\n    // 3. Listen for structural changes broadcast by other windows\n    if (this.channel) {\n      this.channel.onmessage = (ev: MessageEvent<SyncEvent>) => {\n        this.zone.run(() => this._onSyncEvent(ev.data));\n      };\n    }\n  }\n\n  ngOnDestroy() { this.channel?.close(); }\n\n  // ── Page actions ──────────────────────────────────────────────\n\n  getActivePage(): Page | undefined {\n    return this.pagesSubject.value.find(p => p.id === this.activePageIdSubject.value);\n  }\n\n  setActivePage(id: string) { this.activePageIdSubject.next(id); }\n\n  addPage(name: string) {\n    const page: Page = { id: `page-${Date.now()}`, name, widgets: [] };\n    this.pagesSubject.next([...this.pagesSubject.value, page]);\n    this.activePageIdSubject.next(page.id);\n    this._saveShared();\n    this._broadcast({ type: 'PAGE_ADDED', page });\n  }\n\n  removePage(id: string) {\n    const pages = this.pagesSubject.value;\n    if (pages.length <= 1) return;\n    const updated = pages.filter(p => p.id !== id);\n    this.pagesSubject.next(updated);\n    if (this.activePageIdSubject.value === id) {\n      this.activePageIdSubject.next(updated[0].id);\n    }\n    this._saveShared();\n    this._broadcast({ type: 'PAGE_REMOVED', pageId: id });\n  }\n\n  // ── Widget structural actions (synced to all windows) ─────────\n\n  addWidget(widget: Partial<Widget> & { title: string }): Widget {\n    const activePage = this.getActivePage();\n    if (!activePage) throw new Error('No active page');\n    const newWidget: Widget = {\n      id: `widget-${Date.now()}`,\n      x: 0, y: 0, cols: 4, rows: 3,\n      ...widget,\n    };\n    activePage.widgets = [...activePage.widgets, newWidget];\n    this.pagesSubject.next([...this.pagesSubject.value]);\n    this._saveShared();\n    this._savePositions();                          // register default position locally\n    this._broadcast({ type: 'WIDGET_ADDED', pageId: activePage.id, widget: newWidget });\n    return newWidget;\n  }\n\n  updateWidget(widgetId: string, patch: Partial<Omit<Widget,'id'|'x'|'y'|'cols'|'rows'>>) {\n    const pages = this.pagesSubject.value;\n    for (const page of pages) {\n      const w = page.widgets.find(w => w.id === widgetId);\n      if (w) {\n        Object.assign(w, patch);\n        this.pagesSubject.next([...pages]);\n        this._saveShared();\n        this._broadcast({ type: 'WIDGET_META', widgetId, patch });\n        return;\n      }\n    }\n  }\n\n  removeWidget(widgetId: string) {\n    const activePage = this.getActivePage();\n    if (!activePage) return;\n    const pageId = activePage.id;\n    activePage.widgets = activePage.widgets.filter(w => w.id !== widgetId);\n    this.pagesSubject.next([...this.pagesSubject.value]);\n    this._saveShared();\n    this._savePositions();\n    this._broadcast({ type: 'WIDGET_REMOVED', pageId, widgetId });\n  }\n\n  // ── Position actions (local to THIS window only) ──────────────\n\n  /**\n   * Update a widget's grid position/size.\n   * Saved only for this window — never broadcast to others.\n   */\n  updateWidgetPosition(pageId: string, widgetId: string, x: number, y: number, cols: number, rows: number) {\n    const pages = this.pagesSubject.value;\n    const widget = pages.find(p => p.id === pageId)?.widgets.find(w => w.id === widgetId);\n    if (widget) {\n      widget.x = x; widget.y = y; widget.cols = cols; widget.rows = rows;\n      this.pagesSubject.next([...pages]);\n      this._savePositions();          // local only — intentionally no _saveShared / no broadcast\n    }\n  }\n\n  // ── Serialization ─────────────────────────────────────────────\n\n  serializeLayout(): string {\n    return JSON.stringify({\n      pages: this.pagesSubject.value,\n      activePageId: this.activePageIdSubject.value,\n    } as DashboardConfig);\n  }\n\n  loadLayout(config: any) {\n    if (!config?.pages) return;\n    this._applyShared(config, true);\n    this._saveShared();\n    this._savePositions();\n  }\n\n  popOutPage(pageId: string) {\n    const url = `${window.location.origin}${window.location.pathname}#${pageId}`;\n    window.open(url, `workspace_${pageId}`,\n      'width=1280,height=800,menubar=no,toolbar=no,location=no,status=no');\n  }\n\n  // ── Private helpers ───────────────────────────────────────────\n\n  /**\n   * Save page list + widget metadata (titles, cardColor, data) — no positions.\n   */\n  private _saveShared() {\n    const stripped = this.pagesSubject.value.map(p => ({\n      ...p,\n      widgets: p.widgets.map(({ id, title, cardColor, data, cols, rows }) =>\n        // Keep default cols/rows so new windows get a sensible first-open layout.\n        // x/y are intentionally omitted from shared state.\n        ({ id, title, cardColor, data, x: 0, y: 0, cols, rows })\n      ),\n    }));\n    localStorage.setItem(this.SHARED_KEY, JSON.stringify({\n      pages: stripped,\n      activePageId: this.activePageIdSubject.value,\n    }));\n  }\n\n  /**\n   * Save this window's grid positions (x, y, cols, rows) per widget.\n   */\n  private _savePositions() {\n    const positions: PositionMap = {};\n    for (const page of this.pagesSubject.value) {\n      for (const w of page.widgets) {\n        positions[`${page.id}:${w.id}`] = { x: w.x, y: w.y, cols: w.cols, rows: w.rows };\n      }\n    }\n    localStorage.setItem(this.positionsKey, JSON.stringify(positions));\n  }\n\n  /**\n   * Overlay the positions saved for THIS window on top of the current pages.\n   */\n  private _restorePositions() {\n    const raw = localStorage.getItem(this.positionsKey);\n    if (!raw) return;\n    try {\n      const positions: PositionMap = JSON.parse(raw);\n      const pages = this.pagesSubject.value;\n      for (const page of pages) {\n        for (const w of page.widgets) {\n          const pos = positions[`${page.id}:${w.id}`];\n          if (pos) { w.x = pos.x; w.y = pos.y; w.cols = pos.cols; w.rows = pos.rows; }\n        }\n      }\n      this.pagesSubject.next([...pages]);\n    } catch (e) {\n      console.error('[Dashboard] bad positions state', e);\n    }\n  }\n\n  /**\n   * Apply a shared config object (page list + metadata) without touching\n   * this window's saved positions.\n   */\n  private _applyShared(config: any, overwritePositions: boolean) {\n    if (!config?.pages) return;\n    this.pagesSubject.next(config.pages);\n    if (config.activePageId) this.activePageIdSubject.next(config.activePageId);\n    if (!overwritePositions) this._restorePositions();\n  }\n\n  /**\n   * Handle a structural sync event arriving from another window.\n   * Position changes are never sent so we never receive them here.\n   */\n  private _onSyncEvent(event: SyncEvent) {\n    // Work on a shallow clone of pages so Angular detects the change\n    const pages: Page[] = this.pagesSubject.value.map(p => ({\n      ...p, widgets: [...p.widgets]\n    }));\n\n    switch (event.type) {\n\n      case 'PAGE_ADDED':\n        if (!pages.find(p => p.id === event.page.id)) {\n          pages.push({ ...event.page, widgets: [...event.page.widgets] });\n          this.pagesSubject.next(pages);\n          this._saveShared();\n        }\n        break;\n\n      case 'PAGE_REMOVED': {\n        const updated = pages.filter(p => p.id !== event.pageId);\n        if (updated.length !== pages.length) {\n          this.pagesSubject.next(updated);\n          if (this.activePageIdSubject.value === event.pageId) {\n            this.activePageIdSubject.next(updated[0]?.id ?? '');\n          }\n          this._saveShared();\n        }\n        break;\n      }\n\n      case 'WIDGET_ADDED': {\n        const page = pages.find(p => p.id === event.pageId);\n        if (page && !page.widgets.find(w => w.id === event.widget.id)) {\n          page.widgets = [...page.widgets, { ...event.widget }];\n          this.pagesSubject.next(pages);\n          this._saveShared();\n          // Don't touch positions — this window will keep its own layout for\n          // existing widgets; the new one starts at its default position.\n        }\n        break;\n      }\n\n      case 'WIDGET_REMOVED': {\n        const page = pages.find(p => p.id === event.pageId);\n        if (page) {\n          page.widgets = page.widgets.filter(w => w.id !== event.widgetId);\n          this.pagesSubject.next(pages);\n          this._saveShared();\n          this._savePositions();\n        }\n        break;\n      }\n\n      case 'WIDGET_META': {\n        for (const page of pages) {\n          const w = page.widgets.find(w => w.id === event.widgetId);\n          if (w) {\n            Object.assign(w, event.patch);\n            this.pagesSubject.next(pages);\n            this._saveShared();\n            break;\n          }\n        }\n        break;\n      }\n    }\n  }\n\n  private _broadcast(event: SyncEvent) {\n    this.channel?.postMessage(event);\n  }\n}\n"]}