@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 +83 -13
- package/app/custom-grid.component.d.ts +2 -1
- package/app/dashboard-persistence.d.ts +37 -0
- package/app/dashboard-state.service.d.ts +10 -37
- package/esm2020/app/app.module.mjs +10 -3
- package/esm2020/app/custom-grid.component.mjs +30 -5
- package/esm2020/app/dashboard-persistence.mjs +78 -0
- package/esm2020/app/dashboard-state.service.mjs +54 -87
- package/esm2020/public-api.mjs +2 -1
- package/fesm2015/ogidor-dashboard.mjs +184 -93
- package/fesm2015/ogidor-dashboard.mjs.map +1 -1
- package/fesm2020/ogidor-dashboard.mjs +165 -93
- package/fesm2020/ogidor-dashboard.mjs.map +1 -1
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
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
|
|
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
|
-
- **
|
|
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
|
-
|
|
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
|
|
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
|
-
|
|
527
|
+
Provide your own implementation of `DashboardPersistence` and override the `DASHBOARD_PERSISTENCE` token.
|
|
517
528
|
|
|
518
529
|
```typescript
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
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
|
-
|
|
524
|
-
|
|
525
|
-
|
|
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(
|
|
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
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
private readonly
|
|
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
|
-
|
|
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: [
|
|
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: [
|
|
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,
|
|
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.
|
|
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(
|
|
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"]}
|