@yuuvis/client-components 3.0.0-beta.21.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (47) hide show
  1. package/README.md +3 -0
  2. package/autocomplete/README.md +3 -0
  3. package/common/README.md +3 -0
  4. package/datepicker/README.md +3 -0
  5. package/fesm2022/yuuvis-client-components-autocomplete.mjs +236 -0
  6. package/fesm2022/yuuvis-client-components-autocomplete.mjs.map +1 -0
  7. package/fesm2022/yuuvis-client-components-common.mjs +1724 -0
  8. package/fesm2022/yuuvis-client-components-common.mjs.map +1 -0
  9. package/fesm2022/yuuvis-client-components-datepicker.mjs +456 -0
  10. package/fesm2022/yuuvis-client-components-datepicker.mjs.map +1 -0
  11. package/fesm2022/yuuvis-client-components-list.mjs +666 -0
  12. package/fesm2022/yuuvis-client-components-list.mjs.map +1 -0
  13. package/fesm2022/yuuvis-client-components-master-details.mjs +136 -0
  14. package/fesm2022/yuuvis-client-components-master-details.mjs.map +1 -0
  15. package/fesm2022/yuuvis-client-components-overflow-hidden.mjs +109 -0
  16. package/fesm2022/yuuvis-client-components-overflow-hidden.mjs.map +1 -0
  17. package/fesm2022/yuuvis-client-components-overflow-menu.mjs +171 -0
  18. package/fesm2022/yuuvis-client-components-overflow-menu.mjs.map +1 -0
  19. package/fesm2022/yuuvis-client-components-popout.mjs +240 -0
  20. package/fesm2022/yuuvis-client-components-popout.mjs.map +1 -0
  21. package/fesm2022/yuuvis-client-components-split-view.mjs +317 -0
  22. package/fesm2022/yuuvis-client-components-split-view.mjs.map +1 -0
  23. package/fesm2022/yuuvis-client-components-widget-grid.mjs +933 -0
  24. package/fesm2022/yuuvis-client-components-widget-grid.mjs.map +1 -0
  25. package/fesm2022/yuuvis-client-components.mjs +18 -0
  26. package/fesm2022/yuuvis-client-components.mjs.map +1 -0
  27. package/lib/assets/i18n/de.json +56 -0
  28. package/lib/assets/i18n/en.json +56 -0
  29. package/list/README.md +3 -0
  30. package/master-details/README.md +3 -0
  31. package/overflow-hidden/README.md +3 -0
  32. package/overflow-menu/README.md +3 -0
  33. package/package.json +67 -0
  34. package/popout/README.md +3 -0
  35. package/split-view/README.md +3 -0
  36. package/types/yuuvis-client-components-autocomplete.d.ts +89 -0
  37. package/types/yuuvis-client-components-common.d.ts +536 -0
  38. package/types/yuuvis-client-components-datepicker.d.ts +94 -0
  39. package/types/yuuvis-client-components-list.d.ts +380 -0
  40. package/types/yuuvis-client-components-master-details.d.ts +69 -0
  41. package/types/yuuvis-client-components-overflow-hidden.d.ts +72 -0
  42. package/types/yuuvis-client-components-overflow-menu.d.ts +89 -0
  43. package/types/yuuvis-client-components-popout.d.ts +106 -0
  44. package/types/yuuvis-client-components-split-view.d.ts +197 -0
  45. package/types/yuuvis-client-components-widget-grid.d.ts +299 -0
  46. package/types/yuuvis-client-components.d.ts +8 -0
  47. package/widget-grid/README.md +48 -0
@@ -0,0 +1,933 @@
1
+ import * as i1 from '@angular/common';
2
+ import { CommonModule } from '@angular/common';
3
+ import * as i0 from '@angular/core';
4
+ import { Injectable, Input, Component, inject, input, effect, output, viewChild, TemplateRef, signal, computed, untracked, NgModule } from '@angular/core';
5
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
6
+ import { MAT_DIALOG_DATA, MatDialogRef, MatDialog } from '@angular/material/dialog';
7
+ import * as i3$2 from '@angular/material/icon';
8
+ import { MatIconModule } from '@angular/material/icon';
9
+ import { TranslateService, TranslatePipe } from '@yuuvis/client-core';
10
+ import { Gridster, DisplayGrid, GridType, GridsterItem } from 'angular-gridster2';
11
+ import * as i3 from 'ng-dynamic-component';
12
+ import { DynamicIoModule } from 'ng-dynamic-component';
13
+ import { ReplaySubject, Subject } from 'rxjs';
14
+ import * as i3$1 from '@angular/forms';
15
+ import { FormControl, ReactiveFormsModule, FormBuilder, Validators } from '@angular/forms';
16
+ import * as i5 from '@angular/material/button';
17
+ import { MatButtonModule } from '@angular/material/button';
18
+ import * as i2 from '@angular/material/list';
19
+ import { MatListModule } from '@angular/material/list';
20
+ import { DialogComponent, ConfirmService } from '@yuuvis/client-components/common';
21
+ import { YmtButtonDirective, YmtIconButtonDirective } from '@yuuvis/material';
22
+ import { SignalComponentIoModule } from 'ng-dynamic-component/signal-component-io';
23
+ import * as i4 from '@angular/material/tooltip';
24
+ import { MatTooltipModule, MatTooltip } from '@angular/material/tooltip';
25
+ import { MatDividerModule } from '@angular/material/divider';
26
+ import * as i1$1 from '@angular/material/form-field';
27
+ import { MatFormFieldModule } from '@angular/material/form-field';
28
+ import * as i2$1 from '@angular/material/input';
29
+ import { MatInputModule } from '@angular/material/input';
30
+ import * as i4$1 from '@angular/material/menu';
31
+ import { MatMenuModule } from '@angular/material/menu';
32
+ import { TranslatePipe as TranslatePipe$1 } from '@ngx-translate/core';
33
+
34
+ class WidgetGridEventService {
35
+ widgetGridEventSource = new ReplaySubject();
36
+ widgetEvents$ = this.widgetGridEventSource.asObservable();
37
+ trigger(evt) {
38
+ this.widgetGridEventSource.next(evt);
39
+ }
40
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridEventService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
41
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridEventService, providedIn: 'root' });
42
+ }
43
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridEventService, decorators: [{
44
+ type: Injectable,
45
+ args: [{
46
+ providedIn: 'root',
47
+ }]
48
+ }] });
49
+
50
+ class NoopComponent {
51
+ widgetConfig;
52
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: NoopComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
53
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "21.2.9", type: NoopComponent, isStandalone: true, selector: "yuv-noop", inputs: { widgetConfig: "widgetConfig" }, ngImport: i0, template: "<svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 0 24 24\" width=\"24px\" fill=\"#000000\"><path d=\"M0 0h24v24H0V0z\" fill=\"none\"/><circle cx=\"15.5\" cy=\"9.5\" r=\"1.5\"/><circle cx=\"8.5\" cy=\"9.5\" r=\"1.5\"/><path d=\"M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8zm0-6c-2.33 0-4.32 1.45-5.12 3.5h1.67c.69-1.19 1.97-2 3.45-2s2.75.81 3.45 2h1.67c-.8-2.05-2.79-3.5-5.12-3.5z\"/></svg>", styles: [":host{height:100%;display:flex;flex-flow:column;align-items:center;justify-content:center;padding:var(--ymt-spacing-m);background-color:var(--panel-background);opacity:.7}:host svg{width:48px;height:48px;fill:var(--text-color-caption)}:host span{margin-top:var(--ymt-spacing-m)}\n"] });
54
+ }
55
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: NoopComponent, decorators: [{
56
+ type: Component,
57
+ args: [{ selector: 'yuv-noop', template: "<svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 0 24 24\" width=\"24px\" fill=\"#000000\"><path d=\"M0 0h24v24H0V0z\" fill=\"none\"/><circle cx=\"15.5\" cy=\"9.5\" r=\"1.5\"/><circle cx=\"8.5\" cy=\"9.5\" r=\"1.5\"/><path d=\"M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8zm0-6c-2.33 0-4.32 1.45-5.12 3.5h1.67c.69-1.19 1.97-2 3.45-2s2.75.81 3.45 2h1.67c-.8-2.05-2.79-3.5-5.12-3.5z\"/></svg>", styles: [":host{height:100%;display:flex;flex-flow:column;align-items:center;justify-content:center;padding:var(--ymt-spacing-m);background-color:var(--panel-background);opacity:.7}:host svg{width:48px;height:48px;fill:var(--text-color-caption)}:host span{margin-top:var(--ymt-spacing-m)}\n"] }]
58
+ }], propDecorators: { widgetConfig: [{
59
+ type: Input
60
+ }] } });
61
+
62
+ /**
63
+ * This service provides the list of widgets that could be added
64
+ * to a widget grid. You could use it to register your own
65
+ * widgets.
66
+ */
67
+ class WidgetGridRegistry {
68
+ translate = inject(TranslateService);
69
+ /**
70
+ * List of pre-registered widgets provided out-of-the-box
71
+ * by the widget grid module
72
+ */
73
+ registeredWidgets = [];
74
+ /**
75
+ * Buckets are collection of widget references.
76
+ * You can put any widget registered gloabally into a bucket. Later on you can grab
77
+ * widgets from a certain bucket. This enables apps to structure their widgets when e.g.
78
+ * using multiple widget grids with their own set of available widgets.
79
+ */
80
+ widgetBuckets = {};
81
+ /**
82
+ * Get the noop component. This component will be rendered in
83
+ * the grid tile if not matching widget could be found in the
84
+ * list of registered widgets. It will show some kind of 'not
85
+ * found' message and provide the controls to remove that none
86
+ * existing component from the widget grid.
87
+ * @returns NoopComponent
88
+ */
89
+ getNoopWidget() {
90
+ return {
91
+ id: 'noop',
92
+ label: this.translate.instant('yuv.widget-grid.widget.noop.label'),
93
+ widgetComponent: NoopComponent
94
+ };
95
+ }
96
+ /**
97
+ * Setup components are the administrative part of a widget. They
98
+ * are used to set up a widget. Not all the widgets will have a setup
99
+ * component.
100
+ * @param widgetName The widgets name
101
+ * @returns The setup component of a widget. Throws error if there
102
+ * is not widget registered with the given name
103
+ */
104
+ getWidgetSetupComponent(widgetName) {
105
+ const widget = this.registeredWidgets.find((w) => w.id === widgetName);
106
+ if (!widget)
107
+ throw Error('Widget setup component not found');
108
+ return widget.setupComponent;
109
+ }
110
+ /**
111
+ * Get the component for a widget. This is the component that will
112
+ * be rendered in a grid tile.
113
+ * @param widgetName The widgets name
114
+ * @returns The widget component or noop component if there is
115
+ * no component registered with the given name
116
+ */
117
+ getWidgetComponent(widgetName) {
118
+ const widget = this.registeredWidgets.find((w) => w.id === widgetName);
119
+ if (!widget)
120
+ console.error('Widget component not found');
121
+ return widget?.widgetComponent || NoopComponent;
122
+ }
123
+ /**
124
+ * Adds a new widget to the list of registered widgets. That way
125
+ * you can create custom widgets that are then available to be
126
+ * added to a users widget grid.
127
+ * @param widget The widget to be registered
128
+ * @param bucket List of buckets to register to. If a bucket does
129
+ * not exist it'll be created.
130
+ */
131
+ registerGridWidget(widget, buckets) {
132
+ const existingWidget = this.registeredWidgets.find((w) => w.id === widget.id);
133
+ if (!existingWidget)
134
+ this.registeredWidgets.push(widget);
135
+ if (buckets?.length) {
136
+ buckets.forEach((b) => {
137
+ this._addToBucket(b, widget.id);
138
+ });
139
+ }
140
+ }
141
+ /**
142
+ * Register a collection of widgets
143
+ * @param widgets The widgets to be registered
144
+ * @param bucket List of buckets to register to. If a bucket does
145
+ * not exist it'll be created.
146
+ */
147
+ registerGridWidgets(widgets, buckets) {
148
+ const alreadyRegisteredWidgetNames = this.registeredWidgets.map((w) => w.id);
149
+ const widgetsToRegister = widgets.filter((w) => !alreadyRegisteredWidgetNames.includes(w.id));
150
+ this.registeredWidgets = [...this.registeredWidgets, ...widgetsToRegister.map((w) => this.#translateLabel(w))];
151
+ if (buckets?.length) {
152
+ buckets.forEach((b) => widgets.forEach((w) => this._addToBucket(b, w.id)));
153
+ }
154
+ }
155
+ #translateLabel(widget) {
156
+ const translatedLabel = this.translate.instant(widget.label);
157
+ return { ...widget, label: translatedLabel && !translatedLabel.startsWith('!missing') ? translatedLabel : widget.label };
158
+ }
159
+ _addToBucket(bucket, widgetName) {
160
+ if (!this.widgetBuckets[bucket])
161
+ this.widgetBuckets[bucket] = [];
162
+ this.widgetBuckets[bucket].push(widgetName);
163
+ }
164
+ removeRegisteredWidget(id) {
165
+ this.registeredWidgets = this.registeredWidgets.filter((w) => w.id !== id);
166
+ }
167
+ clearRegisteredWidget() {
168
+ this.registeredWidgets = [];
169
+ }
170
+ /**
171
+ * Get registered widgets. This list could be narrowed down by a list
172
+ * of buckets. Buckets are lists of registered widgets registered for
173
+ * a certain key (the buckets name).
174
+ * @param buckets name of buckets to restrict the widgets to
175
+ * @returns Array of grid widgets
176
+ */
177
+ getRegisteredWidgets(buckets) {
178
+ if (buckets?.length) {
179
+ // buckets may contain wildcards ...
180
+ // grab all buckets matching the wildcards
181
+ const matchedBuckets = {};
182
+ buckets.forEach((b) => {
183
+ if (b.match(/[*?]/)) {
184
+ // bucket has a pattern
185
+ Object.keys(this.widgetBuckets).forEach((wb) => {
186
+ if (this._wildcardMatch(wb, b))
187
+ matchedBuckets[wb] = 0;
188
+ });
189
+ }
190
+ else
191
+ matchedBuckets[b] = 0;
192
+ });
193
+ const bucketWidgetNames = {};
194
+ Object.keys(matchedBuckets).forEach((b) => this.widgetBuckets[b].forEach((widgetName) => {
195
+ bucketWidgetNames[widgetName] = 0;
196
+ }));
197
+ const widgetNames = Object.keys(bucketWidgetNames);
198
+ return this.registeredWidgets.filter((w) => widgetNames.includes(w.id)).map((w) => this.#translateLabel(w));
199
+ }
200
+ else
201
+ return this.registeredWidgets.map((w) => this.#translateLabel(w));
202
+ }
203
+ _wildcardMatch(text, pattern) {
204
+ const regexPattern = new RegExp('^' + pattern.replace(/\?/g, '.').replace(/\*/g, '.*') + '$');
205
+ return regexPattern.test(text);
206
+ }
207
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridRegistry, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
208
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridRegistry, providedIn: 'root' });
209
+ }
210
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridRegistry, decorators: [{
211
+ type: Injectable,
212
+ args: [{
213
+ providedIn: 'root'
214
+ }]
215
+ }] });
216
+
217
+ class WidgetGridUtils {
218
+ static PREF_FUNCTION = 'Function';
219
+ static PREF_RANGEVALUE = 'RangeValue';
220
+ /**
221
+ * Takes a `WidgetGridItemConfig` and stringifies it. The challenge
222
+ * here is that callback functions should also be stringified.
223
+ * @param gridItemConfig the config object
224
+ */
225
+ static gridConfigStringify(o) {
226
+ return JSON.stringify(o, (key, value) => {
227
+ if (typeof value === 'function') {
228
+ return `/${WidgetGridUtils.PREF_FUNCTION}(${value.toString()})/`;
229
+ }
230
+ return value;
231
+ });
232
+ }
233
+ /**
234
+ * Takes a string stringified with `gridConfigStringify()` and parses it
235
+ * to output a proper WidgetGridItemConfig.
236
+ * @param stringifiedConfig stringified widget grid config
237
+ */
238
+ static gridConfigParse(stringifiedConfig) {
239
+ return JSON.parse(stringifiedConfig, function (key, value) {
240
+ if (typeof value === 'string' &&
241
+ value.startsWith(`/${WidgetGridUtils.PREF_FUNCTION}(`) &&
242
+ // value.startsWith('/Function(') &&
243
+ value.endsWith(')/')) {
244
+ value = value.substring(WidgetGridUtils.PREF_FUNCTION.length + 2, value.length - 2);
245
+ return (0, eval)('(' + value + ')');
246
+ }
247
+ return value;
248
+ });
249
+ }
250
+ static uuid() {
251
+ return WidgetGridUtils._p8() + WidgetGridUtils._p8(true) + WidgetGridUtils._p8(true) + WidgetGridUtils._p8();
252
+ }
253
+ static _p8(s) {
254
+ const p = (Math.random().toString(16) + '000000000').substr(2, 8);
255
+ return s ? `-${p.substr(0, 4)}-${p.substr(4, 4)}` : p;
256
+ }
257
+ }
258
+
259
+ /**
260
+ * Service managing a widget grid. Also includes a set of labels that will be
261
+ * used by the grid and its widgets. Default labels are provided out of the
262
+ * box. But if you want to use custom ones or add i18n you could overwrite
263
+ * them. The grid as well as the widgets will use those labels or fall back
264
+ * to their default ones. If you are developing own widgets you should
265
+ * consider using those labels as well.
266
+ */
267
+ class WidgetGridService {
268
+ widgetGridRegistry;
269
+ widgetGrid = [];
270
+ widgetGridSource = new ReplaySubject();
271
+ widgetGrid$ = this.widgetGridSource.asObservable();
272
+ widgetGridUpdateSource = new Subject();
273
+ /**
274
+ * Emitted when the widget grid has been updated
275
+ */
276
+ widgetGridUpdate$ = this.widgetGridUpdateSource.asObservable();
277
+ addItemSize = {
278
+ rows: 2,
279
+ cols: 3,
280
+ };
281
+ constructor(widgetGridRegistry) {
282
+ this.widgetGridRegistry = widgetGridRegistry;
283
+ }
284
+ setWidgetGrid(gridItemConfig) {
285
+ if (gridItemConfig) {
286
+ // create actual widget grid from widget grid config
287
+ this.widgetGrid = [...gridItemConfig].map((gic) => {
288
+ const cmp = this.widgetGridRegistry.getWidgetComponent(gic.widgetName);
289
+ const gc = {
290
+ ...gic,
291
+ widget: cmp,
292
+ widgetConfigMap: {
293
+ widgetConfig: cmp === NoopComponent
294
+ ? {
295
+ widgetName: gic.widgetName,
296
+ }
297
+ : gic.widgetConfig,
298
+ },
299
+ };
300
+ return gc;
301
+ });
302
+ }
303
+ else {
304
+ this.widgetGrid = [];
305
+ }
306
+ this.widgetGridSource.next(this.widgetGrid);
307
+ }
308
+ /**
309
+ * Update config of an existing widget
310
+ * @param widgetId ID of the widget to be updated
311
+ * @param setupWidgetConfig The updated configuration for that widget
312
+ */
313
+ updateWidget(widgetId, setupWidgetConfig) {
314
+ const gridWidget = this.widgetGrid.find((w) => w.id === widgetId);
315
+ if (gridWidget && setupWidgetConfig) {
316
+ gridWidget.widgetConfig = setupWidgetConfig;
317
+ gridWidget.widgetConfigMap = { widgetConfig: setupWidgetConfig };
318
+ }
319
+ this.widgetGridSource.next(this.widgetGrid);
320
+ this.widgetGridUpdateSource.next(widgetId);
321
+ }
322
+ replaceWidget(widgetId, widgetName, setupWidgetConfig) {
323
+ const gridWidgetIndex = this.widgetGrid.findIndex((w) => w.id === widgetId);
324
+ if (gridWidgetIndex !== -1) {
325
+ const w = {
326
+ ...this.widgetGrid[gridWidgetIndex],
327
+ ...{
328
+ widgetName,
329
+ widgetConfig: setupWidgetConfig || {},
330
+ widget: this.widgetGridRegistry.getWidgetComponent(widgetName),
331
+ widgetConfigMap: { widgetConfig: setupWidgetConfig || {} },
332
+ },
333
+ };
334
+ this.widgetGrid[gridWidgetIndex] = w;
335
+ this.widgetGridSource.next(this.widgetGrid);
336
+ this.widgetGridUpdateSource.next(widgetId);
337
+ }
338
+ }
339
+ /**
340
+ * Add a new grid item to the widget grid.
341
+ * @param widgetName The name of the grid item to be added
342
+ * @param setupWidgetConfig
343
+ */
344
+ addWidget(widgetName, setupWidgetConfig) {
345
+ const gridWidget = {
346
+ id: WidgetGridUtils.uuid(),
347
+ widgetName,
348
+ widgetConfig: setupWidgetConfig,
349
+ widget: this.widgetGridRegistry.getWidgetComponent(widgetName),
350
+ widgetConfigMap: { widgetConfig: setupWidgetConfig },
351
+ x: 0,
352
+ y: 0,
353
+ rows: this.addItemSize.rows,
354
+ cols: this.addItemSize.cols,
355
+ };
356
+ this.widgetGrid.push(gridWidget);
357
+ this.widgetGridSource.next(this.widgetGrid);
358
+ this.widgetGridUpdateSource.next(gridWidget.id);
359
+ }
360
+ removeWidget(gridItemId) {
361
+ const idx = this.widgetGrid.findIndex((w) => w.id === gridItemId);
362
+ this.widgetGrid.splice(idx, 1);
363
+ this.widgetGridSource.next(this.widgetGrid);
364
+ this.widgetGridUpdateSource.next(gridItemId);
365
+ }
366
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridService, deps: [{ token: WidgetGridRegistry }], target: i0.ɵɵFactoryTarget.Injectable });
367
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridService });
368
+ }
369
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetGridService, decorators: [{
370
+ type: Injectable
371
+ }], ctorParameters: () => [{ type: WidgetGridRegistry }] });
372
+
373
+ class WidgetPickerComponent {
374
+ #widgetGridRegistry = inject(WidgetGridRegistry);
375
+ #widgetGridService = inject(WidgetGridService);
376
+ #dialogData = inject(MAT_DIALOG_DATA);
377
+ #dialogRef = inject((MatDialogRef));
378
+ pickerFormControl = new FormControl();
379
+ widgetId;
380
+ selectedWidget;
381
+ widgetConfigMap = {};
382
+ setupWidgetConfig;
383
+ widgetConfigDirty = false;
384
+ widgetConfigState;
385
+ registeredWidgets = [];
386
+ setupWidgetDynamicOutputs = {
387
+ widgetConfigChange: (widgetConfig) => {
388
+ this.onSetupConfigChange(widgetConfig);
389
+ },
390
+ widgetConfigStateChange: (s) => {
391
+ this.widgetConfigState = s || undefined;
392
+ }
393
+ };
394
+ pickerData = input(this.#dialogData.pickerData, ...(ngDevMode ? [{ debugName: "pickerData" }] : /* istanbul ignore next */ []));
395
+ #pickerDataEffect = effect(() => {
396
+ const pd = this.pickerData();
397
+ if (pd) {
398
+ const gridWidget = this.#widgetGridRegistry
399
+ .getRegisteredWidgets()
400
+ .find((w) => w.id === pd.widgetName);
401
+ this.selectedWidget = gridWidget?.setupComponent ? gridWidget : undefined;
402
+ this.widgetConfigMap = pd.widgetConfigMap || {};
403
+ this.widgetId = pd.widgetId;
404
+ }
405
+ }, ...(ngDevMode ? [{ debugName: "#pickerDataEffect" }] : /* istanbul ignore next */ []));
406
+ /**
407
+ * Collection of buckets to load available widgets from. Wildcards are also posssible:
408
+ * `[buckets]="['app.default', '*.public.*', 'app.no?.widgets']"`
409
+ *
410
+ * `*` represents any character 0-n times
411
+ * `?` represents exactly one character
412
+ */
413
+ buckets = input(this.#dialogData.buckets, ...(ngDevMode ? [{ debugName: "buckets" }] : /* istanbul ignore next */ []));
414
+ picked = output();
415
+ canceled = output();
416
+ constructor() {
417
+ this.pickerFormControl.valueChanges.pipe(takeUntilDestroyed()).subscribe((widgets) => {
418
+ if (widgets.length)
419
+ this.pick(widgets[0]);
420
+ });
421
+ }
422
+ pick(widget) {
423
+ if (widget?.setupComponent) {
424
+ // if the selected widget has a setup component, we'll continue ...
425
+ this.selectedWidget = widget?.setupComponent ? widget : undefined;
426
+ }
427
+ else {
428
+ if (this.widgetId) {
429
+ // update existing widget
430
+ this.#widgetGridService.replaceWidget(this.widgetId, widget.id);
431
+ }
432
+ else {
433
+ this.#widgetGridService.addWidget(widget.id);
434
+ }
435
+ this.picked.emit();
436
+ this.#dialogRef.close();
437
+ }
438
+ }
439
+ onSetupConfigChange(setupConfig) {
440
+ this.widgetConfigDirty = !!setupConfig;
441
+ this.setupWidgetConfig = setupConfig;
442
+ }
443
+ setupComponentSave() {
444
+ // TODO: save/emit config
445
+ if (this.widgetId && this.selectedWidget) {
446
+ if (this.pickerData()?.widgetName !== this.selectedWidget.id) {
447
+ this.#widgetGridService.replaceWidget(this.widgetId, this.selectedWidget.id, this.setupWidgetConfig);
448
+ }
449
+ else {
450
+ this.#widgetGridService.updateWidget(this.widgetId, this.setupWidgetConfig);
451
+ }
452
+ }
453
+ else {
454
+ this.#widgetGridService.addWidget(this.selectedWidget.id, this.setupWidgetConfig);
455
+ }
456
+ this.picked.emit();
457
+ this.#dialogRef.close();
458
+ }
459
+ setupComponentCancel() {
460
+ this.reset();
461
+ }
462
+ cancel() {
463
+ this.canceled.emit();
464
+ this.#dialogRef.close();
465
+ }
466
+ reset() {
467
+ this.selectedWidget = undefined;
468
+ this.setupWidgetConfig = undefined;
469
+ }
470
+ ngOnInit() {
471
+ this.registeredWidgets = this.#widgetGridRegistry.getRegisteredWidgets(this.buckets());
472
+ }
473
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetPickerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
474
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.9", type: WidgetPickerComponent, isStandalone: true, selector: "yuv-widget-picker", inputs: { pickerData: { classPropertyName: "pickerData", publicName: "pickerData", isSignal: true, isRequired: false, transformFunction: null }, buckets: { classPropertyName: "buckets", publicName: "buckets", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { picked: "picked", canceled: "canceled" }, ngImport: i0, template: "<yuv-dialog [headertitel]=\"selectedWidget ? selectedWidget.label : ('yuv.widget-grid.widget-picker.title' | translate)\">\n <!-- <header class=\"dark\" [ngClass]=\"{ listing: !selectedWidget }\">\n @if (selectedWidget) {\n <button mat-icon-button (click)=\"reset()\"><mat-icon>back</mat-icon></button>\n }\n <div class=\"title\">{{ selectedWidget ? selectedWidget.label : ('yuv.widget-grid.widget-picker.title' | translate) }}</div>\n <button mat-icon-button (click)=\"cancel()\"><mat-icon>clear</mat-icon></button>\n </header> -->\n\n <main class=\"{{ !selectedWidget ? 'widget-listing' : 'widget-setup' }}\">\n @if (!selectedWidget) {\n <mat-selection-list role=\"list\" [formControl]=\"pickerFormControl\" [multiple]=\"false\" [hideSingleSelectionIndicator]=\"true\">\n <!-- list of avalable widgets -->\n @for (w of registeredWidgets; track $index) {\n <mat-list-option [value]=\"w\">\n {{ w.label }}\n </mat-list-option>\n } @empty {\n <div class=\"empty\">\n {{ 'yuv.widget-grid.widget-picker.empty' | translate }}\n </div>\n }\n </mat-selection-list>\n } @else {\n <div class=\"component\">\n <ng-container\n *ngComponentOutlet=\"selectedWidget!.setupComponent!; ndcDynamicInputs: widgetConfigMap; ndcDynamicOutputs: setupWidgetDynamicOutputs\"\n ></ng-container>\n </div>\n }\n </main>\n <footer>\n @if (selectedWidget) {\n <button ymtButton=\"secondary\" (click)=\"setupComponentCancel()\">{{ 'yuv.widget-grid.button.cancel' | translate }}</button>\n <button ymtButton=\"primary\" [disabled]=\"!widgetConfigDirty || widgetConfigState === 'INVALID'\" (click)=\"setupComponentSave()\">\n {{ 'yuv.widget-grid.button.save' | translate }}\n </button>\n } @else {\n <button ymtButton=\"secondary\" (click)=\"cancel()\">{{ 'yuv.widget-grid.button.cancel' | translate }}</button>\n }\n </footer>\n</yuv-dialog>\n", styles: [":host .empty{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%}:host main.widget-setup{height:100%;display:grid;grid-template-rows:1fr auto;grid-template-columns:1fr;grid-template-areas:\"component\" \"buttons\"}:host main.widget-listing .widget{padding:var(--app-pane-padding);border-bottom:1px solid var(--panel-divider-color);cursor:pointer}:host main.widget-listing .widget:hover{background-color:var(--item-focus-background-color)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletEnvironmentInjector", "ngComponentOutletContent", "ngComponentOutletNgModule"], exportAs: ["ngComponentOutlet"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: DialogComponent, selector: "yuv-dialog", inputs: ["headertitle", "headertitel"] }, { kind: "directive", type: YmtButtonDirective, selector: "button[ymtButton], a[ymtButton]", inputs: ["ymtButton", "disabled", "aria-disabled", "disableRipple", "disabledInteractive", "button-size"] }, { kind: "ngmodule", type: MatListModule }, { kind: "component", type: i2.MatSelectionList, selector: "mat-selection-list", inputs: ["color", "compareWith", "multiple", "hideSingleSelectionIndicator", "disabled"], outputs: ["selectionChange"], exportAs: ["matSelectionList"] }, { kind: "component", type: i2.MatListOption, selector: "mat-list-option", inputs: ["togglePosition", "checkboxPosition", "color", "value", "selected"], outputs: ["selectedChange"], exportAs: ["matListOption"] }, { kind: "ngmodule", type: DynamicIoModule }, { kind: "directive", type: i3.ComponentOutletInjectorDirective, selector: "[ngComponentOutlet]", exportAs: ["ndcComponentOutletInjector"] }, { kind: "directive", type: i3.ComponentOutletIoDirective, selector: "[ngComponentOutletNdcDynamicInputs],[ngComponentOutletNdcDynamicOutputs]", inputs: ["ngComponentOutletNdcDynamicInputs", "ngComponentOutletNdcDynamicOutputs"], exportAs: ["ndcDynamicIo"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i3$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3$1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: SignalComponentIoModule }, { kind: "pipe", type: TranslatePipe, name: "translate" }] });
475
+ }
476
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WidgetPickerComponent, decorators: [{
477
+ type: Component,
478
+ args: [{ selector: 'yuv-widget-picker', imports: [
479
+ CommonModule,
480
+ MatButtonModule,
481
+ TranslatePipe,
482
+ MatIconModule,
483
+ DialogComponent,
484
+ YmtButtonDirective,
485
+ MatListModule,
486
+ DynamicIoModule,
487
+ ReactiveFormsModule,
488
+ SignalComponentIoModule
489
+ ], template: "<yuv-dialog [headertitel]=\"selectedWidget ? selectedWidget.label : ('yuv.widget-grid.widget-picker.title' | translate)\">\n <!-- <header class=\"dark\" [ngClass]=\"{ listing: !selectedWidget }\">\n @if (selectedWidget) {\n <button mat-icon-button (click)=\"reset()\"><mat-icon>back</mat-icon></button>\n }\n <div class=\"title\">{{ selectedWidget ? selectedWidget.label : ('yuv.widget-grid.widget-picker.title' | translate) }}</div>\n <button mat-icon-button (click)=\"cancel()\"><mat-icon>clear</mat-icon></button>\n </header> -->\n\n <main class=\"{{ !selectedWidget ? 'widget-listing' : 'widget-setup' }}\">\n @if (!selectedWidget) {\n <mat-selection-list role=\"list\" [formControl]=\"pickerFormControl\" [multiple]=\"false\" [hideSingleSelectionIndicator]=\"true\">\n <!-- list of avalable widgets -->\n @for (w of registeredWidgets; track $index) {\n <mat-list-option [value]=\"w\">\n {{ w.label }}\n </mat-list-option>\n } @empty {\n <div class=\"empty\">\n {{ 'yuv.widget-grid.widget-picker.empty' | translate }}\n </div>\n }\n </mat-selection-list>\n } @else {\n <div class=\"component\">\n <ng-container\n *ngComponentOutlet=\"selectedWidget!.setupComponent!; ndcDynamicInputs: widgetConfigMap; ndcDynamicOutputs: setupWidgetDynamicOutputs\"\n ></ng-container>\n </div>\n }\n </main>\n <footer>\n @if (selectedWidget) {\n <button ymtButton=\"secondary\" (click)=\"setupComponentCancel()\">{{ 'yuv.widget-grid.button.cancel' | translate }}</button>\n <button ymtButton=\"primary\" [disabled]=\"!widgetConfigDirty || widgetConfigState === 'INVALID'\" (click)=\"setupComponentSave()\">\n {{ 'yuv.widget-grid.button.save' | translate }}\n </button>\n } @else {\n <button ymtButton=\"secondary\" (click)=\"cancel()\">{{ 'yuv.widget-grid.button.cancel' | translate }}</button>\n }\n </footer>\n</yuv-dialog>\n", styles: [":host .empty{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%}:host main.widget-setup{height:100%;display:grid;grid-template-rows:1fr auto;grid-template-columns:1fr;grid-template-areas:\"component\" \"buttons\"}:host main.widget-listing .widget{padding:var(--app-pane-padding);border-bottom:1px solid var(--panel-divider-color);cursor:pointer}:host main.widget-listing .widget:hover{background-color:var(--item-focus-background-color)}\n"] }]
490
+ }], ctorParameters: () => [], propDecorators: { pickerData: [{ type: i0.Input, args: [{ isSignal: true, alias: "pickerData", required: false }] }], buckets: [{ type: i0.Input, args: [{ isSignal: true, alias: "buckets", required: false }] }], picked: [{ type: i0.Output, args: ["picked"] }], canceled: [{ type: i0.Output, args: ["canceled"] }] } });
491
+
492
+ class YuvWidgetGridComponent {
493
+ #dialog = inject(MatDialog);
494
+ #widgetGridService = inject(WidgetGridService);
495
+ #widgetGridEventService = inject(WidgetGridEventService);
496
+ gridster = viewChild(Gridster, ...(ngDevMode ? [{ debugName: "gridster" }] : /* istanbul ignore next */ []));
497
+ widgetPicker = viewChild.required('widgetPicker', { read: TemplateRef });
498
+ options = {
499
+ gridType: GridType.Fit,
500
+ displayGrid: DisplayGrid.None,
501
+ pushItems: false,
502
+ outerMargin: false,
503
+ swap: false,
504
+ draggable: {
505
+ enabled: false,
506
+ ignoreContent: true,
507
+ dragHandleClass: 'dragHandle'
508
+ },
509
+ resizable: {
510
+ enabled: false
511
+ },
512
+ itemChangeCallback: (item, itemComponent) => {
513
+ this.emitChange();
514
+ }
515
+ };
516
+ gridConfig = input(...(ngDevMode ? [undefined, { debugName: "gridConfig" }] : /* istanbul ignore next */ []));
517
+ #gridConfigEffect = effect(() => {
518
+ const cfg = this.gridConfig();
519
+ if (cfg?.rows) {
520
+ this.options.minRows = cfg.rows;
521
+ this.options.maxRows = cfg.rows;
522
+ }
523
+ if (cfg?.columns) {
524
+ this.options.minCols = cfg.columns;
525
+ this.options.maxCols = cfg.columns;
526
+ }
527
+ if (cfg?.gap) {
528
+ this.options.margin = cfg.gap;
529
+ }
530
+ if (cfg?.newItemWidth || cfg?.newItemHeight) {
531
+ this.#widgetGridService.addItemSize = {
532
+ cols: cfg?.newItemWidth || -1, // default value -1 means the grid will autoposition the new item
533
+ rows: cfg?.newItemHeight || -1 // default value -1 means the grid will autoposition the new item
534
+ };
535
+ }
536
+ if (cfg?.gridType) {
537
+ this.options.gridType = cfg.gridType;
538
+ }
539
+ this.gridster()?.api.calculateLayout();
540
+ }, ...(ngDevMode ? [{ debugName: "#gridConfigEffect" }] : /* istanbul ignore next */ []));
541
+ _editMode = signal(false, ...(ngDevMode ? [{ debugName: "_editMode" }] : /* istanbul ignore next */ []));
542
+ /**
543
+ * Whether or not to enable edit mode. In edit mode controls
544
+ * for editing existing tiles and creating new ones are shown.
545
+ * This mode also enables positioning and resizing of the tiles.
546
+ */
547
+ editMode = input(false, ...(ngDevMode ? [{ debugName: "editMode" }] : /* istanbul ignore next */ []));
548
+ #editModeEffect = effect(() => {
549
+ const e = this.editMode();
550
+ this._editMode.set(e);
551
+ this.options.draggable.enabled = !!e;
552
+ this.options.resizable.enabled = !!e;
553
+ this.options.displayGrid = e ? DisplayGrid.Always : DisplayGrid.None;
554
+ // set timeout to ensure that the grid has been redrawn before notifying the widgets about the change
555
+ setTimeout(() => {
556
+ this.gridster()?.api.calculateLayout();
557
+ }, 0);
558
+ }, ...(ngDevMode ? [{ debugName: "#editModeEffect" }] : /* istanbul ignore next */ []));
559
+ gridItemConfig = input(undefined, ...(ngDevMode ? [{ debugName: "gridItemConfig" }] : /* istanbul ignore next */ []));
560
+ #gridItemConfigEffect = effect(() => {
561
+ this.#widgetGridService.setWidgetGrid(this.gridItemConfig());
562
+ }, ...(ngDevMode ? [{ debugName: "#gridItemConfigEffect" }] : /* istanbul ignore next */ []));
563
+ /**
564
+ * Collection of buckets to load available widgets from. Wildcards are also posssible:
565
+ * `[buckets]="['app.default', '*.public.*', 'app.no?.widgets']"`
566
+ *
567
+ * `*` represents any character 0-n times
568
+ * `?` represents exactly one character
569
+ */
570
+ buckets = input(undefined, ...(ngDevMode ? [{ debugName: "buckets" }] : /* istanbul ignore next */ []));
571
+ // @Input() buckets?: string[];
572
+ /**
573
+ * Emitted when the grid has been changed
574
+ */
575
+ gridChange = output();
576
+ gridItemEvent = output();
577
+ /**
578
+ * Emitted when the widget picker is opened or closed in edit mode
579
+ */
580
+ widgetPickerOpen = output();
581
+ widgetGrid = [];
582
+ widgetPickerData;
583
+ constructor() {
584
+ this.#widgetGridService.widgetGrid$.pipe(takeUntilDestroyed()).subscribe((widgetGrid) => {
585
+ this.widgetGrid = widgetGrid;
586
+ });
587
+ this.#widgetGridEventService.widgetEvents$.pipe(takeUntilDestroyed()).subscribe((evt) => {
588
+ this.gridItemEvent.emit(evt);
589
+ });
590
+ this.#widgetGridService.widgetGridUpdate$.pipe(takeUntilDestroyed()).subscribe((_) => {
591
+ this.emitChange();
592
+ });
593
+ }
594
+ openWidgetPicker(item) {
595
+ if (item) {
596
+ // open setup component for selected grid item
597
+ this.widgetPickerData = {
598
+ widgetName: item.widgetName,
599
+ widgetConfigMap: item.widgetConfigMap,
600
+ widgetId: item.id
601
+ };
602
+ }
603
+ else {
604
+ this.widgetPickerData = undefined;
605
+ }
606
+ this.widgetPickerOpen.emit(true);
607
+ this.#dialog
608
+ .open(this.widgetPicker(), {
609
+ data: {
610
+ pickerData: this.widgetPickerData,
611
+ buckets: this.buckets()
612
+ }
613
+ })
614
+ .afterClosed()
615
+ .subscribe(() => this.widgetPickerOpen.emit(false));
616
+ }
617
+ /**
618
+ * Removes a widget from the grid
619
+ * @param item The widget to be removed
620
+ */
621
+ removeItem(item) {
622
+ this.#widgetGridService.removeWidget(item.id);
623
+ }
624
+ /**
625
+ * Add a new widget to the grid by opening the widget picker
626
+ */
627
+ addItem() {
628
+ this.openWidgetPicker();
629
+ }
630
+ emitChange() {
631
+ const mapped = [];
632
+ this.widgetGrid.forEach((i) => {
633
+ const m = { ...i };
634
+ delete m['widget'];
635
+ delete m['widgetConfigMap'];
636
+ mapped.push(m);
637
+ });
638
+ this.gridChange.emit(mapped);
639
+ }
640
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
641
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.9", type: YuvWidgetGridComponent, isStandalone: true, selector: "yuv-widget-grid", inputs: { gridConfig: { classPropertyName: "gridConfig", publicName: "gridConfig", isSignal: true, isRequired: false, transformFunction: null }, editMode: { classPropertyName: "editMode", publicName: "editMode", isSignal: true, isRequired: false, transformFunction: null }, gridItemConfig: { classPropertyName: "gridItemConfig", publicName: "gridItemConfig", isSignal: true, isRequired: false, transformFunction: null }, buckets: { classPropertyName: "buckets", publicName: "buckets", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { gridChange: "gridChange", gridItemEvent: "gridItemEvent", widgetPickerOpen: "widgetPickerOpen" }, host: { properties: { "class.widget-grid-edit": "editMode()" } }, providers: [WidgetGridService], viewQueries: [{ propertyName: "gridster", first: true, predicate: Gridster, descendants: true, isSignal: true }, { propertyName: "widgetPicker", first: true, predicate: ["widgetPicker"], descendants: true, read: TemplateRef, isSignal: true }], ngImport: i0, template: "@if (_editMode()) {\n <button class=\"fab\" mat-fab [attr.aria-label]=\"'yuv.widget-grid.widget.add.label' | translate\" (click)=\"addItem()\" [matTooltip]=\"'yuv.widget-grid.widget.add.tooltip' | translate\">\n <mat-icon>add</mat-icon>\n </button>\n}\n\n@if (widgetGrid.length === 0 && !_editMode()) {\n <div class=\"empty\">\n <!-- <p>{{ 'yuv.widget-grid.empty' | translate }}</p>\n <button ymtButton=\"primary\" (click)=\"_editMode.set(true)\">\n {{'yuv.widget-grid.empty.create' | translate}}</button> -->\n <ng-content select=\".empty\"><p>{{ 'yuv.widget-grid.empty' | translate }}</p></ng-content>\n </div>\n} @else {\n <gridster [options]=\"options\">\n @for (item of widgetGrid; track item.id) {\n <gridster-item [item]=\"item\">\n @if (_editMode()) {\n <div class=\"toolbar\">\n <!-- drag handle -->\n <mat-icon class=\"dragHandle ymt-icon--size-s\">drag_indicator</mat-icon>\n <!-- edit -->\n <button ymt-icon-button icon-button-size=\"small\"\n [matTooltip]=\"'yuv.widget-grid.widget.edit.tooltip' | translate\"\n (click)=\"openWidgetPicker(item)\"><mat-icon class=\"ymt-icon--size-s\">edit</mat-icon></button>\n <!-- remove -->\n <button ymt-icon-button icon-button-size=\"small\"\n [matTooltip]=\"'yuv.widget-grid.widget.remove.tooltip' | translate\"\n (click)=\"removeItem(item)\"><mat-icon class=\"ymt-icon--size-s\">clear</mat-icon></button>\n </div>\n }\n <div class=\"cmp\">\n <ng-container *ngComponentOutlet=\"item.widget; ndcDynamicInputs: item.widgetConfigMap\"></ng-container>\n </div>\n </gridster-item>\n }\n </gridster>\n}\n<ng-template #widgetPicker>\n <yuv-widget-picker [pickerData]=\"widgetPickerData\" [buckets]=\"buckets()\"></yuv-widget-picker>\n</ng-template>\n", styles: [":host{--_widget-grid-toolbar-background: var(--widget-grid-toolbar-background, var(--ymt-surface));--_widget-grid-on-toolbar: var(--widget-grid-on-toolbar, var(--ymt-on-surface));position:relative}:host .fab{position:absolute;inset-inline-end:var(--ymt-spacing-m);inset-block-end:var(--ymt-spacing-m);z-index:100}:host .cmp{height:100%}:host.widget-grid-edit gridster-item{outline:1px solid var(--ymt-outline);outline-offset:-1px}:host .empty{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%;color:var(--ymt-on-surface);font:var(--ymt-font-body-medium)}:host .empty p{padding:var(--ymt-spacing-m);max-width:45ch;text-align:center}:host gridster{background-color:transparent}:host gridster ::ng-deep .gridster-column,:host gridster ::ng-deep .gridster-row{border-color:var(--ymt-outline-variant);transition:none}:host gridster-item{transition:none;background-color:transparent}:host gridster-item .toolbar{position:absolute;z-index:1;background-color:var(--_widget-grid-toolbar-background);color:var(--_widget-grid-on-toolbar);right:0;top:var(--ymt-spacing-xs);padding:var(--ymt-spacing-xs);gap:var(--ymt-spacing-s);display:flex;flex-flow:row nowrap;align-items:center;outline:1px solid var(--ymt-outline);outline-offset:-1px}:host gridster-item .toolbar button{color:currentColor}:host gridster-item .dragHandle{display:flex;flex-flow:column;align-items:center;justify-content:center;padding:0 var(--ymt-spacing-s);cursor:move}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletEnvironmentInjector", "ngComponentOutletContent", "ngComponentOutletNgModule"], exportAs: ["ngComponentOutlet"] }, { kind: "ngmodule", type: DynamicIoModule }, { kind: "directive", type: i3.ComponentOutletInjectorDirective, selector: "[ngComponentOutlet]", exportAs: ["ndcComponentOutletInjector"] }, { kind: "directive", type: i3.ComponentOutletIoDirective, selector: "[ngComponentOutletNdcDynamicInputs],[ngComponentOutletNdcDynamicOutputs]", inputs: ["ngComponentOutletNdcDynamicInputs", "ngComponentOutletNdcDynamicOutputs"], exportAs: ["ndcDynamicIo"] }, { kind: "ngmodule", type: SignalComponentIoModule }, { kind: "component", type: Gridster, selector: "gridster", inputs: ["options"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3$2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i5.MatFabButton, selector: "button[mat-fab], a[mat-fab], button[matFab], a[matFab]", inputs: ["extended"], exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: WidgetPickerComponent, selector: "yuv-widget-picker", inputs: ["pickerData", "buckets"], outputs: ["picked", "canceled"] }, { kind: "component", type: GridsterItem, selector: "gridster-item", inputs: ["item"], outputs: ["itemInit", "itemChange", "itemResize"] }, { kind: "directive", type: YmtIconButtonDirective, selector: "button[ymtIconButton],button[ymt-icon-button],a[ymtIconButton],a[ymt-icon-button]", inputs: ["disabled", "disableRipple", "aria-disabled", "disabledInteractive", "icon-button-size"] }, { kind: "pipe", type: TranslatePipe, name: "translate" }] });
642
+ }
643
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridComponent, decorators: [{
644
+ type: Component,
645
+ args: [{ selector: 'yuv-widget-grid', imports: [
646
+ CommonModule,
647
+ DynamicIoModule,
648
+ SignalComponentIoModule,
649
+ TranslatePipe,
650
+ Gridster,
651
+ MatIconModule,
652
+ MatTooltipModule,
653
+ MatButtonModule,
654
+ WidgetPickerComponent,
655
+ GridsterItem,
656
+ YmtIconButtonDirective,
657
+ MatTooltip
658
+ ], providers: [WidgetGridService], host: { '[class.widget-grid-edit]': 'editMode()' }, template: "@if (_editMode()) {\n <button class=\"fab\" mat-fab [attr.aria-label]=\"'yuv.widget-grid.widget.add.label' | translate\" (click)=\"addItem()\" [matTooltip]=\"'yuv.widget-grid.widget.add.tooltip' | translate\">\n <mat-icon>add</mat-icon>\n </button>\n}\n\n@if (widgetGrid.length === 0 && !_editMode()) {\n <div class=\"empty\">\n <!-- <p>{{ 'yuv.widget-grid.empty' | translate }}</p>\n <button ymtButton=\"primary\" (click)=\"_editMode.set(true)\">\n {{'yuv.widget-grid.empty.create' | translate}}</button> -->\n <ng-content select=\".empty\"><p>{{ 'yuv.widget-grid.empty' | translate }}</p></ng-content>\n </div>\n} @else {\n <gridster [options]=\"options\">\n @for (item of widgetGrid; track item.id) {\n <gridster-item [item]=\"item\">\n @if (_editMode()) {\n <div class=\"toolbar\">\n <!-- drag handle -->\n <mat-icon class=\"dragHandle ymt-icon--size-s\">drag_indicator</mat-icon>\n <!-- edit -->\n <button ymt-icon-button icon-button-size=\"small\"\n [matTooltip]=\"'yuv.widget-grid.widget.edit.tooltip' | translate\"\n (click)=\"openWidgetPicker(item)\"><mat-icon class=\"ymt-icon--size-s\">edit</mat-icon></button>\n <!-- remove -->\n <button ymt-icon-button icon-button-size=\"small\"\n [matTooltip]=\"'yuv.widget-grid.widget.remove.tooltip' | translate\"\n (click)=\"removeItem(item)\"><mat-icon class=\"ymt-icon--size-s\">clear</mat-icon></button>\n </div>\n }\n <div class=\"cmp\">\n <ng-container *ngComponentOutlet=\"item.widget; ndcDynamicInputs: item.widgetConfigMap\"></ng-container>\n </div>\n </gridster-item>\n }\n </gridster>\n}\n<ng-template #widgetPicker>\n <yuv-widget-picker [pickerData]=\"widgetPickerData\" [buckets]=\"buckets()\"></yuv-widget-picker>\n</ng-template>\n", styles: [":host{--_widget-grid-toolbar-background: var(--widget-grid-toolbar-background, var(--ymt-surface));--_widget-grid-on-toolbar: var(--widget-grid-on-toolbar, var(--ymt-on-surface));position:relative}:host .fab{position:absolute;inset-inline-end:var(--ymt-spacing-m);inset-block-end:var(--ymt-spacing-m);z-index:100}:host .cmp{height:100%}:host.widget-grid-edit gridster-item{outline:1px solid var(--ymt-outline);outline-offset:-1px}:host .empty{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%;color:var(--ymt-on-surface);font:var(--ymt-font-body-medium)}:host .empty p{padding:var(--ymt-spacing-m);max-width:45ch;text-align:center}:host gridster{background-color:transparent}:host gridster ::ng-deep .gridster-column,:host gridster ::ng-deep .gridster-row{border-color:var(--ymt-outline-variant);transition:none}:host gridster-item{transition:none;background-color:transparent}:host gridster-item .toolbar{position:absolute;z-index:1;background-color:var(--_widget-grid-toolbar-background);color:var(--_widget-grid-on-toolbar);right:0;top:var(--ymt-spacing-xs);padding:var(--ymt-spacing-xs);gap:var(--ymt-spacing-s);display:flex;flex-flow:row nowrap;align-items:center;outline:1px solid var(--ymt-outline);outline-offset:-1px}:host gridster-item .toolbar button{color:currentColor}:host gridster-item .dragHandle{display:flex;flex-flow:column;align-items:center;justify-content:center;padding:0 var(--ymt-spacing-s);cursor:move}\n"] }]
659
+ }], ctorParameters: () => [], propDecorators: { gridster: [{ type: i0.ViewChild, args: [i0.forwardRef(() => Gridster), { isSignal: true }] }], widgetPicker: [{ type: i0.ViewChild, args: ['widgetPicker', { ...{ read: TemplateRef }, isSignal: true }] }], gridConfig: [{ type: i0.Input, args: [{ isSignal: true, alias: "gridConfig", required: false }] }], editMode: [{ type: i0.Input, args: [{ isSignal: true, alias: "editMode", required: false }] }], gridItemConfig: [{ type: i0.Input, args: [{ isSignal: true, alias: "gridItemConfig", required: false }] }], buckets: [{ type: i0.Input, args: [{ isSignal: true, alias: "buckets", required: false }] }], gridChange: [{ type: i0.Output, args: ["gridChange"] }], gridItemEvent: [{ type: i0.Output, args: ["gridItemEvent"] }], widgetPickerOpen: [{ type: i0.Output, args: ["widgetPickerOpen"] }] } });
660
+
661
+ class WorkspaceEditComponent {
662
+ dialogRef = inject((MatDialogRef));
663
+ workspaceForm = input.required(...(ngDevMode ? [{ debugName: "workspaceForm" }] : /* istanbul ignore next */ []));
664
+ workspaceSubmit = output();
665
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WorkspaceEditComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
666
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "21.2.9", type: WorkspaceEditComponent, isStandalone: true, selector: "yuv-workspace-edit", inputs: { workspaceForm: { classPropertyName: "workspaceForm", publicName: "workspaceForm", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { workspaceSubmit: "workspaceSubmit" }, ngImport: i0, template: "<yuv-dialog [headertitel]=\"'yuv.widget-grid.workspaces.edit.title' | translate\">\n <main>\n <form id=\"workspaceEditForm\" [formGroup]=\"workspaceForm()\" (ngSubmit)=\"workspaceSubmit.emit()\">\n <mat-form-field class=\"yuv-form-field\">\n <mat-label>{{ 'yuv.widget-grid.workspaces.workspace.label' | translate }}</mat-label>\n <input matInput formControlName=\"label\" />\n </mat-form-field>\n </form>\n </main>\n <footer>\n <button ymtButton=\"secondary\" type=\"button\" (click)=\"dialogRef.close()\">\n {{ 'yuv.widget-grid.workspaces.edit.cancel' | translate }}\n </button>\n <button ymtButton=\"primary\" form=\"workspaceEditForm\" type=\"submit\" [disabled]=\"workspaceForm().invalid\">\n {{ 'yuv.widget-grid.workspaces.edit.save' | translate }}\n </button>\n </footer>\n</yuv-dialog>\n", styles: [":host main{padding:var(--ymt-spacing-m)}:host mat-form-field{width:100%}\n"], dependencies: [{ kind: "component", type: DialogComponent, selector: "yuv-dialog", inputs: ["headertitle", "headertitel"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i1$1.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i1$1.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i2$1.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly", "disabledInteractive"], exportAs: ["matInput"] }, { kind: "directive", type: YmtButtonDirective, selector: "button[ymtButton], a[ymtButton]", inputs: ["ymtButton", "disabled", "aria-disabled", "disableRipple", "disabledInteractive", "button-size"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i3$1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i3$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3$1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],[formArray],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3$1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i3$1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "pipe", type: TranslatePipe$1, name: "translate" }] });
667
+ }
668
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: WorkspaceEditComponent, decorators: [{
669
+ type: Component,
670
+ args: [{ selector: 'yuv-workspace-edit', imports: [
671
+ DialogComponent,
672
+ MatFormFieldModule,
673
+ MatInputModule,
674
+ YmtButtonDirective,
675
+ TranslatePipe$1,
676
+ ReactiveFormsModule
677
+ ], template: "<yuv-dialog [headertitel]=\"'yuv.widget-grid.workspaces.edit.title' | translate\">\n <main>\n <form id=\"workspaceEditForm\" [formGroup]=\"workspaceForm()\" (ngSubmit)=\"workspaceSubmit.emit()\">\n <mat-form-field class=\"yuv-form-field\">\n <mat-label>{{ 'yuv.widget-grid.workspaces.workspace.label' | translate }}</mat-label>\n <input matInput formControlName=\"label\" />\n </mat-form-field>\n </form>\n </main>\n <footer>\n <button ymtButton=\"secondary\" type=\"button\" (click)=\"dialogRef.close()\">\n {{ 'yuv.widget-grid.workspaces.edit.cancel' | translate }}\n </button>\n <button ymtButton=\"primary\" form=\"workspaceEditForm\" type=\"submit\" [disabled]=\"workspaceForm().invalid\">\n {{ 'yuv.widget-grid.workspaces.edit.save' | translate }}\n </button>\n </footer>\n</yuv-dialog>\n", styles: [":host main{padding:var(--ymt-spacing-m)}:host mat-form-field{width:100%}\n"] }]
678
+ }], propDecorators: { workspaceForm: [{ type: i0.Input, args: [{ isSignal: true, alias: "workspaceForm", required: true }] }], workspaceSubmit: [{ type: i0.Output, args: ["workspaceSubmit"] }] } });
679
+
680
+ /**
681
+ * Component for managing multiple widget grids in so called workspaces.
682
+ * Each workspace can have its own configuration and widgets. The user can
683
+ * switch between workspaces to use or edit the widgets in the current workspace.
684
+ */
685
+ class YuvWidgetGridWorkspacesComponent {
686
+ #dialog = inject(MatDialog);
687
+ #fb = inject(FormBuilder);
688
+ #confirm = inject(ConfirmService);
689
+ translate = inject(TranslateService);
690
+ workspaceDialogRef;
691
+ #DEFAULT_WORKSPACE_OPTIONS = {
692
+ gridConfig: {
693
+ columns: 10,
694
+ rows: 10
695
+ }
696
+ };
697
+ workspaceForm = this.#fb.group({
698
+ id: [''],
699
+ label: ['', Validators.required]
700
+ });
701
+ get workspaceLabelControl() {
702
+ return this.workspaceForm.get('label');
703
+ }
704
+ options = input(this.#DEFAULT_WORKSPACE_OPTIONS, ...(ngDevMode ? [{ debugName: "options" }] : /* istanbul ignore next */ []));
705
+ _workspaceOptions = computed(() => ({
706
+ gridConfig: {
707
+ ...this.#DEFAULT_WORKSPACE_OPTIONS.gridConfig,
708
+ ...(this.options().gridConfig || {})
709
+ }
710
+ }), ...(ngDevMode ? [{ debugName: "_workspaceOptions" }] : /* istanbul ignore next */ []));
711
+ workspaceConfig = input(undefined, ...(ngDevMode ? [{ debugName: "workspaceConfig" }] : /* istanbul ignore next */ []));
712
+ _workspaceConfig = signal(undefined, ...(ngDevMode ? [{ debugName: "_workspaceConfig" }] : /* istanbul ignore next */ []));
713
+ #workspaceConfigEffect = effect(() => {
714
+ const wsc = this.workspaceConfig();
715
+ untracked(() => {
716
+ this._workspaceConfig.set(wsc);
717
+ this.#updateOriginalWorkspaceConfig();
718
+ if (wsc?.currentWorkspace) {
719
+ this.setWorkspace(wsc.currentWorkspace, true);
720
+ }
721
+ });
722
+ }, ...(ngDevMode ? [{ debugName: "#workspaceConfigEffect" }] : /* istanbul ignore next */ []));
723
+ /**
724
+ * Collection of buckets to load available widgets from. Wildcards are also posssible:
725
+ * `[buckets]="['app.default', '*.public.*', 'app.no?.widgets']"`
726
+ *
727
+ * `*` represents any character 0-n times
728
+ * `?` represents exactly one character
729
+ */
730
+ buckets = input(undefined, ...(ngDevMode ? [{ debugName: "buckets" }] : /* istanbul ignore next */ []));
731
+ configChange = output();
732
+ gridItemEvent = output();
733
+ editModeChange = output();
734
+ editMode = signal(false, ...(ngDevMode ? [{ debugName: "editMode" }] : /* istanbul ignore next */ []));
735
+ // persist the last 'accepted' workspace config to be able to revert changes
736
+ #originalWidgetGridWorkspaceConfig;
737
+ // currently selected workspace
738
+ workspace = signal(undefined, ...(ngDevMode ? [{ debugName: "workspace" }] : /* istanbul ignore next */ []));
739
+ #workspaceEffect = effect(() => {
740
+ const ws = this.workspace();
741
+ this.workspaceLabel.set(ws ? this.getLabel(ws) : '');
742
+ untracked(() => {
743
+ if (ws)
744
+ this.workspaceForm.patchValue({ label: this.workspaceLabel(), id: ws.id });
745
+ this.gridItemConfig.set(ws ? JSON.parse(JSON.stringify(ws.grid)) : undefined);
746
+ });
747
+ }, ...(ngDevMode ? [{ debugName: "#workspaceEffect" }] : /* istanbul ignore next */ []));
748
+ workspaceLabel = signal('', ...(ngDevMode ? [{ debugName: "workspaceLabel" }] : /* istanbul ignore next */ []));
749
+ gridItemConfig = signal(undefined, ...(ngDevMode ? [{ debugName: "gridItemConfig" }] : /* istanbul ignore next */ []));
750
+ getLabel(workspace) {
751
+ return workspace.translateLabel ? this.translate.instant(workspace.label) : workspace.label;
752
+ }
753
+ setWorkspace(id, silent = false) {
754
+ const workspace = this._workspaceConfig()?.workspaces.find((w) => w.id === id);
755
+ if (workspace) {
756
+ this.workspace.set(workspace);
757
+ this._workspaceConfig.update((cfg) => ({
758
+ ...cfg,
759
+ currentWorkspace: id
760
+ }));
761
+ this.#updateOriginalWorkspaceConfig();
762
+ }
763
+ else {
764
+ this.workspace.set(undefined);
765
+ }
766
+ if (!silent)
767
+ this.emitConfigChange();
768
+ }
769
+ openWorkspaceDialog(create, tplRef) {
770
+ if (create) {
771
+ this.workspaceForm.reset();
772
+ }
773
+ this.workspaceDialogRef = this.#dialog.open(tplRef);
774
+ }
775
+ saveWorkspace() {
776
+ let cfg = this._workspaceConfig();
777
+ if (!cfg) {
778
+ cfg = {
779
+ workspaces: []
780
+ };
781
+ }
782
+ const isNewWorkspace = !this.workspaceForm.value.id;
783
+ if (isNewWorkspace) {
784
+ const newWorkspace = {
785
+ id: WidgetGridUtils.uuid(),
786
+ label: this.workspaceForm.value.label,
787
+ grid: []
788
+ };
789
+ this._workspaceConfig.set({
790
+ ...cfg,
791
+ workspaces: [...cfg.workspaces, newWorkspace],
792
+ currentWorkspace: newWorkspace.id
793
+ });
794
+ this.workspace.set(newWorkspace);
795
+ this.workspaceForm.patchValue({ id: newWorkspace.id, label: newWorkspace.label });
796
+ }
797
+ else {
798
+ const existingIndex = cfg.workspaces.findIndex((w) => w.id === this.workspaceForm.value.id);
799
+ if (existingIndex === -1) {
800
+ const updatedWorkspace = cfg.workspaces[existingIndex];
801
+ updatedWorkspace.label = this.workspaceForm.value.label;
802
+ const workspaces = cfg.workspaces;
803
+ workspaces[existingIndex] = updatedWorkspace;
804
+ this._workspaceConfig.set({
805
+ ...cfg,
806
+ workspaces
807
+ });
808
+ }
809
+ else {
810
+ // workspace label may have changed
811
+ cfg.workspaces[existingIndex].label = this.workspaceForm.value.label;
812
+ }
813
+ }
814
+ this.workspaceDialogRef?.close();
815
+ this.emitConfigChange();
816
+ }
817
+ #updateOriginalWorkspaceConfig() {
818
+ this.#originalWidgetGridWorkspaceConfig = WidgetGridUtils.gridConfigStringify(this.workspaceConfig());
819
+ }
820
+ onGridEvent(e) {
821
+ this.gridItemEvent.emit(e);
822
+ }
823
+ onGridChange(grid) {
824
+ const ws = this.workspace();
825
+ if (ws) {
826
+ ws.grid = grid;
827
+ const wsc = this._workspaceConfig();
828
+ const idx = wsc ? wsc.workspaces.findIndex((w) => w.id === ws.id) : -1;
829
+ if (idx > -1) {
830
+ const wgw = this._workspaceConfig()?.workspaces || [];
831
+ wgw[idx] = ws;
832
+ this._workspaceConfig.update((curr) => ({
833
+ ...curr,
834
+ workspaces: wgw
835
+ }));
836
+ }
837
+ }
838
+ }
839
+ // emit current changes and reset original workspace config internally
840
+ persistWorkspaceConfig() {
841
+ this.saveWorkspace();
842
+ // this.#updateOriginalWorkspaceConfig();
843
+ // this.emitConfigChange();
844
+ this.editMode.set(false);
845
+ }
846
+ revertWorkspaceConfig() {
847
+ if (this.#originalWidgetGridWorkspaceConfig) {
848
+ this._workspaceConfig.set(WidgetGridUtils.gridConfigParse(this.#originalWidgetGridWorkspaceConfig));
849
+ }
850
+ const cws = this.workspaceConfig()?.currentWorkspace;
851
+ if (cws)
852
+ this.setWorkspace(cws, true);
853
+ this.editMode.set(false);
854
+ }
855
+ deleteCurrentWorkspace() {
856
+ const wsc = this._workspaceConfig();
857
+ const ws = this.workspace();
858
+ if (ws && wsc) {
859
+ const idx = wsc.workspaces.findIndex((w) => w.id === ws.id);
860
+ if (idx > -1) {
861
+ this.#confirm
862
+ .confirm({
863
+ message: this.translate.instant('yuv.widget-grid.workspaces.workspace.delete.confirm.message', {
864
+ label: ws.label
865
+ })
866
+ })
867
+ .subscribe((confirmed) => {
868
+ if (!confirmed)
869
+ return;
870
+ const workspaces = wsc.workspaces;
871
+ workspaces.splice(idx, 1);
872
+ // pick another workspace if the current one is deleted
873
+ const newCurrentWorkspace = workspaces.length > 0 ? workspaces[0] : undefined;
874
+ this._workspaceConfig.set({
875
+ ...wsc,
876
+ workspaces,
877
+ currentWorkspace: newCurrentWorkspace?.id
878
+ });
879
+ this.workspace.set(newCurrentWorkspace);
880
+ this.emitConfigChange();
881
+ });
882
+ }
883
+ }
884
+ }
885
+ emitConfigChange() {
886
+ this.configChange.emit(this._workspaceConfig());
887
+ }
888
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridWorkspacesComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
889
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "21.2.9", type: YuvWidgetGridWorkspacesComponent, isStandalone: true, selector: "yuv-widget-grid-workspaces", inputs: { options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: false, transformFunction: null }, workspaceConfig: { classPropertyName: "workspaceConfig", publicName: "workspaceConfig", isSignal: true, isRequired: false, transformFunction: null }, buckets: { classPropertyName: "buckets", publicName: "buckets", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { configChange: "configChange", gridItemEvent: "gridItemEvent", editModeChange: "editModeChange" }, ngImport: i0, template: "@let ws = workspace();\n\n<div class=\"toolbar\">\n @if (ws) {\n @let workspaces = _workspaceConfig()?.workspaces || [];\n\n <button\n mat-icon-button\n [matTooltip]=\"'yuv.widget-grid.workspaces.workspace.menu.tooltip' | translate\"\n [disabled]=\"editMode()\"\n [matMenuTriggerFor]=\"workspacePickerMenu\"\n >\n <mat-icon>menu</mat-icon>\n </button>\n\n <mat-menu #workspacePickerMenu=\"matMenu\">\n @for (ws of workspaces; track ws.id) {\n <button mat-menu-item (click)=\"setWorkspace(ws.id)\">{{ getLabel(ws) }}</button>\n }\n <mat-divider></mat-divider>\n <button mat-menu-item (click)=\"openWorkspaceDialog(true, tplWorkspaceEdit)\">{{ 'yuv.widget-grid.workspaces.workspace.menu.create' | translate }}</button>\n </mat-menu>\n\n @if (!editMode()) {\n <h2 class=\"label\">{{ workspaceLabel() }}</h2>\n @if (!ws.preventEdit) {\n <button mat-icon-button [matMenuTriggerFor]=\"workspaceMenu\"><mat-icon>more_vert</mat-icon></button>\n <mat-menu #workspaceMenu=\"matMenu\">\n <button mat-menu-item (click)=\"editMode.set(!editMode())\">{{ 'yuv.widget-grid.workspaces.workspace.menu.edit' | translate }}</button>\n <button mat-menu-item (click)=\"deleteCurrentWorkspace()\">{{ 'yuv.widget-grid.workspaces.workspace.menu.delete' | translate }}</button>\n </mat-menu>\n }\n } @else {\n @if (ws.preventRename || ws.translateLabel) {\n <h2 class=\"label\">{{ workspaceLabel() }}</h2>\n } @else {\n <input type=\"text\" class=\"label\" [formControl]=\"workspaceLabelControl\" />\n }\n <button ymtButton=\"secondary\" (click)=\"revertWorkspaceConfig()\">\n {{ 'yuv.widget-grid.workspaces.editMode.cancel' | translate }}\n </button>\n <button ymtButton=\"primary\" (click)=\"persistWorkspaceConfig()\">\n {{ 'yuv.widget-grid.workspaces.editMode.save' | translate }}\n </button>\n }\n }\n</div>\n\n@if (ws) {\n <yuv-widget-grid\n [gridConfig]=\"_workspaceOptions().gridConfig\"\n [gridItemConfig]=\"gridItemConfig()\"\n [buckets]=\"buckets()\"\n [editMode]=\"editMode()\"\n (gridChange)=\"onGridChange($event)\"\n (gridItemEvent)=\"onGridEvent($event)\"\n >\n <div class=\"empty\">\n <p>{{ 'yuv.widget-grid.empty' | translate }}</p>\n <button ymtButton=\"primary\" (click)=\"editMode.set(true)\">\n {{ 'yuv.widget-grid.empty.create' | translate }}\n </button>\n </div>\n </yuv-widget-grid>\n} @else {\n <div class=\"empty\">\n <p>{{ 'yuv.widget-grid.workspaces.empty' | translate }}</p>\n <button ymtButton=\"primary\" (click)=\"openWorkspaceDialog(true, tplWorkspaceEdit)\">\n {{ 'yuv.widget-grid.workspaces.empty.button.create' | translate }}\n </button>\n </div>\n}\n\n<ng-template #tplWorkspaceEdit>\n <yuv-workspace-edit [workspaceForm]=\"workspaceForm\" (workspaceSubmit)=\"saveWorkspace()\"></yuv-workspace-edit>\n</ng-template>\n", styles: [":host{display:flex;flex-direction:column;height:100%;overflow-y:auto}:host .toolbar{display:flex;flex-direction:row;align-items:center;gap:var(--ymt-spacing-m)}:host .toolbar .label{flex:1;font:var(--ymt-font-headline-medium);letter-spacing:var(--ymt-font-headline-medium-tracking);margin-inline:0;margin-block:.75em;padding-inline:.25em 0;padding-block:.25em}:host .toolbar input.label{border:0;background-color:transparent}:host .toolbar input.label:not([disabled]){outline:1px solid var(--ymt-outline)}:host yuv-widget-grid{flex:1}:host .empty{flex:1;display:flex;flex-direction:column;align-items:center;justify-content:center}\n"], dependencies: [{ kind: "component", type: YuvWidgetGridComponent, selector: "yuv-widget-grid", inputs: ["gridConfig", "editMode", "gridItemConfig", "buckets"], outputs: ["gridChange", "gridItemEvent", "widgetPickerOpen"] }, { kind: "directive", type: YmtButtonDirective, selector: "button[ymtButton], a[ymtButton]", inputs: ["ymtButton", "disabled", "aria-disabled", "disableRipple", "disabledInteractive", "button-size"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i3$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3$1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i5.MatIconButton, selector: "button[mat-icon-button], a[mat-icon-button], button[matIconButton], a[matIconButton]", exportAs: ["matButton", "matAnchor"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3$2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatMenuModule }, { kind: "component", type: i4$1.MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: i4$1.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i4$1.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "ngmodule", type: MatDividerModule }, { kind: "component", type: i2.MatDivider, selector: "mat-divider", inputs: ["vertical", "inset"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: WorkspaceEditComponent, selector: "yuv-workspace-edit", inputs: ["workspaceForm"], outputs: ["workspaceSubmit"] }, { kind: "pipe", type: TranslatePipe, name: "translate" }] });
890
+ }
891
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridWorkspacesComponent, decorators: [{
892
+ type: Component,
893
+ args: [{ selector: 'yuv-widget-grid-workspaces', imports: [
894
+ YuvWidgetGridComponent,
895
+ YmtButtonDirective,
896
+ ReactiveFormsModule,
897
+ MatInputModule,
898
+ MatButtonModule,
899
+ MatIconModule,
900
+ MatFormFieldModule,
901
+ MatMenuModule,
902
+ MatDividerModule,
903
+ MatTooltipModule,
904
+ TranslatePipe,
905
+ WorkspaceEditComponent
906
+ ], template: "@let ws = workspace();\n\n<div class=\"toolbar\">\n @if (ws) {\n @let workspaces = _workspaceConfig()?.workspaces || [];\n\n <button\n mat-icon-button\n [matTooltip]=\"'yuv.widget-grid.workspaces.workspace.menu.tooltip' | translate\"\n [disabled]=\"editMode()\"\n [matMenuTriggerFor]=\"workspacePickerMenu\"\n >\n <mat-icon>menu</mat-icon>\n </button>\n\n <mat-menu #workspacePickerMenu=\"matMenu\">\n @for (ws of workspaces; track ws.id) {\n <button mat-menu-item (click)=\"setWorkspace(ws.id)\">{{ getLabel(ws) }}</button>\n }\n <mat-divider></mat-divider>\n <button mat-menu-item (click)=\"openWorkspaceDialog(true, tplWorkspaceEdit)\">{{ 'yuv.widget-grid.workspaces.workspace.menu.create' | translate }}</button>\n </mat-menu>\n\n @if (!editMode()) {\n <h2 class=\"label\">{{ workspaceLabel() }}</h2>\n @if (!ws.preventEdit) {\n <button mat-icon-button [matMenuTriggerFor]=\"workspaceMenu\"><mat-icon>more_vert</mat-icon></button>\n <mat-menu #workspaceMenu=\"matMenu\">\n <button mat-menu-item (click)=\"editMode.set(!editMode())\">{{ 'yuv.widget-grid.workspaces.workspace.menu.edit' | translate }}</button>\n <button mat-menu-item (click)=\"deleteCurrentWorkspace()\">{{ 'yuv.widget-grid.workspaces.workspace.menu.delete' | translate }}</button>\n </mat-menu>\n }\n } @else {\n @if (ws.preventRename || ws.translateLabel) {\n <h2 class=\"label\">{{ workspaceLabel() }}</h2>\n } @else {\n <input type=\"text\" class=\"label\" [formControl]=\"workspaceLabelControl\" />\n }\n <button ymtButton=\"secondary\" (click)=\"revertWorkspaceConfig()\">\n {{ 'yuv.widget-grid.workspaces.editMode.cancel' | translate }}\n </button>\n <button ymtButton=\"primary\" (click)=\"persistWorkspaceConfig()\">\n {{ 'yuv.widget-grid.workspaces.editMode.save' | translate }}\n </button>\n }\n }\n</div>\n\n@if (ws) {\n <yuv-widget-grid\n [gridConfig]=\"_workspaceOptions().gridConfig\"\n [gridItemConfig]=\"gridItemConfig()\"\n [buckets]=\"buckets()\"\n [editMode]=\"editMode()\"\n (gridChange)=\"onGridChange($event)\"\n (gridItemEvent)=\"onGridEvent($event)\"\n >\n <div class=\"empty\">\n <p>{{ 'yuv.widget-grid.empty' | translate }}</p>\n <button ymtButton=\"primary\" (click)=\"editMode.set(true)\">\n {{ 'yuv.widget-grid.empty.create' | translate }}\n </button>\n </div>\n </yuv-widget-grid>\n} @else {\n <div class=\"empty\">\n <p>{{ 'yuv.widget-grid.workspaces.empty' | translate }}</p>\n <button ymtButton=\"primary\" (click)=\"openWorkspaceDialog(true, tplWorkspaceEdit)\">\n {{ 'yuv.widget-grid.workspaces.empty.button.create' | translate }}\n </button>\n </div>\n}\n\n<ng-template #tplWorkspaceEdit>\n <yuv-workspace-edit [workspaceForm]=\"workspaceForm\" (workspaceSubmit)=\"saveWorkspace()\"></yuv-workspace-edit>\n</ng-template>\n", styles: [":host{display:flex;flex-direction:column;height:100%;overflow-y:auto}:host .toolbar{display:flex;flex-direction:row;align-items:center;gap:var(--ymt-spacing-m)}:host .toolbar .label{flex:1;font:var(--ymt-font-headline-medium);letter-spacing:var(--ymt-font-headline-medium-tracking);margin-inline:0;margin-block:.75em;padding-inline:.25em 0;padding-block:.25em}:host .toolbar input.label{border:0;background-color:transparent}:host .toolbar input.label:not([disabled]){outline:1px solid var(--ymt-outline)}:host yuv-widget-grid{flex:1}:host .empty{flex:1;display:flex;flex-direction:column;align-items:center;justify-content:center}\n"] }]
907
+ }], propDecorators: { options: [{ type: i0.Input, args: [{ isSignal: true, alias: "options", required: false }] }], workspaceConfig: [{ type: i0.Input, args: [{ isSignal: true, alias: "workspaceConfig", required: false }] }], buckets: [{ type: i0.Input, args: [{ isSignal: true, alias: "buckets", required: false }] }], configChange: [{ type: i0.Output, args: ["configChange"] }], gridItemEvent: [{ type: i0.Output, args: ["gridItemEvent"] }], editModeChange: [{ type: i0.Output, args: ["editModeChange"] }] } });
908
+
909
+ const cmp = [
910
+ YuvWidgetGridComponent,
911
+ YuvWidgetGridWorkspacesComponent
912
+ ];
913
+ class YuvWidgetGridModule {
914
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
915
+ static ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridModule, imports: [YuvWidgetGridComponent,
916
+ YuvWidgetGridWorkspacesComponent], exports: [YuvWidgetGridComponent,
917
+ YuvWidgetGridWorkspacesComponent] });
918
+ static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridModule, imports: [cmp] });
919
+ }
920
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "21.2.9", ngImport: i0, type: YuvWidgetGridModule, decorators: [{
921
+ type: NgModule,
922
+ args: [{
923
+ imports: [cmp],
924
+ exports: [cmp]
925
+ }]
926
+ }] });
927
+
928
+ /**
929
+ * Generated bundle index. Do not edit.
930
+ */
931
+
932
+ export { WidgetGridRegistry, WidgetGridService, YuvWidgetGridComponent, YuvWidgetGridModule, YuvWidgetGridWorkspacesComponent };
933
+ //# sourceMappingURL=yuuvis-client-components-widget-grid.mjs.map