@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.
- package/README.md +283 -48
- package/app/app.module.d.ts +20 -0
- package/app/custom-grid.component.d.ts +93 -0
- package/app/dashboard-state.service.d.ts +70 -0
- package/app/dashboard.component.d.ts +48 -0
- package/app/models.d.ts +53 -0
- package/app/widget-renderer.component.d.ts +13 -0
- package/esm2020/app/app.module.mjs +57 -0
- package/esm2020/app/custom-grid.component.mjs +509 -0
- package/esm2020/app/dashboard-state.service.mjs +288 -0
- package/esm2020/app/dashboard.component.mjs +299 -0
- package/esm2020/app/models.mjs +2 -0
- package/esm2020/app/widget-renderer.component.mjs +83 -0
- package/esm2020/public-api.mjs +10 -0
- package/fesm2015/ogidor-dashboard.mjs +1233 -0
- package/fesm2015/ogidor-dashboard.mjs.map +1 -0
- package/fesm2020/ogidor-dashboard.mjs +1229 -0
- package/fesm2020/ogidor-dashboard.mjs.map +1 -0
- package/package.json +29 -43
- package/{dist-lib/public-api.d.ts → public-api.d.ts} +3 -2
- package/dist-lib/README.md +0 -103
- package/dist-lib/app/app.module.d.ts +0 -22
- package/dist-lib/app/dashboard-state.service.d.ts +0 -49
- package/dist-lib/app/dashboard.component.d.ts +0 -80
- package/dist-lib/app/models.d.ts +0 -73
- package/dist-lib/app/widget-renderer.component.d.ts +0 -40
- package/dist-lib/esm2020/app/app.module.mjs +0 -65
- package/dist-lib/esm2020/app/dashboard-state.service.mjs +0 -218
- package/dist-lib/esm2020/app/dashboard.component.mjs +0 -703
- package/dist-lib/esm2020/app/models.mjs +0 -2
- package/dist-lib/esm2020/app/widget-renderer.component.mjs +0 -163
- package/dist-lib/esm2020/public-api.mjs +0 -9
- package/dist-lib/fesm2015/ogidor-dashboard.mjs +0 -1154
- package/dist-lib/fesm2015/ogidor-dashboard.mjs.map +0 -1
- package/dist-lib/fesm2020/ogidor-dashboard.mjs +0 -1145
- package/dist-lib/fesm2020/ogidor-dashboard.mjs.map +0 -1
- package/dist-lib/package.json +0 -51
- /package/{dist-lib/esm2020 → esm2020}/ogidor-dashboard.mjs +0 -0
- /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
|