@ogidor/dashboard 1.0.2 → 1.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +283 -48
- package/app/app.module.d.ts +7 -9
- package/app/custom-grid.component.d.ts +93 -0
- package/app/dashboard-state.service.d.ts +54 -33
- package/app/dashboard.component.d.ts +27 -59
- package/app/models.d.ts +18 -38
- package/app/widget-renderer.component.d.ts +5 -32
- package/esm2020/app/app.module.mjs +21 -29
- package/esm2020/app/custom-grid.component.mjs +509 -0
- package/esm2020/app/dashboard-state.service.mjs +228 -158
- package/esm2020/app/dashboard.component.mjs +198 -602
- package/esm2020/app/models.mjs +1 -1
- package/esm2020/app/widget-renderer.component.mjs +41 -121
- package/esm2020/public-api.mjs +5 -4
- package/fesm2015/ogidor-dashboard.mjs +988 -909
- package/fesm2015/ogidor-dashboard.mjs.map +1 -1
- package/fesm2020/ogidor-dashboard.mjs +982 -898
- package/fesm2020/ogidor-dashboard.mjs.map +1 -1
- package/package.json +8 -12
- package/public-api.d.ts +3 -2
|
@@ -0,0 +1,509 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ViewChild, ContentChild, TemplateRef, Directive, ChangeDetectionStrategy } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/common";
|
|
4
|
+
/**
|
|
5
|
+
* Directive applied to the template that should be stamped inside each grid cell.
|
|
6
|
+
* Usage:
|
|
7
|
+
* <app-grid [widgets]="...">
|
|
8
|
+
* <ng-template gridCell let-widget="widget">
|
|
9
|
+
* <app-widget-renderer [widget]="widget" ...></app-widget-renderer>
|
|
10
|
+
* </ng-template>
|
|
11
|
+
* </app-grid>
|
|
12
|
+
*/
|
|
13
|
+
export class GridCellDirective {
|
|
14
|
+
constructor(templateRef) {
|
|
15
|
+
this.templateRef = templateRef;
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
GridCellDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GridCellDirective, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
19
|
+
GridCellDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.10", type: GridCellDirective, selector: "[gridCell]", ngImport: i0 });
|
|
20
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: GridCellDirective, decorators: [{
|
|
21
|
+
type: Directive,
|
|
22
|
+
args: [{ selector: '[gridCell]' }]
|
|
23
|
+
}], ctorParameters: function () { return [{ type: i0.TemplateRef }]; } });
|
|
24
|
+
class GridEngine {
|
|
25
|
+
/**
|
|
26
|
+
* Check if two rects overlap.
|
|
27
|
+
*/
|
|
28
|
+
static collides(a, b) {
|
|
29
|
+
return !(a.x + a.cols <= b.x ||
|
|
30
|
+
b.x + b.cols <= a.x ||
|
|
31
|
+
a.y + a.rows <= b.y ||
|
|
32
|
+
b.y + b.rows <= a.y);
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Get ALL widgets that collide with `rect`.
|
|
36
|
+
*/
|
|
37
|
+
static getAllCollisions(widgets, rect) {
|
|
38
|
+
return widgets.filter(w => w.id !== rect.id && GridEngine.collides(w, rect));
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Sort widgets top-to-bottom, left-to-right.
|
|
42
|
+
*/
|
|
43
|
+
static sortByPosition(widgets) {
|
|
44
|
+
return [...widgets].sort((a, b) => a.y - b.y || a.x - b.x);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Move a widget to (x, y) and push any colliding widgets downward.
|
|
48
|
+
* Returns the full list with resolved positions.
|
|
49
|
+
*/
|
|
50
|
+
static moveWidget(allWidgets, widget, newX, newY, columns) {
|
|
51
|
+
const widgets = allWidgets.map(w => ({ ...w }));
|
|
52
|
+
const moving = widgets.find(w => w.id === widget.id);
|
|
53
|
+
if (!moving)
|
|
54
|
+
return widgets;
|
|
55
|
+
moving.x = Math.max(0, Math.min(newX, columns - moving.cols));
|
|
56
|
+
moving.y = Math.max(0, newY);
|
|
57
|
+
const sorted = GridEngine.sortByPosition(widgets);
|
|
58
|
+
GridEngine.resolveCollisions(sorted, moving);
|
|
59
|
+
return GridEngine.compact(sorted, columns);
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Resize a widget and push colliding widgets down.
|
|
63
|
+
*/
|
|
64
|
+
static resizeWidget(allWidgets, widget, newCols, newRows, columns) {
|
|
65
|
+
const widgets = allWidgets.map(w => ({ ...w }));
|
|
66
|
+
const resizing = widgets.find(w => w.id === widget.id);
|
|
67
|
+
if (!resizing)
|
|
68
|
+
return widgets;
|
|
69
|
+
resizing.cols = Math.max(1, Math.min(newCols, columns - resizing.x));
|
|
70
|
+
resizing.rows = Math.max(1, newRows);
|
|
71
|
+
const sorted = GridEngine.sortByPosition(widgets);
|
|
72
|
+
GridEngine.resolveCollisions(sorted, resizing);
|
|
73
|
+
return GridEngine.compact(sorted, columns);
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Push all widgets that collide with `movedWidget` downward, recursively.
|
|
77
|
+
*/
|
|
78
|
+
static resolveCollisions(widgets, movedWidget) {
|
|
79
|
+
const collisions = GridEngine.getAllCollisions(widgets, movedWidget);
|
|
80
|
+
for (const collider of collisions) {
|
|
81
|
+
collider.y = movedWidget.y + movedWidget.rows;
|
|
82
|
+
GridEngine.resolveCollisions(widgets, collider);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Compact the grid: move every widget as far up as possible without overlapping.
|
|
87
|
+
*/
|
|
88
|
+
static compact(widgets, _columns) {
|
|
89
|
+
const sorted = GridEngine.sortByPosition(widgets);
|
|
90
|
+
const placed = [];
|
|
91
|
+
for (const widget of sorted) {
|
|
92
|
+
widget.y = GridEngine.findCompactY(placed, widget);
|
|
93
|
+
placed.push(widget);
|
|
94
|
+
}
|
|
95
|
+
return placed;
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Find the highest Y position a widget can occupy without overlapping any already-placed widget.
|
|
99
|
+
*/
|
|
100
|
+
static findCompactY(placed, widget) {
|
|
101
|
+
let y = 0;
|
|
102
|
+
while (true) {
|
|
103
|
+
const test = { ...widget, y };
|
|
104
|
+
const collision = placed.find(p => GridEngine.collides(p, test));
|
|
105
|
+
if (!collision)
|
|
106
|
+
return y;
|
|
107
|
+
y = collision.y + collision.rows;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Compute the total number of rows the grid needs.
|
|
112
|
+
*/
|
|
113
|
+
static getGridHeight(widgets) {
|
|
114
|
+
if (widgets.length === 0)
|
|
115
|
+
return 0;
|
|
116
|
+
return Math.max(...widgets.map(w => w.y + w.rows));
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
/* ═══════════════════════════════════════════════════════════════════════
|
|
120
|
+
CUSTOM GRID COMPONENT
|
|
121
|
+
═══════════════════════════════════════════════════════════════════════ */
|
|
122
|
+
export class CustomGridComponent {
|
|
123
|
+
constructor(zone, cdr) {
|
|
124
|
+
this.zone = zone;
|
|
125
|
+
this.cdr = cdr;
|
|
126
|
+
this.widgets = [];
|
|
127
|
+
this.columns = 12;
|
|
128
|
+
this.gap = 16;
|
|
129
|
+
this.rowHeight = 80;
|
|
130
|
+
this.minItemCols = 1;
|
|
131
|
+
this.minItemRows = 1;
|
|
132
|
+
this.itemChanged = new EventEmitter();
|
|
133
|
+
this.layoutChanged = new EventEmitter();
|
|
134
|
+
this.placeholder = null;
|
|
135
|
+
this.containerHeight = 400;
|
|
136
|
+
this.dragging = null;
|
|
137
|
+
this.resizing = null;
|
|
138
|
+
this.previewWidgets = [];
|
|
139
|
+
this.boundMouseMove = this.onMouseMove.bind(this);
|
|
140
|
+
this.boundMouseUp = this.onMouseUp.bind(this);
|
|
141
|
+
}
|
|
142
|
+
ngOnInit() {
|
|
143
|
+
this.compactAndApply();
|
|
144
|
+
this.updateContainerHeight();
|
|
145
|
+
this.zone.runOutsideAngular(() => {
|
|
146
|
+
window.addEventListener('mousemove', this.boundMouseMove);
|
|
147
|
+
window.addEventListener('mouseup', this.boundMouseUp);
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
ngOnChanges(_changes) {
|
|
151
|
+
this.updateContainerHeight();
|
|
152
|
+
}
|
|
153
|
+
ngOnDestroy() {
|
|
154
|
+
window.removeEventListener('mousemove', this.boundMouseMove);
|
|
155
|
+
window.removeEventListener('mouseup', this.boundMouseUp);
|
|
156
|
+
}
|
|
157
|
+
trackByFn(_index, item) { return item.id; }
|
|
158
|
+
// ── Geometry ──
|
|
159
|
+
get cellWidth() {
|
|
160
|
+
const containerW = this.gridContainer?.nativeElement?.clientWidth ?? 800;
|
|
161
|
+
return (containerW - (this.columns - 1) * this.gap) / this.columns;
|
|
162
|
+
}
|
|
163
|
+
colToPixel(col) {
|
|
164
|
+
return col * (this.cellWidth + this.gap);
|
|
165
|
+
}
|
|
166
|
+
rowToPixel(row) {
|
|
167
|
+
return row * (this.rowHeight + this.gap);
|
|
168
|
+
}
|
|
169
|
+
pixelToCol(px) {
|
|
170
|
+
return Math.round(px / (this.cellWidth + this.gap));
|
|
171
|
+
}
|
|
172
|
+
pixelToRow(px) {
|
|
173
|
+
return Math.round(px / (this.rowHeight + this.gap));
|
|
174
|
+
}
|
|
175
|
+
colsToPixelWidth(cols) {
|
|
176
|
+
return cols * this.cellWidth + (cols - 1) * this.gap;
|
|
177
|
+
}
|
|
178
|
+
rowsToPixelHeight(rows) {
|
|
179
|
+
return rows * this.rowHeight + (rows - 1) * this.gap;
|
|
180
|
+
}
|
|
181
|
+
getItemLeft(w) {
|
|
182
|
+
return this.dragging?.id === w.id ? this.dragging.currentPixelX : this.colToPixel(w.x);
|
|
183
|
+
}
|
|
184
|
+
getItemTop(w) {
|
|
185
|
+
return this.dragging?.id === w.id ? this.dragging.currentPixelY : this.rowToPixel(w.y);
|
|
186
|
+
}
|
|
187
|
+
getItemWidth(w) {
|
|
188
|
+
return this.resizing?.id === w.id ? this.resizing.currentPixelW : this.colsToPixelWidth(w.cols);
|
|
189
|
+
}
|
|
190
|
+
getItemHeight(w) {
|
|
191
|
+
return this.resizing?.id === w.id ? this.resizing.currentPixelH : this.rowsToPixelHeight(w.rows);
|
|
192
|
+
}
|
|
193
|
+
// ── Drag ──
|
|
194
|
+
onDragStart(event, widget) {
|
|
195
|
+
const target = event.target;
|
|
196
|
+
if (target.closest('button, input, select, textarea, a, .resize-handle'))
|
|
197
|
+
return;
|
|
198
|
+
event.preventDefault();
|
|
199
|
+
event.stopPropagation();
|
|
200
|
+
const rect = this.gridContainer.nativeElement.getBoundingClientRect();
|
|
201
|
+
const itemLeft = this.colToPixel(widget.x);
|
|
202
|
+
const itemTop = this.rowToPixel(widget.y);
|
|
203
|
+
this.dragging = {
|
|
204
|
+
id: widget.id,
|
|
205
|
+
offsetX: event.clientX - rect.left - itemLeft,
|
|
206
|
+
offsetY: event.clientY - rect.top - itemTop,
|
|
207
|
+
currentPixelX: itemLeft,
|
|
208
|
+
currentPixelY: itemTop,
|
|
209
|
+
};
|
|
210
|
+
this.previewWidgets = this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }));
|
|
211
|
+
this.placeholder = {
|
|
212
|
+
left: itemLeft, top: itemTop,
|
|
213
|
+
width: this.colsToPixelWidth(widget.cols),
|
|
214
|
+
height: this.rowsToPixelHeight(widget.rows),
|
|
215
|
+
};
|
|
216
|
+
this.cdr.markForCheck();
|
|
217
|
+
}
|
|
218
|
+
onResizeStart(event, widget) {
|
|
219
|
+
event.preventDefault();
|
|
220
|
+
event.stopPropagation();
|
|
221
|
+
this.resizing = {
|
|
222
|
+
id: widget.id,
|
|
223
|
+
startMouseX: event.clientX,
|
|
224
|
+
startMouseY: event.clientY,
|
|
225
|
+
startCols: widget.cols,
|
|
226
|
+
startRows: widget.rows,
|
|
227
|
+
currentPixelW: this.colsToPixelWidth(widget.cols),
|
|
228
|
+
currentPixelH: this.rowsToPixelHeight(widget.rows),
|
|
229
|
+
};
|
|
230
|
+
this.previewWidgets = this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows }));
|
|
231
|
+
this.placeholder = {
|
|
232
|
+
left: this.colToPixel(widget.x), top: this.rowToPixel(widget.y),
|
|
233
|
+
width: this.colsToPixelWidth(widget.cols),
|
|
234
|
+
height: this.rowsToPixelHeight(widget.rows),
|
|
235
|
+
};
|
|
236
|
+
this.cdr.markForCheck();
|
|
237
|
+
}
|
|
238
|
+
onMouseMove(event) {
|
|
239
|
+
if (this.dragging)
|
|
240
|
+
this.handleDragMove(event);
|
|
241
|
+
else if (this.resizing)
|
|
242
|
+
this.handleResizeMove(event);
|
|
243
|
+
}
|
|
244
|
+
onMouseUp(_event) {
|
|
245
|
+
if (this.dragging)
|
|
246
|
+
this.zone.run(() => this.finalizeDrag());
|
|
247
|
+
else if (this.resizing)
|
|
248
|
+
this.zone.run(() => this.finalizeResize());
|
|
249
|
+
}
|
|
250
|
+
handleDragMove(event) {
|
|
251
|
+
if (!this.dragging)
|
|
252
|
+
return;
|
|
253
|
+
const rect = this.gridContainer.nativeElement.getBoundingClientRect();
|
|
254
|
+
const widget = this.widgets.find(w => w.id === this.dragging.id);
|
|
255
|
+
if (!widget)
|
|
256
|
+
return;
|
|
257
|
+
let px = event.clientX - rect.left - this.dragging.offsetX;
|
|
258
|
+
let py = event.clientY - rect.top - this.dragging.offsetY;
|
|
259
|
+
px = Math.max(0, Math.min(px, rect.width - this.colsToPixelWidth(widget.cols)));
|
|
260
|
+
py = Math.max(0, py);
|
|
261
|
+
this.dragging.currentPixelX = px;
|
|
262
|
+
this.dragging.currentPixelY = py;
|
|
263
|
+
const targetCol = Math.max(0, Math.min(this.pixelToCol(px), this.columns - widget.cols));
|
|
264
|
+
const targetRow = Math.max(0, this.pixelToRow(py));
|
|
265
|
+
const resolved = GridEngine.moveWidget(this.previewWidgets, { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows }, targetCol, targetRow, this.columns);
|
|
266
|
+
this.placeholder = {
|
|
267
|
+
left: this.colToPixel(targetCol), top: this.rowToPixel(targetRow),
|
|
268
|
+
width: this.colsToPixelWidth(widget.cols),
|
|
269
|
+
height: this.rowsToPixelHeight(widget.rows),
|
|
270
|
+
};
|
|
271
|
+
this.zone.run(() => {
|
|
272
|
+
for (const rw of resolved) {
|
|
273
|
+
if (rw.id === widget.id)
|
|
274
|
+
continue;
|
|
275
|
+
const w = this.widgets.find(ww => ww.id === rw.id);
|
|
276
|
+
if (w) {
|
|
277
|
+
w.x = rw.x;
|
|
278
|
+
w.y = rw.y;
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
this.updateContainerHeight();
|
|
282
|
+
this.cdr.markForCheck();
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
finalizeDrag() {
|
|
286
|
+
if (!this.dragging)
|
|
287
|
+
return;
|
|
288
|
+
const widget = this.widgets.find(w => w.id === this.dragging.id);
|
|
289
|
+
const targetCol = Math.max(0, Math.min(this.pixelToCol(this.dragging.currentPixelX), this.columns - widget.cols));
|
|
290
|
+
const targetRow = Math.max(0, this.pixelToRow(this.dragging.currentPixelY));
|
|
291
|
+
const resolved = GridEngine.moveWidget(this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })), { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows }, targetCol, targetRow, this.columns);
|
|
292
|
+
for (const rw of resolved) {
|
|
293
|
+
const w = this.widgets.find(ww => ww.id === rw.id);
|
|
294
|
+
if (w) {
|
|
295
|
+
w.x = rw.x;
|
|
296
|
+
w.y = rw.y;
|
|
297
|
+
w.cols = rw.cols;
|
|
298
|
+
w.rows = rw.rows;
|
|
299
|
+
this.itemChanged.emit(w);
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
this.dragging = null;
|
|
303
|
+
this.placeholder = null;
|
|
304
|
+
this.previewWidgets = [];
|
|
305
|
+
this.updateContainerHeight();
|
|
306
|
+
this.layoutChanged.emit(this.widgets);
|
|
307
|
+
this.cdr.markForCheck();
|
|
308
|
+
}
|
|
309
|
+
handleResizeMove(event) {
|
|
310
|
+
if (!this.resizing)
|
|
311
|
+
return;
|
|
312
|
+
const widget = this.widgets.find(w => w.id === this.resizing.id);
|
|
313
|
+
if (!widget)
|
|
314
|
+
return;
|
|
315
|
+
const dx = event.clientX - this.resizing.startMouseX;
|
|
316
|
+
const dy = event.clientY - this.resizing.startMouseY;
|
|
317
|
+
let newCols = Math.max(this.minItemCols, Math.round((this.colsToPixelWidth(this.resizing.startCols) + dx) / (this.cellWidth + this.gap)));
|
|
318
|
+
let newRows = Math.max(this.minItemRows, Math.round((this.rowsToPixelHeight(this.resizing.startRows) + dy) / (this.rowHeight + this.gap)));
|
|
319
|
+
newCols = Math.min(newCols, this.columns - widget.x);
|
|
320
|
+
this.resizing.currentPixelW = this.colsToPixelWidth(newCols);
|
|
321
|
+
this.resizing.currentPixelH = this.rowsToPixelHeight(newRows);
|
|
322
|
+
const resolved = GridEngine.resizeWidget(this.previewWidgets, { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows }, newCols, newRows, this.columns);
|
|
323
|
+
this.placeholder = {
|
|
324
|
+
left: this.colToPixel(widget.x), top: this.rowToPixel(widget.y),
|
|
325
|
+
width: this.colsToPixelWidth(newCols),
|
|
326
|
+
height: this.rowsToPixelHeight(newRows),
|
|
327
|
+
};
|
|
328
|
+
this.zone.run(() => {
|
|
329
|
+
for (const rw of resolved) {
|
|
330
|
+
if (rw.id === widget.id)
|
|
331
|
+
continue;
|
|
332
|
+
const w = this.widgets.find(ww => ww.id === rw.id);
|
|
333
|
+
if (w) {
|
|
334
|
+
w.x = rw.x;
|
|
335
|
+
w.y = rw.y;
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
this.updateContainerHeight();
|
|
339
|
+
this.cdr.markForCheck();
|
|
340
|
+
});
|
|
341
|
+
}
|
|
342
|
+
finalizeResize() {
|
|
343
|
+
if (!this.resizing)
|
|
344
|
+
return;
|
|
345
|
+
const widget = this.widgets.find(w => w.id === this.resizing.id);
|
|
346
|
+
let newCols = Math.max(this.minItemCols, Math.round(this.resizing.currentPixelW / (this.cellWidth + this.gap)));
|
|
347
|
+
let newRows = Math.max(this.minItemRows, Math.round(this.resizing.currentPixelH / (this.rowHeight + this.gap)));
|
|
348
|
+
newCols = Math.min(newCols, this.columns - widget.x);
|
|
349
|
+
const resolved = GridEngine.resizeWidget(this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })), { id: widget.id, x: widget.x, y: widget.y, cols: widget.cols, rows: widget.rows }, newCols, newRows, this.columns);
|
|
350
|
+
for (const rw of resolved) {
|
|
351
|
+
const w = this.widgets.find(ww => ww.id === rw.id);
|
|
352
|
+
if (w) {
|
|
353
|
+
w.x = rw.x;
|
|
354
|
+
w.y = rw.y;
|
|
355
|
+
w.cols = rw.cols;
|
|
356
|
+
w.rows = rw.rows;
|
|
357
|
+
this.itemChanged.emit(w);
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
this.resizing = null;
|
|
361
|
+
this.placeholder = null;
|
|
362
|
+
this.previewWidgets = [];
|
|
363
|
+
this.updateContainerHeight();
|
|
364
|
+
this.layoutChanged.emit(this.widgets);
|
|
365
|
+
this.cdr.markForCheck();
|
|
366
|
+
}
|
|
367
|
+
// ── Utilities ──
|
|
368
|
+
compactAndApply() {
|
|
369
|
+
if (!this.widgets?.length)
|
|
370
|
+
return;
|
|
371
|
+
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);
|
|
372
|
+
for (const rw of compacted) {
|
|
373
|
+
const w = this.widgets.find(ww => ww.id === rw.id);
|
|
374
|
+
if (w) {
|
|
375
|
+
w.x = rw.x;
|
|
376
|
+
w.y = rw.y;
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
updateContainerHeight() {
|
|
381
|
+
if (!this.widgets?.length) {
|
|
382
|
+
this.containerHeight = 400;
|
|
383
|
+
return;
|
|
384
|
+
}
|
|
385
|
+
const maxRow = GridEngine.getGridHeight(this.widgets.map(w => ({ id: w.id, x: w.x, y: w.y, cols: w.cols, rows: w.rows })));
|
|
386
|
+
this.containerHeight = this.rowToPixel(maxRow) + this.rowHeight + this.gap;
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
CustomGridComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CustomGridComponent, deps: [{ token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
390
|
+
CustomGridComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CustomGridComponent, selector: "app-grid", inputs: { widgets: "widgets", columns: "columns", gap: "gap", rowHeight: "rowHeight", minItemCols: "minItemCols", minItemRows: "minItemRows" }, outputs: { itemChanged: "itemChanged", layoutChanged: "layoutChanged" }, queries: [{ propertyName: "cellTemplate", first: true, predicate: GridCellDirective, descendants: true, read: TemplateRef }], viewQueries: [{ propertyName: "gridContainer", first: true, predicate: ["gridContainer"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: `
|
|
391
|
+
<div
|
|
392
|
+
#gridContainer
|
|
393
|
+
class="grid-container"
|
|
394
|
+
[style.height.px]="containerHeight"
|
|
395
|
+
>
|
|
396
|
+
<!-- Placeholder preview -->
|
|
397
|
+
<div
|
|
398
|
+
class="grid-placeholder"
|
|
399
|
+
*ngIf="placeholder"
|
|
400
|
+
[style.left.px]="placeholder.left"
|
|
401
|
+
[style.top.px]="placeholder.top"
|
|
402
|
+
[style.width.px]="placeholder.width"
|
|
403
|
+
[style.height.px]="placeholder.height"
|
|
404
|
+
></div>
|
|
405
|
+
|
|
406
|
+
<!-- Widget items -->
|
|
407
|
+
<div
|
|
408
|
+
*ngFor="let widget of widgets; trackBy: trackByFn"
|
|
409
|
+
class="grid-item"
|
|
410
|
+
[class.is-dragging]="dragging?.id === widget.id"
|
|
411
|
+
[class.is-resizing]="resizing?.id === widget.id"
|
|
412
|
+
[style.left.px]="getItemLeft(widget)"
|
|
413
|
+
[style.top.px]="getItemTop(widget)"
|
|
414
|
+
[style.width.px]="getItemWidth(widget)"
|
|
415
|
+
[style.height.px]="getItemHeight(widget)"
|
|
416
|
+
[style.z-index]="(dragging?.id === widget.id || resizing?.id === widget.id) ? 100 : 1"
|
|
417
|
+
(mousedown)="onDragStart($event, widget)"
|
|
418
|
+
>
|
|
419
|
+
<!-- Content stamped from parent template, context carries the widget -->
|
|
420
|
+
<ng-container
|
|
421
|
+
*ngIf="cellTemplate"
|
|
422
|
+
[ngTemplateOutlet]="cellTemplate"
|
|
423
|
+
[ngTemplateOutletContext]="{ widget: widget }"
|
|
424
|
+
></ng-container>
|
|
425
|
+
|
|
426
|
+
<!-- Resize handle -->
|
|
427
|
+
<div class="resize-handle" (mousedown)="onResizeStart($event, widget)">
|
|
428
|
+
<svg width="12" height="12" viewBox="0 0 12 12">
|
|
429
|
+
<path d="M11 1v10H1" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
|
|
430
|
+
<path d="M11 5v6H5" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
|
|
431
|
+
<path d="M11 9v2H9" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
|
|
432
|
+
</svg>
|
|
433
|
+
</div>
|
|
434
|
+
</div>
|
|
435
|
+
</div>
|
|
436
|
+
`, isInline: true, styles: [":host{display:block;width:100%;height:100%}.grid-container{position:relative;width:100%;min-height:100%;transition:height .25s ease}.grid-placeholder{position:absolute;background:var(--dash-accent-color, #0a84ff);opacity:.12;border-radius:24px;border:2px dashed var(--dash-accent-color, #0a84ff);transition:left .15s ease,top .15s ease,width .15s ease,height .15s ease;pointer-events:none;z-index:0}.grid-item{position:absolute;display:flex;flex-direction:column;transition:left .25s ease,top .25s ease,width .25s ease,height .25s ease;border-radius:24px;overflow:hidden}.grid-item.is-dragging,.grid-item.is-resizing{transition:none!important;opacity:.88;overflow:visible;filter:drop-shadow(0 20px 40px rgba(0,0,0,.45))}.resize-handle{position:absolute;right:4px;bottom:4px;width:22px;height:22px;cursor:nwse-resize;display:flex;align-items:center;justify-content:center;color:#ffffff40;border-radius:6px;transition:color .2s,background .2s;z-index:20}.resize-handle:hover{color:var(--dash-accent-color, #0a84ff);background:rgba(255,255,255,.06)}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
437
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CustomGridComponent, decorators: [{
|
|
438
|
+
type: Component,
|
|
439
|
+
args: [{ selector: 'app-grid', template: `
|
|
440
|
+
<div
|
|
441
|
+
#gridContainer
|
|
442
|
+
class="grid-container"
|
|
443
|
+
[style.height.px]="containerHeight"
|
|
444
|
+
>
|
|
445
|
+
<!-- Placeholder preview -->
|
|
446
|
+
<div
|
|
447
|
+
class="grid-placeholder"
|
|
448
|
+
*ngIf="placeholder"
|
|
449
|
+
[style.left.px]="placeholder.left"
|
|
450
|
+
[style.top.px]="placeholder.top"
|
|
451
|
+
[style.width.px]="placeholder.width"
|
|
452
|
+
[style.height.px]="placeholder.height"
|
|
453
|
+
></div>
|
|
454
|
+
|
|
455
|
+
<!-- Widget items -->
|
|
456
|
+
<div
|
|
457
|
+
*ngFor="let widget of widgets; trackBy: trackByFn"
|
|
458
|
+
class="grid-item"
|
|
459
|
+
[class.is-dragging]="dragging?.id === widget.id"
|
|
460
|
+
[class.is-resizing]="resizing?.id === widget.id"
|
|
461
|
+
[style.left.px]="getItemLeft(widget)"
|
|
462
|
+
[style.top.px]="getItemTop(widget)"
|
|
463
|
+
[style.width.px]="getItemWidth(widget)"
|
|
464
|
+
[style.height.px]="getItemHeight(widget)"
|
|
465
|
+
[style.z-index]="(dragging?.id === widget.id || resizing?.id === widget.id) ? 100 : 1"
|
|
466
|
+
(mousedown)="onDragStart($event, widget)"
|
|
467
|
+
>
|
|
468
|
+
<!-- Content stamped from parent template, context carries the widget -->
|
|
469
|
+
<ng-container
|
|
470
|
+
*ngIf="cellTemplate"
|
|
471
|
+
[ngTemplateOutlet]="cellTemplate"
|
|
472
|
+
[ngTemplateOutletContext]="{ widget: widget }"
|
|
473
|
+
></ng-container>
|
|
474
|
+
|
|
475
|
+
<!-- Resize handle -->
|
|
476
|
+
<div class="resize-handle" (mousedown)="onResizeStart($event, widget)">
|
|
477
|
+
<svg width="12" height="12" viewBox="0 0 12 12">
|
|
478
|
+
<path d="M11 1v10H1" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
|
|
479
|
+
<path d="M11 5v6H5" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
|
|
480
|
+
<path d="M11 9v2H9" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round"/>
|
|
481
|
+
</svg>
|
|
482
|
+
</div>
|
|
483
|
+
</div>
|
|
484
|
+
</div>
|
|
485
|
+
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: [":host{display:block;width:100%;height:100%}.grid-container{position:relative;width:100%;min-height:100%;transition:height .25s ease}.grid-placeholder{position:absolute;background:var(--dash-accent-color, #0a84ff);opacity:.12;border-radius:24px;border:2px dashed var(--dash-accent-color, #0a84ff);transition:left .15s ease,top .15s ease,width .15s ease,height .15s ease;pointer-events:none;z-index:0}.grid-item{position:absolute;display:flex;flex-direction:column;transition:left .25s ease,top .25s ease,width .25s ease,height .25s ease;border-radius:24px;overflow:hidden}.grid-item.is-dragging,.grid-item.is-resizing{transition:none!important;opacity:.88;overflow:visible;filter:drop-shadow(0 20px 40px rgba(0,0,0,.45))}.resize-handle{position:absolute;right:4px;bottom:4px;width:22px;height:22px;cursor:nwse-resize;display:flex;align-items:center;justify-content:center;color:#ffffff40;border-radius:6px;transition:color .2s,background .2s;z-index:20}.resize-handle:hover{color:var(--dash-accent-color, #0a84ff);background:rgba(255,255,255,.06)}\n"] }]
|
|
486
|
+
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { widgets: [{
|
|
487
|
+
type: Input
|
|
488
|
+
}], columns: [{
|
|
489
|
+
type: Input
|
|
490
|
+
}], gap: [{
|
|
491
|
+
type: Input
|
|
492
|
+
}], rowHeight: [{
|
|
493
|
+
type: Input
|
|
494
|
+
}], minItemCols: [{
|
|
495
|
+
type: Input
|
|
496
|
+
}], minItemRows: [{
|
|
497
|
+
type: Input
|
|
498
|
+
}], itemChanged: [{
|
|
499
|
+
type: Output
|
|
500
|
+
}], layoutChanged: [{
|
|
501
|
+
type: Output
|
|
502
|
+
}], gridContainer: [{
|
|
503
|
+
type: ViewChild,
|
|
504
|
+
args: ['gridContainer', { static: true }]
|
|
505
|
+
}], cellTemplate: [{
|
|
506
|
+
type: ContentChild,
|
|
507
|
+
args: [GridCellDirective, { read: TemplateRef }]
|
|
508
|
+
}] } });
|
|
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-accent-color, #0a84ff);\n      opacity: 0.12;\n      border-radius: 24px;\n      border: 2px dashed 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      filter: drop-shadow(0 20px 40px rgba(0,0,0,0.45));\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: 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"]}
|