@yuuvis/client-framework 3.0.0-beta.21.0 → 3.0.0-beta.21.2

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