@ogidor/dashboard 1.0.16 → 1.0.18

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -2,7 +2,7 @@
2
2
 
3
3
  A lightweight, content-agnostic drag-and-drop dashboard library for Angular 15.
4
4
  Drop it into any project, supply your own card content, and get a fully featured
5
- multi-workspace grid with persistent layout, pop-out windows, and live cross-tab
5
+ multi-workspace grid with pluggable persistence, pop-out windows, and live cross-tab
6
6
  sync — out of the box.
7
7
 
8
8
  ---
@@ -30,7 +30,7 @@ sync — out of the box.
30
30
  - **Multi-Workspace** — unlimited tabbed workspaces, each with its own independent layout.
31
31
  - **Independent Pop-out Layouts** — pop a workspace into its own window; dragging there never affects the main window.
32
32
  - **Structural Sync** — adding, removing, or renaming a widget in any window instantly appears in all open windows via `BroadcastChannel`.
33
- - **Persistent Layout** — state is saved to `localStorage` on every change and restored on reload.
33
+ - **Pluggable Persistence** — localStorage by default, or inject your own DB/API adapter.
34
34
  - **Content-Agnostic Cards** — you own the card body; stamp any Angular component, chart, or markup inside via `ng-template`.
35
35
  - **Fully Themeable** — 36 colour tokens plus font family, controllable via a typed `DashboardTheme` input or CSS custom properties.
36
36
 
@@ -252,6 +252,13 @@ export { WidgetRendererComponent } from '@ogidor/dashboard';
252
252
  export { CustomGridComponent } from '@ogidor/dashboard';
253
253
  export { GridCellDirective } from '@ogidor/dashboard';
254
254
  export { DashboardStateService } from '@ogidor/dashboard';
255
+ export {
256
+ DashboardPersistence,
257
+ LocalStorageDashboardPersistence,
258
+ DASHBOARD_PERSISTENCE,
259
+ DASHBOARD_PERSISTENCE_CONTEXT,
260
+ PositionMap,
261
+ } from '@ogidor/dashboard';
255
262
  export { Widget, Page, DashboardConfig, DashboardTheme } from '@ogidor/dashboard';
256
263
  ```
257
264
 
@@ -505,26 +512,89 @@ this.state.popOutPage(pageId);
505
512
 
506
513
  ## Persisting Layouts
507
514
 
508
- State is saved to `localStorage` automatically on every change.
515
+ By default, state is saved to `localStorage` automatically on every change.
509
516
 
510
517
  | Key | Contents |
511
518
  |---|---|
512
- | `ogidor_shared` | Page list and widget metadata. Shared across all windows. |
513
- | `ogidor_positions_main` | Grid positions for the main window. |
514
- | `ogidor_positions_<pageId>` | Grid positions for each pop-out window. |
519
+ | `ogidor_shared` | Page list and widget metadata for the default dashboard context. |
520
+ | `ogidor_positions_main` | Grid positions for the main window (default context). |
521
+ | `ogidor_positions_<pageId>` | Grid positions for each pop-out window (default context). |
522
+
523
+ For multi-tenant scenarios, provide `DASHBOARD_PERSISTENCE_CONTEXT` with a custom `dashboardId`; keys are namespaced automatically in the default localStorage adapter.
524
+
525
+ ### Use a database/API instead of localStorage
515
526
 
516
- To save and restore via a backend:
527
+ Provide your own implementation of `DashboardPersistence` and override the `DASHBOARD_PERSISTENCE` token.
517
528
 
518
529
  ```typescript
519
- // Save
520
- const json = this.dash.serializeLayout();
521
- await api.save(json);
530
+ import { Injectable, NgModule } from '@angular/core';
531
+ import { HttpClient } from '@angular/common/http';
532
+ import {
533
+ DashboardConfig,
534
+ DashboardModule,
535
+ DashboardPersistence,
536
+ DASHBOARD_PERSISTENCE,
537
+ DASHBOARD_PERSISTENCE_CONTEXT,
538
+ PositionMap,
539
+ } from '@ogidor/dashboard';
540
+
541
+ @Injectable()
542
+ export class HttpDashboardPersistence extends DashboardPersistence {
543
+ constructor(private http: HttpClient) { super(); }
544
+
545
+ loadShared(dashboardId: string) {
546
+ return this.http
547
+ .get<DashboardConfig | null>(`/api/dashboards/${encodeURIComponent(dashboardId)}/shared`)
548
+ .toPromise();
549
+ }
550
+
551
+ saveShared(dashboardId: string, shared: DashboardConfig) {
552
+ return this.http
553
+ .put<void>(`/api/dashboards/${encodeURIComponent(dashboardId)}/shared`, shared)
554
+ .toPromise()
555
+ .then(() => undefined);
556
+ }
557
+
558
+ loadPositions(dashboardId: string, windowId: string) {
559
+ return this.http
560
+ .get<PositionMap | null>(`/api/dashboards/${encodeURIComponent(dashboardId)}/positions/${encodeURIComponent(windowId)}`)
561
+ .toPromise();
562
+ }
563
+
564
+ savePositions(dashboardId: string, windowId: string, positions: PositionMap) {
565
+ return this.http
566
+ .put<void>(`/api/dashboards/${encodeURIComponent(dashboardId)}/positions/${encodeURIComponent(windowId)}`, positions)
567
+ .toPromise()
568
+ .then(() => undefined);
569
+ }
570
+ }
522
571
 
523
- // Restore
524
- const json = await api.load();
525
- this.dash.stateService.loadLayout(JSON.parse(json));
572
+ @NgModule({
573
+ imports: [DashboardModule],
574
+ providers: [
575
+ { provide: DASHBOARD_PERSISTENCE, useClass: HttpDashboardPersistence },
576
+ { provide: DASHBOARD_PERSISTENCE_CONTEXT, useValue: { dashboardId: 'customer-123' } },
577
+ ],
578
+ })
579
+ export class AppModule {}
526
580
  ```
527
581
 
582
+ Recommended backend endpoints:
583
+
584
+ - `GET /api/dashboards/:dashboardId/shared`
585
+ - `PUT /api/dashboards/:dashboardId/shared`
586
+ - `GET /api/dashboards/:dashboardId/positions/:windowId`
587
+ - `PUT /api/dashboards/:dashboardId/positions/:windowId`
588
+
589
+ ### Migration tip
590
+
591
+ If users already have data in localStorage, run a one-time migration in your app:
592
+
593
+ 1. Read `ogidor_shared` and `ogidor_positions_*` from localStorage.
594
+ 2. Upload them to your backend using the new API.
595
+ 3. Set a migration flag (for example `ogidor_migrated=true`).
596
+ 4. Use your HTTP persistence adapter as the source of truth from then on.
597
+
528
598
  ---
529
599
 
530
600
  ## Full Integration Example
@@ -64,7 +64,7 @@ export declare class CustomGridComponent implements OnInit, OnDestroy, OnChanges
64
64
  private boundMouseUp;
65
65
  constructor(zone: NgZone, cdr: ChangeDetectorRef);
66
66
  ngOnInit(): void;
67
- ngOnChanges(_changes: SimpleChanges): void;
67
+ ngOnChanges(changes: SimpleChanges): void;
68
68
  ngOnDestroy(): void;
69
69
  trackByFn(_index: number, item: Widget): string;
70
70
  private get cellWidth();
@@ -87,6 +87,7 @@ export declare class CustomGridComponent implements OnInit, OnDestroy, OnChanges
87
87
  private handleResizeMove;
88
88
  private finalizeResize;
89
89
  private compactAndApply;
90
+ private applyCompactionAndSync;
90
91
  private updateContainerHeight;
91
92
  static ɵfac: i0.ɵɵFactoryDeclaration<CustomGridComponent, never>;
92
93
  static ɵcmp: i0.ɵɵComponentDeclaration<CustomGridComponent, "app-grid", never, { "widgets": "widgets"; "columns": "columns"; "gap": "gap"; "rowHeight": "rowHeight"; "minItemCols": "minItemCols"; "minItemRows": "minItemRows"; }, { "itemChanged": "itemChanged"; "layoutChanged": "layoutChanged"; }, ["cellTemplate"], never, false, never>;
@@ -0,0 +1,37 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ import { DashboardConfig } from './models';
3
+ import * as i0 from "@angular/core";
4
+ export type PositionMap = Record<string, {
5
+ x: number;
6
+ y: number;
7
+ cols: number;
8
+ rows: number;
9
+ }>;
10
+ export interface DashboardPersistenceContext {
11
+ dashboardId: string;
12
+ }
13
+ export declare const DASHBOARD_PERSISTENCE: InjectionToken<DashboardPersistence>;
14
+ export declare const DASHBOARD_PERSISTENCE_CONTEXT: InjectionToken<DashboardPersistenceContext>;
15
+ export declare abstract class DashboardPersistence {
16
+ abstract loadShared(dashboardId: string): Promise<DashboardConfig | null>;
17
+ abstract saveShared(dashboardId: string, shared: DashboardConfig): Promise<void>;
18
+ abstract loadPositions(dashboardId: string, windowId: string): Promise<PositionMap | null>;
19
+ abstract savePositions(dashboardId: string, windowId: string, positions: PositionMap): Promise<void>;
20
+ }
21
+ export declare class LocalStorageDashboardPersistence implements DashboardPersistence {
22
+ loadShared(dashboardId: string): Promise<DashboardConfig | null>;
23
+ saveShared(dashboardId: string, shared: DashboardConfig): Promise<void>;
24
+ loadPositions(dashboardId: string, windowId: string): Promise<PositionMap | null>;
25
+ savePositions(dashboardId: string, windowId: string, positions: PositionMap): Promise<void>;
26
+ private _sharedKey;
27
+ private _positionsKey;
28
+ static ɵfac: i0.ɵɵFactoryDeclaration<LocalStorageDashboardPersistence, never>;
29
+ static ɵprov: i0.ɵɵInjectableDeclaration<LocalStorageDashboardPersistence>;
30
+ }
31
+ export declare function resolveDefaultDashboardId(): string;
32
+ export declare class DashboardPersistenceFacade {
33
+ readonly dashboardId: string;
34
+ constructor(context: DashboardPersistenceContext | null);
35
+ static ɵfac: i0.ɵɵFactoryDeclaration<DashboardPersistenceFacade, [{ optional: true; }]>;
36
+ static ɵprov: i0.ɵɵInjectableDeclaration<DashboardPersistenceFacade>;
37
+ }
@@ -1,21 +1,14 @@
1
1
  import { NgZone, OnDestroy } from '@angular/core';
2
2
  import { Page, Widget } from './models';
3
+ import { DashboardPersistence, DashboardPersistenceFacade, LocalStorageDashboardPersistence } from './dashboard-persistence';
3
4
  import * as i0 from "@angular/core";
4
5
  export declare class DashboardStateService implements OnDestroy {
5
6
  private zone;
6
- /**
7
- * Shared key — stores page list + widget metadata (no positions).
8
- * Read and written by every window.
9
- */
10
- private readonly SHARED_KEY;
11
- /**
12
- * Per-window positions key — stores {pageId:widgetId → {x,y,cols,rows}}.
13
- * Pop-out windows get a key scoped to their pageId so they never overwrite
14
- * the main window or each other.
15
- * main window → ogidor_positions_main
16
- * pop-out #abc → ogidor_positions_abc
17
- */
18
- private readonly positionsKey;
7
+ private persistenceFacade;
8
+ private localPersistence;
9
+ private persistence;
10
+ private readonly windowId;
11
+ private readonly dashboardId;
19
12
  private readonly CHANNEL_NAME;
20
13
  private channel;
21
14
  private readonly initialPages;
@@ -23,7 +16,7 @@ export declare class DashboardStateService implements OnDestroy {
23
16
  private activePageIdSubject;
24
17
  pages$: import("rxjs").Observable<Page[]>;
25
18
  activePageId$: import("rxjs").Observable<string>;
26
- constructor(zone: NgZone);
19
+ constructor(zone: NgZone, persistenceFacade: DashboardPersistenceFacade, localPersistence: LocalStorageDashboardPersistence, persistence: DashboardPersistence | null);
27
20
  ngOnDestroy(): void;
28
21
  getActivePage(): Page | undefined;
29
22
  setActivePage(id: string): void;
@@ -34,37 +27,17 @@ export declare class DashboardStateService implements OnDestroy {
34
27
  }): Widget;
35
28
  updateWidget(widgetId: string, patch: Partial<Omit<Widget, 'id' | 'x' | 'y' | 'cols' | 'rows'>>): void;
36
29
  removeWidget(widgetId: string): void;
37
- /**
38
- * Update a widget's grid position/size.
39
- * Saved only for this window — never broadcast to others.
40
- */
41
30
  updateWidgetPosition(pageId: string, widgetId: string, x: number, y: number, cols: number, rows: number): void;
42
31
  serializeLayout(): string;
43
32
  loadLayout(config: any): void;
44
33
  popOutPage(pageId: string): void;
45
- /**
46
- * Save page list + widget metadata (titles, cardColor, data) — no positions.
47
- */
48
34
  private _saveShared;
49
- /**
50
- * Save this window's grid positions (x, y, cols, rows) per widget.
51
- */
52
35
  private _savePositions;
53
- /**
54
- * Overlay the positions saved for THIS window on top of the current pages.
55
- */
56
- private _restorePositions;
57
- /**
58
- * Apply a shared config object (page list + metadata) without touching
59
- * this window's saved positions.
60
- */
36
+ private _applyPositions;
61
37
  private _applyShared;
62
- /**
63
- * Handle a structural sync event arriving from another window.
64
- * Position changes are never sent so we never receive them here.
65
- */
66
38
  private _onSyncEvent;
67
39
  private _broadcast;
68
- static ɵfac: i0.ɵɵFactoryDeclaration<DashboardStateService, never>;
40
+ private _bootstrapFromPersistence;
41
+ static ɵfac: i0.ɵɵFactoryDeclaration<DashboardStateService, [null, null, null, { optional: true; }]>;
69
42
  static ɵprov: i0.ɵɵInjectableDeclaration<DashboardStateService>;
70
43
  }
@@ -5,6 +5,7 @@ import { DashboardComponent, OverflowActivePipe } from './dashboard.component';
5
5
  import { WidgetRendererComponent } from './widget-renderer.component';
6
6
  import { CustomGridComponent, GridCellDirective } from './custom-grid.component';
7
7
  import { DashboardStateService } from './dashboard-state.service';
8
+ import { DASHBOARD_PERSISTENCE, LocalStorageDashboardPersistence, } from './dashboard-persistence';
8
9
  import * as i0 from "@angular/core";
9
10
  export class DashboardModule {
10
11
  }
@@ -18,7 +19,10 @@ DashboardModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version
18
19
  WidgetRendererComponent,
19
20
  CustomGridComponent,
20
21
  GridCellDirective] });
21
- DashboardModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardModule, providers: [DashboardStateService], imports: [CommonModule] });
22
+ DashboardModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardModule, providers: [
23
+ DashboardStateService,
24
+ { provide: DASHBOARD_PERSISTENCE, useExisting: LocalStorageDashboardPersistence },
25
+ ], imports: [CommonModule] });
22
26
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DashboardModule, decorators: [{
23
27
  type: NgModule,
24
28
  args: [{
@@ -32,7 +36,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
32
36
  imports: [
33
37
  CommonModule,
34
38
  ],
35
- providers: [DashboardStateService],
39
+ providers: [
40
+ DashboardStateService,
41
+ { provide: DASHBOARD_PERSISTENCE, useExisting: LocalStorageDashboardPersistence },
42
+ ],
36
43
  exports: [
37
44
  DashboardComponent,
38
45
  OverflowActivePipe,
@@ -58,4 +65,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
58
65
  bootstrap: [DashboardComponent],
59
66
  }]
60
67
  }] });
61
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hcHAvYXBwLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFFL0MsT0FBTyxFQUFFLGtCQUFrQixFQUFFLGtCQUFrQixFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDL0UsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDdEUsT0FBTyxFQUFFLG1CQUFtQixFQUFFLGlCQUFpQixFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDakYsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sMkJBQTJCLENBQUM7O0FBc0JsRSxNQUFNLE9BQU8sZUFBZTs7NkdBQWYsZUFBZTs4R0FBZixlQUFlLGlCQWxCeEIsa0JBQWtCO1FBQ2xCLGtCQUFrQjtRQUNsQix1QkFBdUI7UUFDdkIsbUJBQW1CO1FBQ25CLGlCQUFpQixhQUdqQixZQUFZLGFBSVosa0JBQWtCO1FBQ2xCLGtCQUFrQjtRQUNsQix1QkFBdUI7UUFDdkIsbUJBQW1CO1FBQ25CLGlCQUFpQjs4R0FHUixlQUFlLGFBVGYsQ0FBQyxxQkFBcUIsQ0FBQyxZQUZoQyxZQUFZOzRGQVdILGVBQWU7a0JBcEIzQixRQUFRO21CQUFDO29CQUNSLFlBQVksRUFBRTt3QkFDWixrQkFBa0I7d0JBQ2xCLGtCQUFrQjt3QkFDbEIsdUJBQXVCO3dCQUN2QixtQkFBbUI7d0JBQ25CLGlCQUFpQjtxQkFDbEI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNQLFlBQVk7cUJBQ2I7b0JBQ0QsU0FBUyxFQUFFLENBQUMscUJBQXFCLENBQUM7b0JBQ2xDLE9BQU8sRUFBRTt3QkFDUCxrQkFBa0I7d0JBQ2xCLGtCQUFrQjt3QkFDbEIsdUJBQXVCO3dCQUN2QixtQkFBbUI7d0JBQ25CLGlCQUFpQjtxQkFDbEI7aUJBQ0Y7O0FBR0Q7OztHQUdHO0FBS0gsTUFBTSxPQUFPLFNBQVM7O3VHQUFULFNBQVM7d0dBQVQsU0FBUyxjQUZSLGtCQUFrQixhQURwQixhQUFhLEVBUFosZUFBZTt3R0FVZixTQUFTLFlBSFYsYUFBYSxFQUFFLGVBQWU7NEZBRzdCLFNBQVM7a0JBSnJCLFFBQVE7bUJBQUM7b0JBQ1IsT0FBTyxFQUFFLENBQUMsYUFBYSxFQUFFLGVBQWUsQ0FBQztvQkFDekMsU0FBUyxFQUFFLENBQUMsa0JBQWtCLENBQUM7aUJBQ2hDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEJyb3dzZXJNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9wbGF0Zm9ybS1icm93c2VyJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5cbmltcG9ydCB7IERhc2hib2FyZENvbXBvbmVudCwgT3ZlcmZsb3dBY3RpdmVQaXBlIH0gZnJvbSAnLi9kYXNoYm9hcmQuY29tcG9uZW50JztcbmltcG9ydCB7IFdpZGdldFJlbmRlcmVyQ29tcG9uZW50IH0gZnJvbSAnLi93aWRnZXQtcmVuZGVyZXIuY29tcG9uZW50JztcbmltcG9ydCB7IEN1c3RvbUdyaWRDb21wb25lbnQsIEdyaWRDZWxsRGlyZWN0aXZlIH0gZnJvbSAnLi9jdXN0b20tZ3JpZC5jb21wb25lbnQnO1xuaW1wb3J0IHsgRGFzaGJvYXJkU3RhdGVTZXJ2aWNlIH0gZnJvbSAnLi9kYXNoYm9hcmQtc3RhdGUuc2VydmljZSc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1xuICAgIERhc2hib2FyZENvbXBvbmVudCxcbiAgICBPdmVyZmxvd0FjdGl2ZVBpcGUsXG4gICAgV2lkZ2V0UmVuZGVyZXJDb21wb25lbnQsXG4gICAgQ3VzdG9tR3JpZENvbXBvbmVudCxcbiAgICBHcmlkQ2VsbERpcmVjdGl2ZSxcbiAgXSxcbiAgaW1wb3J0czogW1xuICAgIENvbW1vbk1vZHVsZSxcbiAgXSxcbiAgcHJvdmlkZXJzOiBbRGFzaGJvYXJkU3RhdGVTZXJ2aWNlXSxcbiAgZXhwb3J0czogW1xuICAgIERhc2hib2FyZENvbXBvbmVudCxcbiAgICBPdmVyZmxvd0FjdGl2ZVBpcGUsXG4gICAgV2lkZ2V0UmVuZGVyZXJDb21wb25lbnQsXG4gICAgQ3VzdG9tR3JpZENvbXBvbmVudCxcbiAgICBHcmlkQ2VsbERpcmVjdGl2ZSxcbiAgXSxcbn0pXG5leHBvcnQgY2xhc3MgRGFzaGJvYXJkTW9kdWxlIHt9XG5cbi8qKlxuICogUm9vdCBtb2R1bGUgZm9yIGxvY2FsIGRlbW8gLyB0ZXN0aW5nIHB1cnBvc2VzIG9ubHkuXG4gKiBMaWJyYXJ5IGNvbnN1bWVycyBpbXBvcnQgRGFzaGJvYXJkTW9kdWxlLCBub3QgQXBwTW9kdWxlLlxuICovXG5ATmdNb2R1bGUoe1xuICBpbXBvcnRzOiBbQnJvd3Nlck1vZHVsZSwgRGFzaGJvYXJkTW9kdWxlXSxcbiAgYm9vdHN0cmFwOiBbRGFzaGJvYXJkQ29tcG9uZW50XSxcbn0pXG5leHBvcnQgY2xhc3MgQXBwTW9kdWxlIHt9XG4iXX0=
68
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hcHAvYXBwLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFFL0MsT0FBTyxFQUFFLGtCQUFrQixFQUFFLGtCQUFrQixFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDL0UsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDdEUsT0FBTyxFQUFFLG1CQUFtQixFQUFFLGlCQUFpQixFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDakYsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDbEUsT0FBTyxFQUNMLHFCQUFxQixFQUNyQixnQ0FBZ0MsR0FDakMsTUFBTSx5QkFBeUIsQ0FBQzs7QUF5QmpDLE1BQU0sT0FBTyxlQUFlOzs2R0FBZixlQUFlOzhHQUFmLGVBQWUsaUJBckJ4QixrQkFBa0I7UUFDbEIsa0JBQWtCO1FBQ2xCLHVCQUF1QjtRQUN2QixtQkFBbUI7UUFDbkIsaUJBQWlCLGFBR2pCLFlBQVksYUFPWixrQkFBa0I7UUFDbEIsa0JBQWtCO1FBQ2xCLHVCQUF1QjtRQUN2QixtQkFBbUI7UUFDbkIsaUJBQWlCOzhHQUdSLGVBQWUsYUFaZjtRQUNULHFCQUFxQjtRQUNyQixFQUFFLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxXQUFXLEVBQUUsZ0NBQWdDLEVBQUU7S0FDbEYsWUFMQyxZQUFZOzRGQWNILGVBQWU7a0JBdkIzQixRQUFRO21CQUFDO29CQUNSLFlBQVksRUFBRTt3QkFDWixrQkFBa0I7d0JBQ2xCLGtCQUFrQjt3QkFDbEIsdUJBQXVCO3dCQUN2QixtQkFBbUI7d0JBQ25CLGlCQUFpQjtxQkFDbEI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNQLFlBQVk7cUJBQ2I7b0JBQ0QsU0FBUyxFQUFFO3dCQUNULHFCQUFxQjt3QkFDckIsRUFBRSxPQUFPLEVBQUUscUJBQXFCLEVBQUUsV0FBVyxFQUFFLGdDQUFnQyxFQUFFO3FCQUNsRjtvQkFDRCxPQUFPLEVBQUU7d0JBQ1Asa0JBQWtCO3dCQUNsQixrQkFBa0I7d0JBQ2xCLHVCQUF1Qjt3QkFDdkIsbUJBQW1CO3dCQUNuQixpQkFBaUI7cUJBQ2xCO2lCQUNGOztBQUdEOzs7R0FHRztBQUtILE1BQU0sT0FBTyxTQUFTOzt1R0FBVCxTQUFTO3dHQUFULFNBQVMsY0FGUixrQkFBa0IsYUFEcEIsYUFBYSxFQVBaLGVBQWU7d0dBVWYsU0FBUyxZQUhWLGFBQWEsRUFBRSxlQUFlOzRGQUc3QixTQUFTO2tCQUpyQixRQUFRO21CQUFDO29CQUNSLE9BQU8sRUFBRSxDQUFDLGFBQWEsRUFBRSxlQUFlLENBQUM7b0JBQ3pDLFNBQVMsRUFBRSxDQUFDLGtCQUFrQixDQUFDO2lCQUNoQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBCcm93c2VyTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvcGxhdGZvcm0tYnJvd3Nlcic7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuXG5pbXBvcnQgeyBEYXNoYm9hcmRDb21wb25lbnQsIE92ZXJmbG93QWN0aXZlUGlwZSB9IGZyb20gJy4vZGFzaGJvYXJkLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBXaWRnZXRSZW5kZXJlckNvbXBvbmVudCB9IGZyb20gJy4vd2lkZ2V0LXJlbmRlcmVyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBDdXN0b21HcmlkQ29tcG9uZW50LCBHcmlkQ2VsbERpcmVjdGl2ZSB9IGZyb20gJy4vY3VzdG9tLWdyaWQuY29tcG9uZW50JztcbmltcG9ydCB7IERhc2hib2FyZFN0YXRlU2VydmljZSB9IGZyb20gJy4vZGFzaGJvYXJkLXN0YXRlLnNlcnZpY2UnO1xuaW1wb3J0IHtcbiAgREFTSEJPQVJEX1BFUlNJU1RFTkNFLFxuICBMb2NhbFN0b3JhZ2VEYXNoYm9hcmRQZXJzaXN0ZW5jZSxcbn0gZnJvbSAnLi9kYXNoYm9hcmQtcGVyc2lzdGVuY2UnO1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtcbiAgICBEYXNoYm9hcmRDb21wb25lbnQsXG4gICAgT3ZlcmZsb3dBY3RpdmVQaXBlLFxuICAgIFdpZGdldFJlbmRlcmVyQ29tcG9uZW50LFxuICAgIEN1c3RvbUdyaWRDb21wb25lbnQsXG4gICAgR3JpZENlbGxEaXJlY3RpdmUsXG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGUsXG4gIF0sXG4gIHByb3ZpZGVyczogW1xuICAgIERhc2hib2FyZFN0YXRlU2VydmljZSxcbiAgICB7IHByb3ZpZGU6IERBU0hCT0FSRF9QRVJTSVNURU5DRSwgdXNlRXhpc3Rpbmc6IExvY2FsU3RvcmFnZURhc2hib2FyZFBlcnNpc3RlbmNlIH0sXG4gIF0sXG4gIGV4cG9ydHM6IFtcbiAgICBEYXNoYm9hcmRDb21wb25lbnQsXG4gICAgT3ZlcmZsb3dBY3RpdmVQaXBlLFxuICAgIFdpZGdldFJlbmRlcmVyQ29tcG9uZW50LFxuICAgIEN1c3RvbUdyaWRDb21wb25lbnQsXG4gICAgR3JpZENlbGxEaXJlY3RpdmUsXG4gIF0sXG59KVxuZXhwb3J0IGNsYXNzIERhc2hib2FyZE1vZHVsZSB7fVxuXG4vKipcbiAqIFJvb3QgbW9kdWxlIGZvciBsb2NhbCBkZW1vIC8gdGVzdGluZyBwdXJwb3NlcyBvbmx5LlxuICogTGlicmFyeSBjb25zdW1lcnMgaW1wb3J0IERhc2hib2FyZE1vZHVsZSwgbm90IEFwcE1vZHVsZS5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgaW1wb3J0czogW0Jyb3dzZXJNb2R1bGUsIERhc2hib2FyZE1vZHVsZV0sXG4gIGJvb3RzdHJhcDogW0Rhc2hib2FyZENvbXBvbmVudF0sXG59KVxuZXhwb3J0IGNsYXNzIEFwcE1vZHVsZSB7fVxuIl19
@@ -140,15 +140,23 @@ export class CustomGridComponent {
140
140
  this.boundMouseUp = this.onMouseUp.bind(this);
141
141
  }
142
142
  ngOnInit() {
143
- this.compactAndApply();
143
+ this.applyCompactionAndSync();
144
144
  this.updateContainerHeight();
145
145
  this.zone.runOutsideAngular(() => {
146
146
  window.addEventListener('mousemove', this.boundMouseMove);
147
147
  window.addEventListener('mouseup', this.boundMouseUp);
148
148
  });
149
149
  }
150
- ngOnChanges(_changes) {
150
+ ngOnChanges(changes) {
151
+ if (this.dragging || this.resizing) {
152
+ this.updateContainerHeight();
153
+ return;
154
+ }
155
+ if (changes['widgets'] || changes['columns']) {
156
+ this.applyCompactionAndSync();
157
+ }
151
158
  this.updateContainerHeight();
159
+ this.cdr.markForCheck();
152
160
  }
153
161
  ngOnDestroy() {
154
162
  window.removeEventListener('mousemove', this.boundMouseMove);
@@ -367,15 +375,32 @@ export class CustomGridComponent {
367
375
  // ── Utilities ──
368
376
  compactAndApply() {
369
377
  if (!this.widgets?.length)
370
- return;
378
+ return [];
371
379
  const compacted = GridEngine.compact(this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })), this.columns);
380
+ const changedWidgets = [];
372
381
  for (const rw of compacted) {
373
382
  const w = this.widgets.find(ww => ww.id === rw.id);
374
- if (w) {
383
+ if (!w)
384
+ continue;
385
+ const changed = w.x !== rw.x || w.y !== rw.y || w.cols !== rw.cols || w.rows !== rw.rows;
386
+ if (changed) {
375
387
  w.x = rw.x;
376
388
  w.y = rw.y;
389
+ w.cols = rw.cols;
390
+ w.rows = rw.rows;
391
+ changedWidgets.push(w);
377
392
  }
378
393
  }
394
+ return changedWidgets;
395
+ }
396
+ applyCompactionAndSync() {
397
+ const changedWidgets = this.compactAndApply();
398
+ if (!changedWidgets.length)
399
+ return;
400
+ for (const widget of changedWidgets) {
401
+ this.itemChanged.emit(widget);
402
+ }
403
+ this.layoutChanged.emit(this.widgets);
379
404
  }
380
405
  updateContainerHeight() {
381
406
  if (!this.widgets?.length) {
@@ -506,4 +531,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
506
531
  type: ContentChild,
507
532
  args: [GridCellDirective, { read: TemplateRef }]
508
533
  }] } });
509
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-grid.component.js","sourceRoot":"","sources":["../../../src/app/custom-grid.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, Output, EventEmitter, ElementRef,\n  ViewChild, OnDestroy, OnInit, NgZone, ChangeDetectorRef,\n  ContentChild, TemplateRef,\n  Directive, ChangeDetectionStrategy, OnChanges, SimpleChanges\n} from '@angular/core';\nimport { Widget } from './models';\n\n/**\n * Directive applied to the template that should be stamped inside each grid cell.\n * Usage:\n *   <app-grid [widgets]=\"...\">\n *     <ng-template gridCell let-widget=\"widget\">\n *       <app-widget-renderer [widget]=\"widget\" ...></app-widget-renderer>\n *     </ng-template>\n *   </app-grid>\n */\n@Directive({ selector: '[gridCell]' })\nexport class GridCellDirective {\n  constructor(public templateRef: TemplateRef<{ widget: Widget }>) {}\n}\n\n/* ═══════════════════════════════════════════════════════════════════════\n   GRID ENGINE — pure data, zero DOM\n   Handles collision detection, resolution, and compaction.\n   Inspired by react-grid-layout / gridster algorithms.\n   ═══════════════════════════════════════════════════════════════════════ */\n\ninterface GridRect {\n  id: string;\n  x: number;\n  y: number;\n  cols: number;\n  rows: number;\n}\n\nclass GridEngine {\n\n  /**\n   * Check if two rects overlap.\n   */\n  static collides(a: GridRect, b: GridRect): boolean {\n    return !(\n      a.x + a.cols <= b.x ||\n      b.x + b.cols <= a.x ||\n      a.y + a.rows <= b.y ||\n      b.y + b.rows <= a.y\n    );\n  }\n\n  /**\n   * Get ALL widgets that collide with `rect`.\n   */\n  static getAllCollisions(widgets: GridRect[], rect: GridRect): GridRect[] {\n    return widgets.filter(w => w.id !== rect.id && GridEngine.collides(w, rect));\n  }\n\n  /**\n   * Sort widgets top-to-bottom, left-to-right.\n   */\n  static sortByPosition(widgets: GridRect[]): GridRect[] {\n    return [...widgets].sort((a, b) => a.y - b.y || a.x - b.x);\n  }\n\n  /**\n   * Move a widget to (x, y) and push any colliding widgets downward.\n   * Returns the full list with resolved positions.\n   */\n  static moveWidget(\n    allWidgets: GridRect[],\n    widget: GridRect,\n    newX: number,\n    newY: number,\n    columns: number\n  ): GridRect[] {\n    const widgets = allWidgets.map(w => ({ ...w }));\n    const moving = widgets.find(w => w.id === widget.id);\n    if (!moving) return widgets;\n\n    moving.x = Math.max(0, Math.min(newX, columns - moving.cols));\n    moving.y = Math.max(0, newY);\n\n    const sorted = GridEngine.sortByPosition(widgets);\n    GridEngine.resolveCollisions(sorted, moving);\n    return GridEngine.compact(sorted, columns);\n  }\n\n  /**\n   * Resize a widget and push colliding widgets down.\n   */\n  static resizeWidget(\n    allWidgets: GridRect[],\n    widget: GridRect,\n    newCols: number,\n    newRows: number,\n    columns: number\n  ): GridRect[] {\n    const widgets = allWidgets.map(w => ({ ...w }));\n    const resizing = widgets.find(w => w.id === widget.id);\n    if (!resizing) return widgets;\n\n    resizing.cols = Math.max(1, Math.min(newCols, columns - resizing.x));\n    resizing.rows = Math.max(1, newRows);\n\n    const sorted = GridEngine.sortByPosition(widgets);\n    GridEngine.resolveCollisions(sorted, resizing);\n    return GridEngine.compact(sorted, columns);\n  }\n\n  /**\n   * Push all widgets that collide with `movedWidget` downward, recursively.\n   */\n  private static resolveCollisions(widgets: GridRect[], movedWidget: GridRect): void {\n    const collisions = GridEngine.getAllCollisions(widgets, movedWidget);\n    for (const collider of collisions) {\n      collider.y = movedWidget.y + movedWidget.rows;\n      GridEngine.resolveCollisions(widgets, collider);\n    }\n  }\n\n  /**\n   * Compact the grid: move every widget as far up as possible without overlapping.\n   */\n  static compact(widgets: GridRect[], _columns: number): GridRect[] {\n    const sorted = GridEngine.sortByPosition(widgets);\n    const placed: GridRect[] = [];\n\n    for (const widget of sorted) {\n      widget.y = GridEngine.findCompactY(placed, widget);\n      placed.push(widget);\n    }\n\n    return placed;\n  }\n\n  /**\n   * Find the highest Y position a widget can occupy without overlapping any already-placed widget.\n   */\n  private static findCompactY(placed: GridRect[], widget: GridRect): number {\n    let y = 0;\n    while (true) {\n      const test: GridRect = { ...widget, y };\n      const collision = placed.find(p => GridEngine.collides(p, test));\n      if (!collision) return y;\n      y = collision.y + collision.rows;\n    }\n  }\n\n  /**\n   * Compute the total number of rows the grid needs.\n   */\n  static getGridHeight(widgets: GridRect[]): number {\n    if (widgets.length === 0) return 0;\n    return Math.max(...widgets.map(w => w.y + w.rows));\n  }\n}\n\n/* ═══════════════════════════════════════════════════════════════════════\n   CUSTOM GRID COMPONENT\n   ═══════════════════════════════════════════════════════════════════════ */\n\n@Component({\n  selector: 'app-grid',\n  template: `\n    <div\n      #gridContainer\n      class=\"grid-container\"\n      [style.height.px]=\"containerHeight\"\n    >\n      <!-- Placeholder preview -->\n      <div\n        class=\"grid-placeholder\"\n        *ngIf=\"placeholder\"\n        [style.left.px]=\"placeholder.left\"\n        [style.top.px]=\"placeholder.top\"\n        [style.width.px]=\"placeholder.width\"\n        [style.height.px]=\"placeholder.height\"\n      ></div>\n\n      <!-- Widget items -->\n      <div\n        *ngFor=\"let widget of widgets; trackBy: trackByFn\"\n        class=\"grid-item\"\n        [class.is-dragging]=\"dragging?.id === widget.id\"\n        [class.is-resizing]=\"resizing?.id === widget.id\"\n        [style.left.px]=\"getItemLeft(widget)\"\n        [style.top.px]=\"getItemTop(widget)\"\n        [style.width.px]=\"getItemWidth(widget)\"\n        [style.height.px]=\"getItemHeight(widget)\"\n        [style.z-index]=\"(dragging?.id === widget.id || resizing?.id === widget.id) ? 100 : 1\"\n        (mousedown)=\"onDragStart($event, widget)\"\n      >\n        <!-- Content stamped from parent template, context carries the widget -->\n        <ng-container\n          *ngIf=\"cellTemplate\"\n          [ngTemplateOutlet]=\"cellTemplate\"\n          [ngTemplateOutletContext]=\"{ widget: widget }\"\n        ></ng-container>\n\n        <!-- Resize handle -->\n        <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, widget)\">\n          <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\">\n            <path d=\"M11 1v10H1\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n            <path d=\"M11 5v6H5\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n            <path d=\"M11 9v2H9\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n          </svg>\n        </div>\n      </div>\n    </div>\n  `,\n  styles: [`\n    :host { display: block; width: 100%; height: 100%; }\n\n    .grid-container {\n      position: relative;\n      width: 100%;\n      min-height: 100%;\n      transition: height 0.25s ease;\n    }\n\n    .grid-placeholder {\n      position: absolute;\n      background: var(--dash-placeholder-color, var(--dash-accent-color, #0a84ff));\n      opacity: 0.12;\n      border-radius: 24px;\n      border: 2px dashed var(--dash-placeholder-color, var(--dash-accent-color, #0a84ff));\n      transition: left 0.15s ease, top 0.15s ease, width 0.15s ease, height 0.15s ease;\n      pointer-events: none;\n      z-index: 0;\n    }\n\n    .grid-item {\n      position: absolute;\n      display: flex;\n      flex-direction: column;\n      transition: left 0.25s ease, top 0.25s ease, width 0.25s ease, height 0.25s ease;\n      border-radius: 24px;\n      overflow: hidden;\n    }\n\n    .grid-item.is-dragging,\n    .grid-item.is-resizing {\n      transition: none !important;\n      opacity: 0.88;\n      overflow: visible;\n    }\n\n    /* Resize Handle */\n    .resize-handle {\n      position: absolute;\n      right: 4px;\n      bottom: 4px;\n      width: 22px;\n      height: 22px;\n      cursor: nwse-resize;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      color: var(--dash-resize-handle-color, rgba(255,255,255,0.25));\n      border-radius: 6px;\n      transition: color 0.2s, background 0.2s;\n      z-index: 20;\n    }\n    .resize-handle:hover {\n      color: var(--dash-accent-color, #0a84ff);\n      background: rgba(255,255,255,0.06);\n    }\n  `],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class CustomGridComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() widgets: Widget[] = [];\n  @Input() columns: number = 12;\n  @Input() gap: number = 16;\n  @Input() rowHeight: number = 80;\n  @Input() minItemCols: number = 1;\n  @Input() minItemRows: number = 1;\n\n  @Output() itemChanged = new EventEmitter<Widget>();\n  @Output() layoutChanged = new EventEmitter<Widget[]>();\n\n  @ViewChild('gridContainer', { static: true }) gridContainer!: ElementRef<HTMLDivElement>;\n\n  /** The ng-template decorated with gridCell from the parent */\n  @ContentChild(GridCellDirective, { read: TemplateRef }) cellTemplate!: TemplateRef<{ widget: Widget }>;\n\n  placeholder: { left: number; top: number; width: number; height: number } | null = null;\n  containerHeight = 400;\n\n  dragging: {\n    id: string;\n    offsetX: number;\n    offsetY: number;\n    currentPixelX: number;\n    currentPixelY: number;\n  } | null = null;\n\n  resizing: {\n    id: string;\n    startMouseX: number;\n    startMouseY: number;\n    startCols: number;\n    startRows: number;\n    currentPixelW: number;\n    currentPixelH: number;\n  } | null = null;\n\n  private previewWidgets: GridRect[] = [];\n  private boundMouseMove = this.onMouseMove.bind(this);\n  private boundMouseUp = this.onMouseUp.bind(this);\n\n  constructor(private zone: NgZone, private cdr: ChangeDetectorRef) {}\n\n  ngOnInit() {\n    this.compactAndApply();\n    this.updateContainerHeight();\n    this.zone.runOutsideAngular(() => {\n      window.addEventListener('mousemove', this.boundMouseMove);\n      window.addEventListener('mouseup', this.boundMouseUp);\n    });\n  }\n\n  ngOnChanges(_changes: SimpleChanges) {\n    this.updateContainerHeight();\n  }\n\n  ngOnDestroy() {\n    window.removeEventListener('mousemove', this.boundMouseMove);\n    window.removeEventListener('mouseup', this.boundMouseUp);\n  }\n\n  trackByFn(_index: number, item: Widget) { return item.id; }\n\n  // ── Geometry ──\n\n  private get cellWidth(): number {\n    const containerW = this.gridContainer?.nativeElement?.clientWidth ?? 800;\n    return (containerW - (this.columns - 1) * this.gap) / this.columns;\n  }\n\n  private colToPixel(col: number): number {\n    return col * (this.cellWidth + this.gap);\n  }\n\n  private rowToPixel(row: number): number {\n    return row * (this.rowHeight + this.gap);\n  }\n\n  private pixelToCol(px: number): number {\n    return Math.round(px / (this.cellWidth + this.gap));\n  }\n\n  private pixelToRow(px: number): number {\n    return Math.round(px / (this.rowHeight + this.gap));\n  }\n\n  private colsToPixelWidth(cols: number): number {\n    return cols * this.cellWidth + (cols - 1) * this.gap;\n  }\n\n  private rowsToPixelHeight(rows: number): number {\n    return rows * this.rowHeight + (rows - 1) * this.gap;\n  }\n\n  getItemLeft(w: Widget): number {\n    return this.dragging?.id === w.id ? this.dragging.currentPixelX : this.colToPixel(w.x);\n  }\n\n  getItemTop(w: Widget): number {\n    return this.dragging?.id === w.id ? this.dragging.currentPixelY : this.rowToPixel(w.y);\n  }\n\n  getItemWidth(w: Widget): number {\n    return this.resizing?.id === w.id ? this.resizing.currentPixelW : this.colsToPixelWidth(w.cols);\n  }\n\n  getItemHeight(w: Widget): number {\n    return this.resizing?.id === w.id ? this.resizing.currentPixelH : this.rowsToPixelHeight(w.rows);\n  }\n\n  // ── Drag ──\n\n  onDragStart(event: MouseEvent, widget: Widget) {\n    const target = event.target as HTMLElement;\n    if (target.closest('button, input, select, textarea, a, .resize-handle')) return;\n    event.preventDefault();\n    event.stopPropagation();\n\n    const rect = this.gridContainer.nativeElement.getBoundingClientRect();\n    const itemLeft = this.colToPixel(widget.x);\n    const itemTop = this.rowToPixel(widget.y);\n\n    this.dragging = {\n      id: widget.id,\n      offsetX: event.clientX - rect.left - itemLeft,\n      offsetY: event.clientY - rect.top - itemTop,\n      currentPixelX: itemLeft,\n      currentPixelY: itemTop,\n    };\n    this.previewWidgets = this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }));\n    this.placeholder = {\n      left: itemLeft, top: itemTop,\n      width: this.colsToPixelWidth(widget.cols),\n      height: this.rowsToPixelHeight(widget.rows),\n    };\n    this.cdr.markForCheck();\n  }\n\n  onResizeStart(event: MouseEvent, widget: Widget) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.resizing = {\n      id: widget.id,\n      startMouseX: event.clientX,\n      startMouseY: event.clientY,\n      startCols: widget.cols,\n      startRows: widget.rows,\n      currentPixelW: this.colsToPixelWidth(widget.cols),\n      currentPixelH: this.rowsToPixelHeight(widget.rows),\n    };\n    this.previewWidgets = this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }));\n    this.placeholder = {\n      left: this.colToPixel(widget.x), top: this.rowToPixel(widget.y),\n      width: this.colsToPixelWidth(widget.cols),\n      height: this.rowsToPixelHeight(widget.rows),\n    };\n    this.cdr.markForCheck();\n  }\n\n  private onMouseMove(event: MouseEvent) {\n    if (this.dragging) this.handleDragMove(event);\n    else if (this.resizing) this.handleResizeMove(event);\n  }\n\n  private onMouseUp(_event: MouseEvent) {\n    if (this.dragging) this.zone.run(() => this.finalizeDrag());\n    else if (this.resizing) this.zone.run(() => this.finalizeResize());\n  }\n\n  private handleDragMove(event: MouseEvent) {\n    if (!this.dragging) return;\n    const rect = this.gridContainer.nativeElement.getBoundingClientRect();\n    const widget = this.widgets.find(w => w.id === this.dragging!.id)!;\n    if (!widget) return;\n\n    let px = event.clientX - rect.left - this.dragging.offsetX;\n    let py = event.clientY - rect.top - this.dragging.offsetY;\n    px = Math.max(0, Math.min(px, rect.width - this.colsToPixelWidth(widget.cols)));\n    py = Math.max(0, py);\n\n    this.dragging.currentPixelX = px;\n    this.dragging.currentPixelY = py;\n\n    const targetCol = Math.max(0, Math.min(this.pixelToCol(px), this.columns - widget.cols));\n    const targetRow = Math.max(0, this.pixelToRow(py));\n\n    const resolved = GridEngine.moveWidget(\n      this.previewWidgets,\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      targetCol, targetRow, this.columns\n    );\n\n    this.placeholder = {\n      left: this.colToPixel(targetCol), top: this.rowToPixel(targetRow),\n      width: this.colsToPixelWidth(widget.cols),\n      height: this.rowsToPixelHeight(widget.rows),\n    };\n\n    this.zone.run(() => {\n      for (const rw of resolved) {\n        if (rw.id === widget.id) continue;\n        const w = this.widgets.find(ww => ww.id === rw.id);\n        if (w) { w.x = rw.x; w.y = rw.y; }\n      }\n      this.updateContainerHeight();\n      this.cdr.markForCheck();\n    });\n  }\n\n  private finalizeDrag() {\n    if (!this.dragging) return;\n    const widget = this.widgets.find(w => w.id === this.dragging!.id)!;\n    const targetCol = Math.max(0, Math.min(this.pixelToCol(this.dragging.currentPixelX), this.columns - widget.cols));\n    const targetRow = Math.max(0, this.pixelToRow(this.dragging.currentPixelY));\n\n    const resolved = GridEngine.moveWidget(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })),\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      targetCol, targetRow, this.columns\n    );\n    for (const rw of resolved) {\n      const w = this.widgets.find(ww => ww.id === rw.id);\n      if (w) { w.x = rw.x; w.y = rw.y; w.cols = rw.cols; w.rows = rw.rows; this.itemChanged.emit(w); }\n    }\n    this.dragging = null;\n    this.placeholder = null;\n    this.previewWidgets = [];\n    this.updateContainerHeight();\n    this.layoutChanged.emit(this.widgets);\n    this.cdr.markForCheck();\n  }\n\n  private handleResizeMove(event: MouseEvent) {\n    if (!this.resizing) return;\n    const widget = this.widgets.find(w => w.id === this.resizing!.id)!;\n    if (!widget) return;\n\n    const dx = event.clientX - this.resizing.startMouseX;\n    const dy = event.clientY - this.resizing.startMouseY;\n\n    let newCols = Math.max(this.minItemCols, Math.round((this.colsToPixelWidth(this.resizing.startCols) + dx) / (this.cellWidth + this.gap)));\n    let newRows = Math.max(this.minItemRows, Math.round((this.rowsToPixelHeight(this.resizing.startRows) + dy) / (this.rowHeight + this.gap)));\n    newCols = Math.min(newCols, this.columns - widget.x);\n\n    this.resizing.currentPixelW = this.colsToPixelWidth(newCols);\n    this.resizing.currentPixelH = this.rowsToPixelHeight(newRows);\n\n    const resolved = GridEngine.resizeWidget(\n      this.previewWidgets,\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      newCols, newRows, this.columns\n    );\n\n    this.placeholder = {\n      left: this.colToPixel(widget.x), top: this.rowToPixel(widget.y),\n      width: this.colsToPixelWidth(newCols),\n      height: this.rowsToPixelHeight(newRows),\n    };\n\n    this.zone.run(() => {\n      for (const rw of resolved) {\n        if (rw.id === widget.id) continue;\n        const w = this.widgets.find(ww => ww.id === rw.id);\n        if (w) { w.x = rw.x; w.y = rw.y; }\n      }\n      this.updateContainerHeight();\n      this.cdr.markForCheck();\n    });\n  }\n\n  private finalizeResize() {\n    if (!this.resizing) return;\n    const widget = this.widgets.find(w => w.id === this.resizing!.id)!;\n\n    let newCols = Math.max(this.minItemCols, Math.round(this.resizing.currentPixelW / (this.cellWidth + this.gap)));\n    let newRows = Math.max(this.minItemRows, Math.round(this.resizing.currentPixelH / (this.rowHeight + this.gap)));\n    newCols = Math.min(newCols, this.columns - widget.x);\n\n    const resolved = GridEngine.resizeWidget(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })),\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      newCols, newRows, this.columns\n    );\n    for (const rw of resolved) {\n      const w = this.widgets.find(ww => ww.id === rw.id);\n      if (w) { w.x = rw.x; w.y = rw.y; w.cols = rw.cols; w.rows = rw.rows; this.itemChanged.emit(w); }\n    }\n    this.resizing = null;\n    this.placeholder = null;\n    this.previewWidgets = [];\n    this.updateContainerHeight();\n    this.layoutChanged.emit(this.widgets);\n    this.cdr.markForCheck();\n  }\n\n  // ── Utilities ──\n\n  private compactAndApply() {\n    if (!this.widgets?.length) return;\n    const compacted = GridEngine.compact(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })),\n      this.columns\n    );\n    for (const rw of compacted) {\n      const w = this.widgets.find(ww => ww.id === rw.id);\n      if (w) { w.x = rw.x; w.y = rw.y; }\n    }\n  }\n\n  private updateContainerHeight() {\n    if (!this.widgets?.length) { this.containerHeight = 400; return; }\n    const maxRow = GridEngine.getGridHeight(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }))\n    );\n    this.containerHeight = this.rowToPixel(maxRow) + this.rowHeight + this.gap;\n  }\n}\n"]}
534
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-grid.component.js","sourceRoot":"","sources":["../../../src/app/custom-grid.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, Output, EventEmitter, ElementRef,\n  ViewChild, OnDestroy, OnInit, NgZone, ChangeDetectorRef,\n  ContentChild, TemplateRef,\n  Directive, ChangeDetectionStrategy, OnChanges, SimpleChanges\n} from '@angular/core';\nimport { Widget } from './models';\n\n/**\n * Directive applied to the template that should be stamped inside each grid cell.\n * Usage:\n *   <app-grid [widgets]=\"...\">\n *     <ng-template gridCell let-widget=\"widget\">\n *       <app-widget-renderer [widget]=\"widget\" ...></app-widget-renderer>\n *     </ng-template>\n *   </app-grid>\n */\n@Directive({ selector: '[gridCell]' })\nexport class GridCellDirective {\n  constructor(public templateRef: TemplateRef<{ widget: Widget }>) {}\n}\n\n/* ═══════════════════════════════════════════════════════════════════════\n   GRID ENGINE — pure data, zero DOM\n   Handles collision detection, resolution, and compaction.\n   Inspired by react-grid-layout / gridster algorithms.\n   ═══════════════════════════════════════════════════════════════════════ */\n\ninterface GridRect {\n  id: string;\n  x: number;\n  y: number;\n  cols: number;\n  rows: number;\n}\n\nclass GridEngine {\n\n  /**\n   * Check if two rects overlap.\n   */\n  static collides(a: GridRect, b: GridRect): boolean {\n    return !(\n      a.x + a.cols <= b.x ||\n      b.x + b.cols <= a.x ||\n      a.y + a.rows <= b.y ||\n      b.y + b.rows <= a.y\n    );\n  }\n\n  /**\n   * Get ALL widgets that collide with `rect`.\n   */\n  static getAllCollisions(widgets: GridRect[], rect: GridRect): GridRect[] {\n    return widgets.filter(w => w.id !== rect.id && GridEngine.collides(w, rect));\n  }\n\n  /**\n   * Sort widgets top-to-bottom, left-to-right.\n   */\n  static sortByPosition(widgets: GridRect[]): GridRect[] {\n    return [...widgets].sort((a, b) => a.y - b.y || a.x - b.x);\n  }\n\n  /**\n   * Move a widget to (x, y) and push any colliding widgets downward.\n   * Returns the full list with resolved positions.\n   */\n  static moveWidget(\n    allWidgets: GridRect[],\n    widget: GridRect,\n    newX: number,\n    newY: number,\n    columns: number\n  ): GridRect[] {\n    const widgets = allWidgets.map(w => ({ ...w }));\n    const moving = widgets.find(w => w.id === widget.id);\n    if (!moving) return widgets;\n\n    moving.x = Math.max(0, Math.min(newX, columns - moving.cols));\n    moving.y = Math.max(0, newY);\n\n    const sorted = GridEngine.sortByPosition(widgets);\n    GridEngine.resolveCollisions(sorted, moving);\n    return GridEngine.compact(sorted, columns);\n  }\n\n  /**\n   * Resize a widget and push colliding widgets down.\n   */\n  static resizeWidget(\n    allWidgets: GridRect[],\n    widget: GridRect,\n    newCols: number,\n    newRows: number,\n    columns: number\n  ): GridRect[] {\n    const widgets = allWidgets.map(w => ({ ...w }));\n    const resizing = widgets.find(w => w.id === widget.id);\n    if (!resizing) return widgets;\n\n    resizing.cols = Math.max(1, Math.min(newCols, columns - resizing.x));\n    resizing.rows = Math.max(1, newRows);\n\n    const sorted = GridEngine.sortByPosition(widgets);\n    GridEngine.resolveCollisions(sorted, resizing);\n    return GridEngine.compact(sorted, columns);\n  }\n\n  /**\n   * Push all widgets that collide with `movedWidget` downward, recursively.\n   */\n  private static resolveCollisions(widgets: GridRect[], movedWidget: GridRect): void {\n    const collisions = GridEngine.getAllCollisions(widgets, movedWidget);\n    for (const collider of collisions) {\n      collider.y = movedWidget.y + movedWidget.rows;\n      GridEngine.resolveCollisions(widgets, collider);\n    }\n  }\n\n  /**\n   * Compact the grid: move every widget as far up as possible without overlapping.\n   */\n  static compact(widgets: GridRect[], _columns: number): GridRect[] {\n    const sorted = GridEngine.sortByPosition(widgets);\n    const placed: GridRect[] = [];\n\n    for (const widget of sorted) {\n      widget.y = GridEngine.findCompactY(placed, widget);\n      placed.push(widget);\n    }\n\n    return placed;\n  }\n\n  /**\n   * Find the highest Y position a widget can occupy without overlapping any already-placed widget.\n   */\n  private static findCompactY(placed: GridRect[], widget: GridRect): number {\n    let y = 0;\n    while (true) {\n      const test: GridRect = { ...widget, y };\n      const collision = placed.find(p => GridEngine.collides(p, test));\n      if (!collision) return y;\n      y = collision.y + collision.rows;\n    }\n  }\n\n  /**\n   * Compute the total number of rows the grid needs.\n   */\n  static getGridHeight(widgets: GridRect[]): number {\n    if (widgets.length === 0) return 0;\n    return Math.max(...widgets.map(w => w.y + w.rows));\n  }\n}\n\n/* ═══════════════════════════════════════════════════════════════════════\n   CUSTOM GRID COMPONENT\n   ═══════════════════════════════════════════════════════════════════════ */\n\n@Component({\n  selector: 'app-grid',\n  template: `\n    <div\n      #gridContainer\n      class=\"grid-container\"\n      [style.height.px]=\"containerHeight\"\n    >\n      <!-- Placeholder preview -->\n      <div\n        class=\"grid-placeholder\"\n        *ngIf=\"placeholder\"\n        [style.left.px]=\"placeholder.left\"\n        [style.top.px]=\"placeholder.top\"\n        [style.width.px]=\"placeholder.width\"\n        [style.height.px]=\"placeholder.height\"\n      ></div>\n\n      <!-- Widget items -->\n      <div\n        *ngFor=\"let widget of widgets; trackBy: trackByFn\"\n        class=\"grid-item\"\n        [class.is-dragging]=\"dragging?.id === widget.id\"\n        [class.is-resizing]=\"resizing?.id === widget.id\"\n        [style.left.px]=\"getItemLeft(widget)\"\n        [style.top.px]=\"getItemTop(widget)\"\n        [style.width.px]=\"getItemWidth(widget)\"\n        [style.height.px]=\"getItemHeight(widget)\"\n        [style.z-index]=\"(dragging?.id === widget.id || resizing?.id === widget.id) ? 100 : 1\"\n        (mousedown)=\"onDragStart($event, widget)\"\n      >\n        <!-- Content stamped from parent template, context carries the widget -->\n        <ng-container\n          *ngIf=\"cellTemplate\"\n          [ngTemplateOutlet]=\"cellTemplate\"\n          [ngTemplateOutletContext]=\"{ widget: widget }\"\n        ></ng-container>\n\n        <!-- Resize handle -->\n        <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, widget)\">\n          <svg width=\"12\" height=\"12\" viewBox=\"0 0 12 12\">\n            <path d=\"M11 1v10H1\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n            <path d=\"M11 5v6H5\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n            <path d=\"M11 9v2H9\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n          </svg>\n        </div>\n      </div>\n    </div>\n  `,\n  styles: [`\n    :host { display: block; width: 100%; height: 100%; }\n\n    .grid-container {\n      position: relative;\n      width: 100%;\n      min-height: 100%;\n      transition: height 0.25s ease;\n    }\n\n    .grid-placeholder {\n      position: absolute;\n      background: var(--dash-placeholder-color, var(--dash-accent-color, #0a84ff));\n      opacity: 0.12;\n      border-radius: 24px;\n      border: 2px dashed var(--dash-placeholder-color, var(--dash-accent-color, #0a84ff));\n      transition: left 0.15s ease, top 0.15s ease, width 0.15s ease, height 0.15s ease;\n      pointer-events: none;\n      z-index: 0;\n    }\n\n    .grid-item {\n      position: absolute;\n      display: flex;\n      flex-direction: column;\n      transition: left 0.25s ease, top 0.25s ease, width 0.25s ease, height 0.25s ease;\n      border-radius: 24px;\n      overflow: hidden;\n    }\n\n    .grid-item.is-dragging,\n    .grid-item.is-resizing {\n      transition: none !important;\n      opacity: 0.88;\n      overflow: visible;\n    }\n\n    /* Resize Handle */\n    .resize-handle {\n      position: absolute;\n      right: 4px;\n      bottom: 4px;\n      width: 22px;\n      height: 22px;\n      cursor: nwse-resize;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      color: var(--dash-resize-handle-color, rgba(255,255,255,0.25));\n      border-radius: 6px;\n      transition: color 0.2s, background 0.2s;\n      z-index: 20;\n    }\n    .resize-handle:hover {\n      color: var(--dash-accent-color, #0a84ff);\n      background: rgba(255,255,255,0.06);\n    }\n  `],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class CustomGridComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() widgets: Widget[] = [];\n  @Input() columns: number = 12;\n  @Input() gap: number = 16;\n  @Input() rowHeight: number = 80;\n  @Input() minItemCols: number = 1;\n  @Input() minItemRows: number = 1;\n\n  @Output() itemChanged = new EventEmitter<Widget>();\n  @Output() layoutChanged = new EventEmitter<Widget[]>();\n\n  @ViewChild('gridContainer', { static: true }) gridContainer!: ElementRef<HTMLDivElement>;\n\n  /** The ng-template decorated with gridCell from the parent */\n  @ContentChild(GridCellDirective, { read: TemplateRef }) cellTemplate!: TemplateRef<{ widget: Widget }>;\n\n  placeholder: { left: number; top: number; width: number; height: number } | null = null;\n  containerHeight = 400;\n\n  dragging: {\n    id: string;\n    offsetX: number;\n    offsetY: number;\n    currentPixelX: number;\n    currentPixelY: number;\n  } | null = null;\n\n  resizing: {\n    id: string;\n    startMouseX: number;\n    startMouseY: number;\n    startCols: number;\n    startRows: number;\n    currentPixelW: number;\n    currentPixelH: number;\n  } | null = null;\n\n  private previewWidgets: GridRect[] = [];\n  private boundMouseMove = this.onMouseMove.bind(this);\n  private boundMouseUp = this.onMouseUp.bind(this);\n\n  constructor(private zone: NgZone, private cdr: ChangeDetectorRef) {}\n\n  ngOnInit() {\n    this.applyCompactionAndSync();\n    this.updateContainerHeight();\n    this.zone.runOutsideAngular(() => {\n      window.addEventListener('mousemove', this.boundMouseMove);\n      window.addEventListener('mouseup', this.boundMouseUp);\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (this.dragging || this.resizing) {\n      this.updateContainerHeight();\n      return;\n    }\n\n    if (changes['widgets'] || changes['columns']) {\n      this.applyCompactionAndSync();\n    }\n\n    this.updateContainerHeight();\n    this.cdr.markForCheck();\n  }\n\n  ngOnDestroy() {\n    window.removeEventListener('mousemove', this.boundMouseMove);\n    window.removeEventListener('mouseup', this.boundMouseUp);\n  }\n\n  trackByFn(_index: number, item: Widget) { return item.id; }\n\n  // ── Geometry ──\n\n  private get cellWidth(): number {\n    const containerW = this.gridContainer?.nativeElement?.clientWidth ?? 800;\n    return (containerW - (this.columns - 1) * this.gap) / this.columns;\n  }\n\n  private colToPixel(col: number): number {\n    return col * (this.cellWidth + this.gap);\n  }\n\n  private rowToPixel(row: number): number {\n    return row * (this.rowHeight + this.gap);\n  }\n\n  private pixelToCol(px: number): number {\n    return Math.round(px / (this.cellWidth + this.gap));\n  }\n\n  private pixelToRow(px: number): number {\n    return Math.round(px / (this.rowHeight + this.gap));\n  }\n\n  private colsToPixelWidth(cols: number): number {\n    return cols * this.cellWidth + (cols - 1) * this.gap;\n  }\n\n  private rowsToPixelHeight(rows: number): number {\n    return rows * this.rowHeight + (rows - 1) * this.gap;\n  }\n\n  getItemLeft(w: Widget): number {\n    return this.dragging?.id === w.id ? this.dragging.currentPixelX : this.colToPixel(w.x);\n  }\n\n  getItemTop(w: Widget): number {\n    return this.dragging?.id === w.id ? this.dragging.currentPixelY : this.rowToPixel(w.y);\n  }\n\n  getItemWidth(w: Widget): number {\n    return this.resizing?.id === w.id ? this.resizing.currentPixelW : this.colsToPixelWidth(w.cols);\n  }\n\n  getItemHeight(w: Widget): number {\n    return this.resizing?.id === w.id ? this.resizing.currentPixelH : this.rowsToPixelHeight(w.rows);\n  }\n\n  // ── Drag ──\n\n  onDragStart(event: MouseEvent, widget: Widget) {\n    const target = event.target as HTMLElement;\n    if (target.closest('button, input, select, textarea, a, .resize-handle')) return;\n    event.preventDefault();\n    event.stopPropagation();\n\n    const rect = this.gridContainer.nativeElement.getBoundingClientRect();\n    const itemLeft = this.colToPixel(widget.x);\n    const itemTop = this.rowToPixel(widget.y);\n\n    this.dragging = {\n      id: widget.id,\n      offsetX: event.clientX - rect.left - itemLeft,\n      offsetY: event.clientY - rect.top - itemTop,\n      currentPixelX: itemLeft,\n      currentPixelY: itemTop,\n    };\n    this.previewWidgets = this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }));\n    this.placeholder = {\n      left: itemLeft, top: itemTop,\n      width: this.colsToPixelWidth(widget.cols),\n      height: this.rowsToPixelHeight(widget.rows),\n    };\n    this.cdr.markForCheck();\n  }\n\n  onResizeStart(event: MouseEvent, widget: Widget) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.resizing = {\n      id: widget.id,\n      startMouseX: event.clientX,\n      startMouseY: event.clientY,\n      startCols: widget.cols,\n      startRows: widget.rows,\n      currentPixelW: this.colsToPixelWidth(widget.cols),\n      currentPixelH: this.rowsToPixelHeight(widget.rows),\n    };\n    this.previewWidgets = this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }));\n    this.placeholder = {\n      left: this.colToPixel(widget.x), top: this.rowToPixel(widget.y),\n      width: this.colsToPixelWidth(widget.cols),\n      height: this.rowsToPixelHeight(widget.rows),\n    };\n    this.cdr.markForCheck();\n  }\n\n  private onMouseMove(event: MouseEvent) {\n    if (this.dragging) this.handleDragMove(event);\n    else if (this.resizing) this.handleResizeMove(event);\n  }\n\n  private onMouseUp(_event: MouseEvent) {\n    if (this.dragging) this.zone.run(() => this.finalizeDrag());\n    else if (this.resizing) this.zone.run(() => this.finalizeResize());\n  }\n\n  private handleDragMove(event: MouseEvent) {\n    if (!this.dragging) return;\n    const rect = this.gridContainer.nativeElement.getBoundingClientRect();\n    const widget = this.widgets.find(w => w.id === this.dragging!.id)!;\n    if (!widget) return;\n\n    let px = event.clientX - rect.left - this.dragging.offsetX;\n    let py = event.clientY - rect.top - this.dragging.offsetY;\n    px = Math.max(0, Math.min(px, rect.width - this.colsToPixelWidth(widget.cols)));\n    py = Math.max(0, py);\n\n    this.dragging.currentPixelX = px;\n    this.dragging.currentPixelY = py;\n\n    const targetCol = Math.max(0, Math.min(this.pixelToCol(px), this.columns - widget.cols));\n    const targetRow = Math.max(0, this.pixelToRow(py));\n\n    const resolved = GridEngine.moveWidget(\n      this.previewWidgets,\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      targetCol, targetRow, this.columns\n    );\n\n    this.placeholder = {\n      left: this.colToPixel(targetCol), top: this.rowToPixel(targetRow),\n      width: this.colsToPixelWidth(widget.cols),\n      height: this.rowsToPixelHeight(widget.rows),\n    };\n\n    this.zone.run(() => {\n      for (const rw of resolved) {\n        if (rw.id === widget.id) continue;\n        const w = this.widgets.find(ww => ww.id === rw.id);\n        if (w) { w.x = rw.x; w.y = rw.y; }\n      }\n      this.updateContainerHeight();\n      this.cdr.markForCheck();\n    });\n  }\n\n  private finalizeDrag() {\n    if (!this.dragging) return;\n    const widget = this.widgets.find(w => w.id === this.dragging!.id)!;\n    const targetCol = Math.max(0, Math.min(this.pixelToCol(this.dragging.currentPixelX), this.columns - widget.cols));\n    const targetRow = Math.max(0, this.pixelToRow(this.dragging.currentPixelY));\n\n    const resolved = GridEngine.moveWidget(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })),\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      targetCol, targetRow, this.columns\n    );\n    for (const rw of resolved) {\n      const w = this.widgets.find(ww => ww.id === rw.id);\n      if (w) { w.x = rw.x; w.y = rw.y; w.cols = rw.cols; w.rows = rw.rows; this.itemChanged.emit(w); }\n    }\n    this.dragging = null;\n    this.placeholder = null;\n    this.previewWidgets = [];\n    this.updateContainerHeight();\n    this.layoutChanged.emit(this.widgets);\n    this.cdr.markForCheck();\n  }\n\n  private handleResizeMove(event: MouseEvent) {\n    if (!this.resizing) return;\n    const widget = this.widgets.find(w => w.id === this.resizing!.id)!;\n    if (!widget) return;\n\n    const dx = event.clientX - this.resizing.startMouseX;\n    const dy = event.clientY - this.resizing.startMouseY;\n\n    let newCols = Math.max(this.minItemCols, Math.round((this.colsToPixelWidth(this.resizing.startCols) + dx) / (this.cellWidth + this.gap)));\n    let newRows = Math.max(this.minItemRows, Math.round((this.rowsToPixelHeight(this.resizing.startRows) + dy) / (this.rowHeight + this.gap)));\n    newCols = Math.min(newCols, this.columns - widget.x);\n\n    this.resizing.currentPixelW = this.colsToPixelWidth(newCols);\n    this.resizing.currentPixelH = this.rowsToPixelHeight(newRows);\n\n    const resolved = GridEngine.resizeWidget(\n      this.previewWidgets,\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      newCols, newRows, this.columns\n    );\n\n    this.placeholder = {\n      left: this.colToPixel(widget.x), top: this.rowToPixel(widget.y),\n      width: this.colsToPixelWidth(newCols),\n      height: this.rowsToPixelHeight(newRows),\n    };\n\n    this.zone.run(() => {\n      for (const rw of resolved) {\n        if (rw.id === widget.id) continue;\n        const w = this.widgets.find(ww => ww.id === rw.id);\n        if (w) { w.x = rw.x; w.y = rw.y; }\n      }\n      this.updateContainerHeight();\n      this.cdr.markForCheck();\n    });\n  }\n\n  private finalizeResize() {\n    if (!this.resizing) return;\n    const widget = this.widgets.find(w => w.id === this.resizing!.id)!;\n\n    let newCols = Math.max(this.minItemCols, Math.round(this.resizing.currentPixelW / (this.cellWidth + this.gap)));\n    let newRows = Math.max(this.minItemRows, Math.round(this.resizing.currentPixelH / (this.rowHeight + this.gap)));\n    newCols = Math.min(newCols, this.columns - widget.x);\n\n    const resolved = GridEngine.resizeWidget(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })),\n      { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows },\n      newCols, newRows, this.columns\n    );\n    for (const rw of resolved) {\n      const w = this.widgets.find(ww => ww.id === rw.id);\n      if (w) { w.x = rw.x; w.y = rw.y; w.cols = rw.cols; w.rows = rw.rows; this.itemChanged.emit(w); }\n    }\n    this.resizing = null;\n    this.placeholder = null;\n    this.previewWidgets = [];\n    this.updateContainerHeight();\n    this.layoutChanged.emit(this.widgets);\n    this.cdr.markForCheck();\n  }\n\n  // ── Utilities ──\n\n  private compactAndApply(): Widget[] {\n    if (!this.widgets?.length) return [];\n    const compacted = GridEngine.compact(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })),\n      this.columns\n    );\n\n    const changedWidgets: Widget[] = [];\n    for (const rw of compacted) {\n      const w = this.widgets.find(ww => ww.id === rw.id);\n      if (!w) continue;\n\n      const changed = w.x !== rw.x || w.y !== rw.y || w.cols !== rw.cols || w.rows !== rw.rows;\n      if (changed) {\n        w.x = rw.x;\n        w.y = rw.y;\n        w.cols = rw.cols;\n        w.rows = rw.rows;\n        changedWidgets.push(w);\n      }\n    }\n\n    return changedWidgets;\n  }\n\n  private applyCompactionAndSync() {\n    const changedWidgets = this.compactAndApply();\n    if (!changedWidgets.length) return;\n\n    for (const widget of changedWidgets) {\n      this.itemChanged.emit(widget);\n    }\n    this.layoutChanged.emit(this.widgets);\n  }\n\n  private updateContainerHeight() {\n    if (!this.widgets?.length) {\n      this.containerHeight = 400;\n      return;\n    }\n\n    const maxRow = GridEngine.getGridHeight(\n      this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }))\n    );\n    this.containerHeight = this.rowToPixel(maxRow) + this.rowHeight + this.gap;\n  }\n}\n"]}