@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.
- package/README.md +283 -48
- package/app/app.module.d.ts +7 -9
- package/app/custom-grid.component.d.ts +93 -0
- package/app/dashboard-state.service.d.ts +54 -33
- package/app/dashboard.component.d.ts +27 -59
- package/app/models.d.ts +18 -38
- package/app/widget-renderer.component.d.ts +5 -32
- package/esm2020/app/app.module.mjs +21 -29
- package/esm2020/app/custom-grid.component.mjs +509 -0
- package/esm2020/app/dashboard-state.service.mjs +228 -158
- package/esm2020/app/dashboard.component.mjs +198 -602
- package/esm2020/app/models.mjs +1 -1
- package/esm2020/app/widget-renderer.component.mjs +41 -121
- package/esm2020/public-api.mjs +5 -4
- package/fesm2015/ogidor-dashboard.mjs +988 -909
- package/fesm2015/ogidor-dashboard.mjs.map +1 -1
- package/fesm2020/ogidor-dashboard.mjs +982 -898
- package/fesm2020/ogidor-dashboard.mjs.map +1 -1
- package/package.json +8 -12
- package/public-api.d.ts +3 -2
|
@@ -1,218 +1,288 @@
|
|
|
1
1
|
import { Injectable } from '@angular/core';
|
|
2
|
-
import { BehaviorSubject
|
|
2
|
+
import { BehaviorSubject } from 'rxjs';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
export class DashboardStateService {
|
|
5
|
-
constructor() {
|
|
6
|
-
this.
|
|
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
|
-
|
|
46
|
-
|
|
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.
|
|
30
|
+
this._applyShared(JSON.parse(shared), false);
|
|
49
31
|
}
|
|
50
32
|
catch (e) {
|
|
51
|
-
console.error('
|
|
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
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
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
|
|
77
|
-
this.pagesSubject.next(
|
|
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(
|
|
65
|
+
this.activePageIdSubject.next(updated[0].id);
|
|
80
66
|
}
|
|
81
|
-
this.
|
|
67
|
+
this._saveShared();
|
|
68
|
+
this._broadcast({ type: 'PAGE_REMOVED', pageId: id });
|
|
82
69
|
}
|
|
83
|
-
|
|
70
|
+
// ── Widget structural actions (synced to all windows) ─────────
|
|
71
|
+
addWidget(widget) {
|
|
84
72
|
const activePage = this.getActivePage();
|
|
85
73
|
if (!activePage)
|
|
86
|
-
|
|
74
|
+
throw new Error('No active page');
|
|
87
75
|
const newWidget = {
|
|
88
76
|
id: `widget-${Date.now()}`,
|
|
89
|
-
|
|
90
|
-
|
|
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.
|
|
97
|
-
this.
|
|
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
|
-
|
|
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
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
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
|
|
129
|
-
if (
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
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
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
this.
|
|
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
|
-
*
|
|
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
|
-
|
|
155
|
-
const
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
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
|
-
|
|
173
|
+
localStorage.setItem(this.positionsKey, JSON.stringify(positions));
|
|
166
174
|
}
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
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
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
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
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
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
|
-
|
|
207
|
-
|
|
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
|
-
|
|
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"]}
|