@ngstato/angular 0.4.0 → 0.4.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,6 +1,6 @@
1
- import { Component, signal } from '@angular/core';
1
+ import { Component, signal, computed } from '@angular/core';
2
2
  import { devTools } from '@ngstato/core';
3
- import { JsonPipe } from '@angular/common';
3
+ import { JsonPipe, KeyValuePipe } from '@angular/common';
4
4
  import * as i0 from "@angular/core";
5
5
  export class StatoDevToolsComponent {
6
6
  unsub;
@@ -10,11 +10,36 @@ export class StatoDevToolsComponent {
10
10
  activeTab = signal('actions');
11
11
  logs = signal([]);
12
12
  selectedLog = signal(null);
13
- // Position et taille
13
+ activeLogId = signal(null);
14
+ isTimeTraveling = signal(false);
15
+ // Snapshot global — latest known state per store
16
+ globalState = computed(() => {
17
+ const seen = new Map();
18
+ for (const log of this.logs()) {
19
+ if (!seen.has(log.storeName)) {
20
+ seen.set(log.storeName, log.nextState);
21
+ }
22
+ }
23
+ return seen;
24
+ });
25
+ // Time-travel computed
26
+ canUndo = computed(() => this.logs().length > 0);
27
+ canRedo = computed(() => {
28
+ if (!this.isTimeTraveling())
29
+ return false;
30
+ const id = this.activeLogId();
31
+ if (id === null)
32
+ return false;
33
+ if (id === -1)
34
+ return this.logs().length > 0;
35
+ const idx = this.logs().findIndex(l => l.id === id);
36
+ return idx > 0; // can go forward (lower index = newer)
37
+ });
38
+ // Position & size
14
39
  posX = signal(24);
15
- posY = signal(window.innerHeight - 500);
16
- panelWidth = signal(420);
17
- panelHeight = signal(460);
40
+ posY = signal(window.innerHeight - 520);
41
+ panelWidth = signal(440);
42
+ panelHeight = signal(480);
18
43
  // Drag state
19
44
  isDragging = false;
20
45
  isResizing = false;
@@ -31,6 +56,8 @@ export class StatoDevToolsComponent {
31
56
  this.unsub = devTools.subscribe((state) => {
32
57
  this.logs.set(state.logs);
33
58
  this.isOpen.set(state.isOpen);
59
+ this.activeLogId.set(state.activeLogId);
60
+ this.isTimeTraveling.set(state.isTimeTraveling);
34
61
  });
35
62
  document.addEventListener('mousemove', this.boundMouseMove);
36
63
  document.addEventListener('mouseup', this.boundMouseUp);
@@ -50,6 +77,65 @@ export class StatoDevToolsComponent {
50
77
  formatTime(iso) {
51
78
  return new Date(iso).toTimeString().slice(0, 8);
52
79
  }
80
+ // ── Time-travel actions ────────────────────────────
81
+ onTravelTo(log) {
82
+ // Toggle detail view
83
+ this.selectLog(log);
84
+ // Jump to this action's state
85
+ devTools.travelTo(log.id);
86
+ }
87
+ onUndo() { devTools.undo(); }
88
+ onRedo() { devTools.redo(); }
89
+ onResume() { devTools.resume(); }
90
+ onReplay(log, event) {
91
+ event.stopPropagation();
92
+ devTools.replay(log.id);
93
+ }
94
+ isFutureLog(log) {
95
+ if (!this.isTimeTraveling())
96
+ return false;
97
+ const activeId = this.activeLogId();
98
+ if (activeId === null)
99
+ return false;
100
+ if (activeId === -1)
101
+ return true; // all logs are "future"
102
+ const activeIdx = this.logs().findIndex(l => l.id === activeId);
103
+ const logIdx = this.logs().findIndex(l => l.id === log.id);
104
+ return logIdx < activeIdx; // newer logs (lower index) are "future"
105
+ }
106
+ // ── Export/Import ──────────────────────────────────
107
+ onExport() {
108
+ const snapshot = devTools.exportSnapshot();
109
+ const json = JSON.stringify(snapshot, null, 2);
110
+ const blob = new Blob([json], { type: 'application/json' });
111
+ const url = URL.createObjectURL(blob);
112
+ const a = document.createElement('a');
113
+ a.href = url;
114
+ a.download = `ngstato-snapshot-${new Date().toISOString().slice(0, 19).replace(/:/g, '-')}.json`;
115
+ a.click();
116
+ URL.revokeObjectURL(url);
117
+ }
118
+ onImport() {
119
+ const input = document.querySelector('input[type="file"]');
120
+ input?.click();
121
+ }
122
+ onFileSelected(event) {
123
+ const file = event.target.files?.[0];
124
+ if (!file)
125
+ return;
126
+ const reader = new FileReader();
127
+ reader.onload = () => {
128
+ try {
129
+ const snapshot = JSON.parse(reader.result);
130
+ devTools.importSnapshot(snapshot);
131
+ }
132
+ catch (e) {
133
+ console.error('[ngStato DevTools] Invalid snapshot file:', e);
134
+ }
135
+ };
136
+ reader.readAsText(file);
137
+ event.target.value = '';
138
+ }
53
139
  // ── Drag ───────────────────────────────────────────
54
140
  onDragStart(e) {
55
141
  if (e.target.classList.contains('btn-icon'))
@@ -88,8 +174,8 @@ export class StatoDevToolsComponent {
88
174
  this.isResizing = false;
89
175
  }
90
176
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StatoDevToolsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
91
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: StatoDevToolsComponent, isStandalone: true, selector: "ngstato-devtools", ngImport: i0, template: `
92
- <!-- Bouton flottant -->
177
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: StatoDevToolsComponent, isStandalone: true, selector: "ngstato-devtools, stato-devtools", ngImport: i0, template: `
178
+ <!-- Floating button -->
93
179
  @if (!isOpen()) {
94
180
  <button class="devtools-fab" (click)="toggle()">
95
181
  🛠 Stato
@@ -103,7 +189,7 @@ export class StatoDevToolsComponent {
103
189
  [class.devtools-panel--minimized]="isMinimized()"
104
190
  [style.left.px]="posX()"
105
191
  [style.top.px]="posY()"
106
- [style.width.px]="isMinimized() ? 200 : panelWidth()"
192
+ [style.width.px]="isMinimized() ? 220 : panelWidth()"
107
193
  [style.height]="isMinimized() ? 'auto' : panelHeight() + 'px'"
108
194
  >
109
195
 
@@ -112,19 +198,24 @@ export class StatoDevToolsComponent {
112
198
  class="devtools-header"
113
199
  (mousedown)="onDragStart($event)"
114
200
  >
115
- <span class="devtools-title">🛠 Stato</span>
201
+ <span class="devtools-title">
202
+ 🛠 Stato
203
+ @if (isTimeTraveling()) {
204
+ <span class="tt-badge">TIME-TRAVEL</span>
205
+ }
206
+ </span>
116
207
  <div class="devtools-header-actions">
117
208
  @if (!isMinimized()) {
118
- <button class="btn-icon" (click)="clear()" title="Vider">🗑</button>
209
+ <button class="btn-icon" (click)="clear()" title="Clear">🗑</button>
119
210
  }
120
- <button class="btn-icon" (click)="toggleMinimize()" title="Minimiser/Agrandir">
211
+ <button class="btn-icon" (click)="toggleMinimize()" title="Minimize">
121
212
  {{ isMinimized() ? '▲' : '▼' }}
122
213
  </button>
123
- <button class="btn-icon" (click)="toggle()" title="Fermer">✕</button>
214
+ <button class="btn-icon" (click)="toggle()" title="Close">✕</button>
124
215
  </div>
125
216
  </div>
126
217
 
127
- <!-- Resize handle — coin bas droite -->
218
+ <!-- Resize handle -->
128
219
  @if (!isMinimized()) {
129
220
  <div
130
221
  class="devtools-resize"
@@ -152,17 +243,55 @@ export class StatoDevToolsComponent {
152
243
  </button>
153
244
  </div>
154
245
 
155
- <!-- Tab Actions -->
246
+ <!-- Time-travel toolbar -->
247
+ @if (activeTab() === 'actions' && logs().length) {
248
+ <div class="tt-toolbar">
249
+ <button
250
+ class="tt-btn"
251
+ (click)="onUndo()"
252
+ [disabled]="!canUndo()"
253
+ title="Undo (step back)"
254
+ >⏪</button>
255
+ <button
256
+ class="tt-btn"
257
+ (click)="onRedo()"
258
+ [disabled]="!canRedo()"
259
+ title="Redo (step forward)"
260
+ >⏩</button>
261
+ @if (isTimeTraveling()) {
262
+ <button
263
+ class="tt-btn tt-btn--resume"
264
+ (click)="onResume()"
265
+ title="Resume live state"
266
+ >▶ Live</button>
267
+ }
268
+ <div class="tt-spacer"></div>
269
+ <button
270
+ class="tt-btn tt-btn--export"
271
+ (click)="onExport()"
272
+ title="Export state snapshot (JSON)"
273
+ >📤</button>
274
+ <button
275
+ class="tt-btn tt-btn--import"
276
+ (click)="onImport()"
277
+ title="Import state snapshot"
278
+ >📥</button>
279
+ </div>
280
+ }
281
+
282
+ <!-- Tab: Actions -->
156
283
  @if (activeTab() === 'actions') {
157
284
  <div class="devtools-content">
158
285
  @if (!logs().length) {
159
- <div class="devtools-empty">Aucune action pour l'instant</div>
286
+ <div class="devtools-empty">No actions yet</div>
160
287
  }
161
288
  @for (log of logs(); track log.id) {
162
289
  <div
163
290
  class="log-item"
164
291
  [class.log-item--error]="log.status === 'error'"
165
- (click)="selectLog(log)"
292
+ [class.log-item--active]="activeLogId() === log.id"
293
+ [class.log-item--future]="isFutureLog(log)"
294
+ (click)="onTravelTo(log)"
166
295
  >
167
296
  <div class="log-item__left">
168
297
  <span class="log-status">{{ log.status === 'success' ? '✓' : '✗' }}</span>
@@ -170,11 +299,16 @@ export class StatoDevToolsComponent {
170
299
  </div>
171
300
  <div class="log-item__right">
172
301
  @if (log.status === 'error') {
173
- <span class="log-error-badge">erreur</span>
302
+ <span class="log-error-badge">error</span>
174
303
  } @else {
175
304
  <span class="log-duration">{{ log.duration }}ms</span>
176
305
  }
177
306
  <span class="log-time">{{ formatTime(log.at) }}</span>
307
+ <button
308
+ class="btn-icon btn-replay"
309
+ (click)="onReplay(log, $event)"
310
+ title="Replay this action"
311
+ >🔄</button>
178
312
  </div>
179
313
  </div>
180
314
 
@@ -184,11 +318,11 @@ export class StatoDevToolsComponent {
184
318
  <div class="log-detail__error">{{ log.error }}</div>
185
319
  }
186
320
  <div class="log-detail__section">
187
- <span class="log-detail__label">Avant</span>
321
+ <span class="log-detail__label">Before</span>
188
322
  <pre>{{ log.prevState | json }}</pre>
189
323
  </div>
190
324
  <div class="log-detail__section">
191
- <span class="log-detail__label">Après</span>
325
+ <span class="log-detail__label">After</span>
192
326
  <pre>{{ log.nextState | json }}</pre>
193
327
  </div>
194
328
  </div>
@@ -197,13 +331,18 @@ export class StatoDevToolsComponent {
197
331
  </div>
198
332
  }
199
333
 
200
- <!-- Tab State -->
334
+ <!-- Tab: State -->
201
335
  @if (activeTab() === 'state') {
202
336
  <div class="devtools-content">
203
- @if (logs().length) {
204
- <pre class="state-view">{{ logs()[0].nextState | json }}</pre>
337
+ @if (globalState().size) {
338
+ @for (entry of globalState() | keyvalue; track entry.key) {
339
+ <div class="state-store-block">
340
+ <div class="state-store-name">{{ entry.key }}</div>
341
+ <pre class="state-view">{{ entry.value | json }}</pre>
342
+ </div>
343
+ }
205
344
  } @else {
206
- <div class="devtools-empty">Aucun state disponible</div>
345
+ <div class="devtools-empty">No state available</div>
207
346
  }
208
347
  </div>
209
348
  }
@@ -211,12 +350,21 @@ export class StatoDevToolsComponent {
211
350
 
212
351
  </div>
213
352
  }
214
- `, isInline: true, styles: [".devtools-fab{position:fixed;bottom:1.5rem;left:1.5rem;background:#1e293b;color:#fff;border:none;border-radius:999px;padding:.5rem 1rem;font-size:.85rem;font-weight:600;cursor:pointer;z-index:9999;box-shadow:0 4px 12px #0000004d;transition:background .15s}.devtools-fab:hover{background:#334155}.devtools-panel{position:fixed;background:#0f172a;border-radius:12px;box-shadow:0 8px 32px #0006;z-index:9999;display:flex;flex-direction:column;overflow:hidden;font-family:Courier New,monospace;min-width:200px;min-height:40px}.devtools-panel--minimized{border-radius:8px}.devtools-header{display:flex;justify-content:space-between;align-items:center;padding:.6rem .75rem;background:#1e293b;border-bottom:1px solid #334155;cursor:grab;-webkit-user-select:none;user-select:none}.devtools-header:active{cursor:grabbing}.devtools-title{color:#e2e8f0;font-size:.82rem;font-weight:600;font-family:system-ui}.devtools-header-actions{display:flex;gap:.25rem}.btn-icon{background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.8rem;padding:.15rem .35rem;border-radius:4px;line-height:1}.btn-icon:hover{background:#334155;color:#fff}.devtools-resize{position:absolute;bottom:2px;right:4px;color:#334155;font-size:.9rem;cursor:nwse-resize;-webkit-user-select:none;user-select:none;line-height:1}.devtools-resize:hover{color:#64748b}.devtools-tabs{display:flex;background:#1e293b;border-bottom:1px solid #334155}.tab{padding:.4rem .75rem;background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.78rem;font-family:system-ui}.tab:hover{color:#e2e8f0}.tab--active{color:#3b82f6;border-bottom:2px solid #3b82f6}.devtools-content{overflow-y:auto;flex:1;padding:.25rem 0}.devtools-empty{padding:2rem;text-align:center;color:#475569;font-size:.78rem;font-family:system-ui}.log-item{display:flex;justify-content:space-between;align-items:center;padding:.35rem .75rem;cursor:pointer;border-bottom:1px solid #1e293b}.log-item:hover{background:#1e293b}.log-item--error{background:#1a0a0a}.log-item__left{display:flex;align-items:center;gap:.4rem;overflow:hidden}.log-item__right{display:flex;align-items:center;gap:.4rem;flex-shrink:0}.log-status{font-size:.72rem;color:#22c55e;flex-shrink:0}.log-item--error .log-status{color:#ef4444}.log-name{color:#e2e8f0;font-size:.75rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.log-duration{color:#64748b;font-size:.7rem}.log-time{color:#475569;font-size:.68rem}.log-error-badge{background:#7f1d1d;color:#fca5a5;font-size:.68rem;padding:.1rem .35rem;border-radius:4px}.log-detail{background:#0a0f1a;padding:.6rem .75rem;border-left:3px solid #3b82f6;margin:0 .4rem .4rem;border-radius:0 4px 4px 0}.log-detail__error{color:#fca5a5;font-size:.72rem;margin-bottom:.4rem}.log-detail__section{margin-bottom:.4rem}.log-detail__label{color:#64748b;font-size:.68rem;display:block;margin-bottom:.2rem;font-family:system-ui}pre{color:#86efac;font-size:.7rem;margin:0;white-space:pre-wrap;word-break:break-all;max-height:140px;overflow-y:auto}.state-view{color:#86efac;font-size:.7rem;padding:.75rem;margin:0;white-space:pre-wrap;word-break:break-all}\n"], dependencies: [{ kind: "pipe", type: JsonPipe, name: "json" }] });
353
+
354
+ <!-- Hidden file input for import -->
355
+ <input
356
+ #fileInput
357
+ type="file"
358
+ accept=".json"
359
+ style="display: none"
360
+ (change)="onFileSelected($event)"
361
+ />
362
+ `, isInline: true, styles: [":host{font-family:system-ui,-apple-system,sans-serif}.devtools-fab{position:fixed;bottom:1.5rem;left:1.5rem;background:#1e293b;color:#fff;border:none;border-radius:999px;padding:.5rem 1rem;font-size:.85rem;font-weight:600;cursor:pointer;z-index:9999;box-shadow:0 4px 12px #0000004d;transition:background .15s}.devtools-fab:hover{background:#334155}.devtools-panel{position:fixed;background:#0f172a;border-radius:12px;box-shadow:0 8px 32px #00000080;z-index:9999;display:flex;flex-direction:column;overflow:hidden;font-family:Courier New,monospace;min-width:200px;min-height:40px;border:1px solid #1e293b}.devtools-panel--minimized{border-radius:8px}.devtools-header{display:flex;justify-content:space-between;align-items:center;padding:.6rem .75rem;background:#1e293b;border-bottom:1px solid #334155;cursor:grab;-webkit-user-select:none;user-select:none}.devtools-header:active{cursor:grabbing}.devtools-title{color:#e2e8f0;font-size:.82rem;font-weight:600;font-family:system-ui;display:flex;align-items:center;gap:.4rem}.tt-badge{background:#7c3aed;color:#fff;font-size:.6rem;padding:.12rem .4rem;border-radius:4px;font-weight:700;letter-spacing:.05em;animation:tt-pulse 1.5s ease-in-out infinite}@keyframes tt-pulse{0%,to{opacity:1}50%{opacity:.6}}.devtools-header-actions{display:flex;gap:.25rem}.btn-icon{background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.8rem;padding:.15rem .35rem;border-radius:4px;line-height:1}.btn-icon:hover{background:#334155;color:#fff}.devtools-resize{position:absolute;bottom:2px;right:4px;color:#334155;font-size:.9rem;cursor:nwse-resize;-webkit-user-select:none;user-select:none;line-height:1}.devtools-resize:hover{color:#64748b}.devtools-tabs{display:flex;background:#1e293b;border-bottom:1px solid #334155}.tab{padding:.4rem .75rem;background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.78rem;font-family:system-ui;transition:color .15s}.tab:hover{color:#e2e8f0}.tab--active{color:#3b82f6;border-bottom:2px solid #3b82f6}.tt-toolbar{display:flex;align-items:center;gap:.2rem;padding:.3rem .5rem;background:#0f172a;border-bottom:1px solid #1e293b}.tt-btn{background:#1e293b;border:1px solid #334155;color:#94a3b8;padding:.2rem .5rem;border-radius:4px;font-size:.72rem;cursor:pointer;transition:all .15s;font-family:system-ui}.tt-btn:hover:not(:disabled){background:#334155;color:#e2e8f0;border-color:#475569}.tt-btn:disabled{opacity:.3;cursor:not-allowed}.tt-btn--resume{background:#7c3aed;border-color:#7c3aed;color:#fff;font-weight:600}.tt-btn--resume:hover:not(:disabled){background:#6d28d9}.tt-btn--export,.tt-btn--import{font-size:.68rem}.tt-spacer{flex:1}.devtools-content{overflow-y:auto;flex:1;padding:.25rem 0}.devtools-empty{padding:2rem;text-align:center;color:#475569;font-size:.78rem;font-family:system-ui}.log-item{display:flex;justify-content:space-between;align-items:center;padding:.35rem .75rem;cursor:pointer;border-bottom:1px solid #1e293b;transition:background .1s}.log-item:hover{background:#1e293b}.log-item--error{background:#1a0a0a}.log-item--active{background:#1e1b4b!important;border-left:3px solid #7c3aed}.log-item--future{opacity:.35}.log-item__left{display:flex;align-items:center;gap:.4rem;overflow:hidden;flex:1}.log-item__right{display:flex;align-items:center;gap:.35rem;flex-shrink:0}.log-status{font-size:.72rem;color:#22c55e;flex-shrink:0}.log-item--error .log-status{color:#ef4444}.log-name{color:#e2e8f0;font-size:.75rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.log-duration{color:#64748b;font-size:.7rem}.log-time{color:#475569;font-size:.68rem}.log-error-badge{background:#7f1d1d;color:#fca5a5;font-size:.68rem;padding:.1rem .35rem;border-radius:4px}.btn-replay{font-size:.65rem;opacity:.5}.btn-replay:hover{opacity:1}.log-detail{background:#0a0f1a;padding:.6rem .75rem;border-left:3px solid #3b82f6;margin:0 .4rem .4rem;border-radius:0 4px 4px 0}.log-detail__error{color:#fca5a5;font-size:.72rem;margin-bottom:.4rem}.log-detail__section{margin-bottom:.4rem}.log-detail__label{color:#64748b;font-size:.68rem;display:block;margin-bottom:.2rem;font-family:system-ui}pre{color:#86efac;font-size:.7rem;margin:0;white-space:pre-wrap;word-break:break-all;max-height:140px;overflow-y:auto}.state-store-block{border-bottom:1px solid #1e293b}.state-store-block:last-child{border-bottom:none}.state-store-name{padding:.4rem .75rem .2rem;color:#3b82f6;font-size:.72rem;font-family:system-ui;font-weight:600;letter-spacing:.03em;text-transform:uppercase}.state-view{color:#86efac;font-size:.7rem;padding:.25rem .75rem .75rem;margin:0;white-space:pre-wrap;word-break:break-all}\n"], dependencies: [{ kind: "pipe", type: JsonPipe, name: "json" }, { kind: "pipe", type: KeyValuePipe, name: "keyvalue" }] });
215
363
  }
216
364
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StatoDevToolsComponent, decorators: [{
217
365
  type: Component,
218
- args: [{ selector: 'ngstato-devtools', standalone: true, imports: [JsonPipe], template: `
219
- <!-- Bouton flottant -->
366
+ args: [{ selector: 'ngstato-devtools, stato-devtools', standalone: true, imports: [JsonPipe, KeyValuePipe], template: `
367
+ <!-- Floating button -->
220
368
  @if (!isOpen()) {
221
369
  <button class="devtools-fab" (click)="toggle()">
222
370
  🛠 Stato
@@ -230,7 +378,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
230
378
  [class.devtools-panel--minimized]="isMinimized()"
231
379
  [style.left.px]="posX()"
232
380
  [style.top.px]="posY()"
233
- [style.width.px]="isMinimized() ? 200 : panelWidth()"
381
+ [style.width.px]="isMinimized() ? 220 : panelWidth()"
234
382
  [style.height]="isMinimized() ? 'auto' : panelHeight() + 'px'"
235
383
  >
236
384
 
@@ -239,19 +387,24 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
239
387
  class="devtools-header"
240
388
  (mousedown)="onDragStart($event)"
241
389
  >
242
- <span class="devtools-title">🛠 Stato</span>
390
+ <span class="devtools-title">
391
+ 🛠 Stato
392
+ @if (isTimeTraveling()) {
393
+ <span class="tt-badge">TIME-TRAVEL</span>
394
+ }
395
+ </span>
243
396
  <div class="devtools-header-actions">
244
397
  @if (!isMinimized()) {
245
- <button class="btn-icon" (click)="clear()" title="Vider">🗑</button>
398
+ <button class="btn-icon" (click)="clear()" title="Clear">🗑</button>
246
399
  }
247
- <button class="btn-icon" (click)="toggleMinimize()" title="Minimiser/Agrandir">
400
+ <button class="btn-icon" (click)="toggleMinimize()" title="Minimize">
248
401
  {{ isMinimized() ? '▲' : '▼' }}
249
402
  </button>
250
- <button class="btn-icon" (click)="toggle()" title="Fermer">✕</button>
403
+ <button class="btn-icon" (click)="toggle()" title="Close">✕</button>
251
404
  </div>
252
405
  </div>
253
406
 
254
- <!-- Resize handle — coin bas droite -->
407
+ <!-- Resize handle -->
255
408
  @if (!isMinimized()) {
256
409
  <div
257
410
  class="devtools-resize"
@@ -279,17 +432,55 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
279
432
  </button>
280
433
  </div>
281
434
 
282
- <!-- Tab Actions -->
435
+ <!-- Time-travel toolbar -->
436
+ @if (activeTab() === 'actions' && logs().length) {
437
+ <div class="tt-toolbar">
438
+ <button
439
+ class="tt-btn"
440
+ (click)="onUndo()"
441
+ [disabled]="!canUndo()"
442
+ title="Undo (step back)"
443
+ >⏪</button>
444
+ <button
445
+ class="tt-btn"
446
+ (click)="onRedo()"
447
+ [disabled]="!canRedo()"
448
+ title="Redo (step forward)"
449
+ >⏩</button>
450
+ @if (isTimeTraveling()) {
451
+ <button
452
+ class="tt-btn tt-btn--resume"
453
+ (click)="onResume()"
454
+ title="Resume live state"
455
+ >▶ Live</button>
456
+ }
457
+ <div class="tt-spacer"></div>
458
+ <button
459
+ class="tt-btn tt-btn--export"
460
+ (click)="onExport()"
461
+ title="Export state snapshot (JSON)"
462
+ >📤</button>
463
+ <button
464
+ class="tt-btn tt-btn--import"
465
+ (click)="onImport()"
466
+ title="Import state snapshot"
467
+ >📥</button>
468
+ </div>
469
+ }
470
+
471
+ <!-- Tab: Actions -->
283
472
  @if (activeTab() === 'actions') {
284
473
  <div class="devtools-content">
285
474
  @if (!logs().length) {
286
- <div class="devtools-empty">Aucune action pour l'instant</div>
475
+ <div class="devtools-empty">No actions yet</div>
287
476
  }
288
477
  @for (log of logs(); track log.id) {
289
478
  <div
290
479
  class="log-item"
291
480
  [class.log-item--error]="log.status === 'error'"
292
- (click)="selectLog(log)"
481
+ [class.log-item--active]="activeLogId() === log.id"
482
+ [class.log-item--future]="isFutureLog(log)"
483
+ (click)="onTravelTo(log)"
293
484
  >
294
485
  <div class="log-item__left">
295
486
  <span class="log-status">{{ log.status === 'success' ? '✓' : '✗' }}</span>
@@ -297,11 +488,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
297
488
  </div>
298
489
  <div class="log-item__right">
299
490
  @if (log.status === 'error') {
300
- <span class="log-error-badge">erreur</span>
491
+ <span class="log-error-badge">error</span>
301
492
  } @else {
302
493
  <span class="log-duration">{{ log.duration }}ms</span>
303
494
  }
304
495
  <span class="log-time">{{ formatTime(log.at) }}</span>
496
+ <button
497
+ class="btn-icon btn-replay"
498
+ (click)="onReplay(log, $event)"
499
+ title="Replay this action"
500
+ >🔄</button>
305
501
  </div>
306
502
  </div>
307
503
 
@@ -311,11 +507,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
311
507
  <div class="log-detail__error">{{ log.error }}</div>
312
508
  }
313
509
  <div class="log-detail__section">
314
- <span class="log-detail__label">Avant</span>
510
+ <span class="log-detail__label">Before</span>
315
511
  <pre>{{ log.prevState | json }}</pre>
316
512
  </div>
317
513
  <div class="log-detail__section">
318
- <span class="log-detail__label">Après</span>
514
+ <span class="log-detail__label">After</span>
319
515
  <pre>{{ log.nextState | json }}</pre>
320
516
  </div>
321
517
  </div>
@@ -324,13 +520,18 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
324
520
  </div>
325
521
  }
326
522
 
327
- <!-- Tab State -->
523
+ <!-- Tab: State -->
328
524
  @if (activeTab() === 'state') {
329
525
  <div class="devtools-content">
330
- @if (logs().length) {
331
- <pre class="state-view">{{ logs()[0].nextState | json }}</pre>
526
+ @if (globalState().size) {
527
+ @for (entry of globalState() | keyvalue; track entry.key) {
528
+ <div class="state-store-block">
529
+ <div class="state-store-name">{{ entry.key }}</div>
530
+ <pre class="state-view">{{ entry.value | json }}</pre>
531
+ </div>
532
+ }
332
533
  } @else {
333
- <div class="devtools-empty">Aucun state disponible</div>
534
+ <div class="devtools-empty">No state available</div>
334
535
  }
335
536
  </div>
336
537
  }
@@ -338,6 +539,15 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
338
539
 
339
540
  </div>
340
541
  }
341
- `, styles: [".devtools-fab{position:fixed;bottom:1.5rem;left:1.5rem;background:#1e293b;color:#fff;border:none;border-radius:999px;padding:.5rem 1rem;font-size:.85rem;font-weight:600;cursor:pointer;z-index:9999;box-shadow:0 4px 12px #0000004d;transition:background .15s}.devtools-fab:hover{background:#334155}.devtools-panel{position:fixed;background:#0f172a;border-radius:12px;box-shadow:0 8px 32px #0006;z-index:9999;display:flex;flex-direction:column;overflow:hidden;font-family:Courier New,monospace;min-width:200px;min-height:40px}.devtools-panel--minimized{border-radius:8px}.devtools-header{display:flex;justify-content:space-between;align-items:center;padding:.6rem .75rem;background:#1e293b;border-bottom:1px solid #334155;cursor:grab;-webkit-user-select:none;user-select:none}.devtools-header:active{cursor:grabbing}.devtools-title{color:#e2e8f0;font-size:.82rem;font-weight:600;font-family:system-ui}.devtools-header-actions{display:flex;gap:.25rem}.btn-icon{background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.8rem;padding:.15rem .35rem;border-radius:4px;line-height:1}.btn-icon:hover{background:#334155;color:#fff}.devtools-resize{position:absolute;bottom:2px;right:4px;color:#334155;font-size:.9rem;cursor:nwse-resize;-webkit-user-select:none;user-select:none;line-height:1}.devtools-resize:hover{color:#64748b}.devtools-tabs{display:flex;background:#1e293b;border-bottom:1px solid #334155}.tab{padding:.4rem .75rem;background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.78rem;font-family:system-ui}.tab:hover{color:#e2e8f0}.tab--active{color:#3b82f6;border-bottom:2px solid #3b82f6}.devtools-content{overflow-y:auto;flex:1;padding:.25rem 0}.devtools-empty{padding:2rem;text-align:center;color:#475569;font-size:.78rem;font-family:system-ui}.log-item{display:flex;justify-content:space-between;align-items:center;padding:.35rem .75rem;cursor:pointer;border-bottom:1px solid #1e293b}.log-item:hover{background:#1e293b}.log-item--error{background:#1a0a0a}.log-item__left{display:flex;align-items:center;gap:.4rem;overflow:hidden}.log-item__right{display:flex;align-items:center;gap:.4rem;flex-shrink:0}.log-status{font-size:.72rem;color:#22c55e;flex-shrink:0}.log-item--error .log-status{color:#ef4444}.log-name{color:#e2e8f0;font-size:.75rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.log-duration{color:#64748b;font-size:.7rem}.log-time{color:#475569;font-size:.68rem}.log-error-badge{background:#7f1d1d;color:#fca5a5;font-size:.68rem;padding:.1rem .35rem;border-radius:4px}.log-detail{background:#0a0f1a;padding:.6rem .75rem;border-left:3px solid #3b82f6;margin:0 .4rem .4rem;border-radius:0 4px 4px 0}.log-detail__error{color:#fca5a5;font-size:.72rem;margin-bottom:.4rem}.log-detail__section{margin-bottom:.4rem}.log-detail__label{color:#64748b;font-size:.68rem;display:block;margin-bottom:.2rem;font-family:system-ui}pre{color:#86efac;font-size:.7rem;margin:0;white-space:pre-wrap;word-break:break-all;max-height:140px;overflow-y:auto}.state-view{color:#86efac;font-size:.7rem;padding:.75rem;margin:0;white-space:pre-wrap;word-break:break-all}\n"] }]
542
+
543
+ <!-- Hidden file input for import -->
544
+ <input
545
+ #fileInput
546
+ type="file"
547
+ accept=".json"
548
+ style="display: none"
549
+ (change)="onFileSelected($event)"
550
+ />
551
+ `, styles: [":host{font-family:system-ui,-apple-system,sans-serif}.devtools-fab{position:fixed;bottom:1.5rem;left:1.5rem;background:#1e293b;color:#fff;border:none;border-radius:999px;padding:.5rem 1rem;font-size:.85rem;font-weight:600;cursor:pointer;z-index:9999;box-shadow:0 4px 12px #0000004d;transition:background .15s}.devtools-fab:hover{background:#334155}.devtools-panel{position:fixed;background:#0f172a;border-radius:12px;box-shadow:0 8px 32px #00000080;z-index:9999;display:flex;flex-direction:column;overflow:hidden;font-family:Courier New,monospace;min-width:200px;min-height:40px;border:1px solid #1e293b}.devtools-panel--minimized{border-radius:8px}.devtools-header{display:flex;justify-content:space-between;align-items:center;padding:.6rem .75rem;background:#1e293b;border-bottom:1px solid #334155;cursor:grab;-webkit-user-select:none;user-select:none}.devtools-header:active{cursor:grabbing}.devtools-title{color:#e2e8f0;font-size:.82rem;font-weight:600;font-family:system-ui;display:flex;align-items:center;gap:.4rem}.tt-badge{background:#7c3aed;color:#fff;font-size:.6rem;padding:.12rem .4rem;border-radius:4px;font-weight:700;letter-spacing:.05em;animation:tt-pulse 1.5s ease-in-out infinite}@keyframes tt-pulse{0%,to{opacity:1}50%{opacity:.6}}.devtools-header-actions{display:flex;gap:.25rem}.btn-icon{background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.8rem;padding:.15rem .35rem;border-radius:4px;line-height:1}.btn-icon:hover{background:#334155;color:#fff}.devtools-resize{position:absolute;bottom:2px;right:4px;color:#334155;font-size:.9rem;cursor:nwse-resize;-webkit-user-select:none;user-select:none;line-height:1}.devtools-resize:hover{color:#64748b}.devtools-tabs{display:flex;background:#1e293b;border-bottom:1px solid #334155}.tab{padding:.4rem .75rem;background:transparent;color:#64748b;border:none;cursor:pointer;font-size:.78rem;font-family:system-ui;transition:color .15s}.tab:hover{color:#e2e8f0}.tab--active{color:#3b82f6;border-bottom:2px solid #3b82f6}.tt-toolbar{display:flex;align-items:center;gap:.2rem;padding:.3rem .5rem;background:#0f172a;border-bottom:1px solid #1e293b}.tt-btn{background:#1e293b;border:1px solid #334155;color:#94a3b8;padding:.2rem .5rem;border-radius:4px;font-size:.72rem;cursor:pointer;transition:all .15s;font-family:system-ui}.tt-btn:hover:not(:disabled){background:#334155;color:#e2e8f0;border-color:#475569}.tt-btn:disabled{opacity:.3;cursor:not-allowed}.tt-btn--resume{background:#7c3aed;border-color:#7c3aed;color:#fff;font-weight:600}.tt-btn--resume:hover:not(:disabled){background:#6d28d9}.tt-btn--export,.tt-btn--import{font-size:.68rem}.tt-spacer{flex:1}.devtools-content{overflow-y:auto;flex:1;padding:.25rem 0}.devtools-empty{padding:2rem;text-align:center;color:#475569;font-size:.78rem;font-family:system-ui}.log-item{display:flex;justify-content:space-between;align-items:center;padding:.35rem .75rem;cursor:pointer;border-bottom:1px solid #1e293b;transition:background .1s}.log-item:hover{background:#1e293b}.log-item--error{background:#1a0a0a}.log-item--active{background:#1e1b4b!important;border-left:3px solid #7c3aed}.log-item--future{opacity:.35}.log-item__left{display:flex;align-items:center;gap:.4rem;overflow:hidden;flex:1}.log-item__right{display:flex;align-items:center;gap:.35rem;flex-shrink:0}.log-status{font-size:.72rem;color:#22c55e;flex-shrink:0}.log-item--error .log-status{color:#ef4444}.log-name{color:#e2e8f0;font-size:.75rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.log-duration{color:#64748b;font-size:.7rem}.log-time{color:#475569;font-size:.68rem}.log-error-badge{background:#7f1d1d;color:#fca5a5;font-size:.68rem;padding:.1rem .35rem;border-radius:4px}.btn-replay{font-size:.65rem;opacity:.5}.btn-replay:hover{opacity:1}.log-detail{background:#0a0f1a;padding:.6rem .75rem;border-left:3px solid #3b82f6;margin:0 .4rem .4rem;border-radius:0 4px 4px 0}.log-detail__error{color:#fca5a5;font-size:.72rem;margin-bottom:.4rem}.log-detail__section{margin-bottom:.4rem}.log-detail__label{color:#64748b;font-size:.68rem;display:block;margin-bottom:.2rem;font-family:system-ui}pre{color:#86efac;font-size:.7rem;margin:0;white-space:pre-wrap;word-break:break-all;max-height:140px;overflow-y:auto}.state-store-block{border-bottom:1px solid #1e293b}.state-store-block:last-child{border-bottom:none}.state-store-name{padding:.4rem .75rem .2rem;color:#3b82f6;font-size:.72rem;font-family:system-ui;font-weight:600;letter-spacing:.03em;text-transform:uppercase}.state-view{color:#86efac;font-size:.7rem;padding:.25rem .75rem .75rem;margin:0;white-space:pre-wrap;word-break:break-all}\n"] }]
342
552
  }] });
343
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"devtools.component.js","sourceRoot":"","sources":["../../src/devtools.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  OnDestroy,\n  signal\n}                        from '@angular/core'\nimport { devTools }      from '@ngstato/core'\nimport type { ActionLog } from '@ngstato/core'\nimport { JsonPipe } from '@angular/common'\n\n@Component({\n  selector:   'ngstato-devtools',\n  standalone: true,\n  imports:    [JsonPipe],\n  template: `\n    <!-- Bouton flottant -->\n    @if (!isOpen()) {\n      <button class=\"devtools-fab\" (click)=\"toggle()\">\n        🛠 Stato\n      </button>\n    }\n\n    <!-- Panel -->\n    @if (isOpen()) {\n      <div\n        class=\"devtools-panel\"\n        [class.devtools-panel--minimized]=\"isMinimized()\"\n        [style.left.px]=\"posX()\"\n        [style.top.px]=\"posY()\"\n        [style.width.px]=\"isMinimized() ? 200 : panelWidth()\"\n        [style.height]=\"isMinimized() ? 'auto' : panelHeight() + 'px'\"\n      >\n\n        <!-- Header — draggable -->\n        <div\n          class=\"devtools-header\"\n          (mousedown)=\"onDragStart($event)\"\n        >\n          <span class=\"devtools-title\">🛠 Stato</span>\n          <div class=\"devtools-header-actions\">\n            @if (!isMinimized()) {\n              <button class=\"btn-icon\" (click)=\"clear()\" title=\"Vider\">🗑</button>\n            }\n            <button class=\"btn-icon\" (click)=\"toggleMinimize()\" title=\"Minimiser/Agrandir\">\n              {{ isMinimized() ? '▲' : '▼' }}\n            </button>\n            <button class=\"btn-icon\" (click)=\"toggle()\" title=\"Fermer\">✕</button>\n          </div>\n        </div>\n\n        <!-- Resize handle — coin bas droite -->\n        @if (!isMinimized()) {\n          <div\n            class=\"devtools-resize\"\n            (mousedown)=\"onResizeStart($event)\"\n          >⊿</div>\n        }\n\n        @if (!isMinimized()) {\n\n          <!-- Tabs -->\n          <div class=\"devtools-tabs\">\n            <button\n              class=\"tab\"\n              [class.tab--active]=\"activeTab() === 'actions'\"\n              (click)=\"activeTab.set('actions')\"\n            >\n              Actions ({{ logs().length }})\n            </button>\n            <button\n              class=\"tab\"\n              [class.tab--active]=\"activeTab() === 'state'\"\n              (click)=\"activeTab.set('state')\"\n            >\n              State\n            </button>\n          </div>\n\n          <!-- Tab Actions -->\n          @if (activeTab() === 'actions') {\n            <div class=\"devtools-content\">\n              @if (!logs().length) {\n                <div class=\"devtools-empty\">Aucune action pour l'instant</div>\n              }\n              @for (log of logs(); track log.id) {\n                <div\n                  class=\"log-item\"\n                  [class.log-item--error]=\"log.status === 'error'\"\n                  (click)=\"selectLog(log)\"\n                >\n                  <div class=\"log-item__left\">\n                    <span class=\"log-status\">{{ log.status === 'success' ? '✓' : '✗' }}</span>\n                    <span class=\"log-name\">{{ log.name }}</span>\n                  </div>\n                  <div class=\"log-item__right\">\n                    @if (log.status === 'error') {\n                      <span class=\"log-error-badge\">erreur</span>\n                    } @else {\n                      <span class=\"log-duration\">{{ log.duration }}ms</span>\n                    }\n                    <span class=\"log-time\">{{ formatTime(log.at) }}</span>\n                  </div>\n                </div>\n\n                @if (selectedLog()?.id === log.id) {\n                  <div class=\"log-detail\">\n                    @if (log.error) {\n                      <div class=\"log-detail__error\">{{ log.error }}</div>\n                    }\n                    <div class=\"log-detail__section\">\n                      <span class=\"log-detail__label\">Avant</span>\n                      <pre>{{ log.prevState | json }}</pre>\n                    </div>\n                    <div class=\"log-detail__section\">\n                      <span class=\"log-detail__label\">Après</span>\n                      <pre>{{ log.nextState | json }}</pre>\n                    </div>\n                  </div>\n                }\n              }\n            </div>\n          }\n\n          <!-- Tab State -->\n          @if (activeTab() === 'state') {\n            <div class=\"devtools-content\">\n              @if (logs().length) {\n                <pre class=\"state-view\">{{ logs()[0].nextState | json }}</pre>\n              } @else {\n                <div class=\"devtools-empty\">Aucun state disponible</div>\n              }\n            </div>\n          }\n        }\n\n      </div>\n    }\n  `,\n  styles: [`\n    .devtools-fab {\n      position:      fixed;\n      bottom:        1.5rem;\n      left:          1.5rem;\n      background:    #1e293b;\n      color:         white;\n      border:        none;\n      border-radius: 999px;\n      padding:       0.5rem 1rem;\n      font-size:     0.85rem;\n      font-weight:   600;\n      cursor:        pointer;\n      z-index:       9999;\n      box-shadow:    0 4px 12px rgba(0,0,0,0.3);\n      transition:    background 0.15s;\n    }\n    .devtools-fab:hover { background: #334155; }\n\n    .devtools-panel {\n      position:       fixed;\n      background:     #0f172a;\n      border-radius:  12px;\n      box-shadow:     0 8px 32px rgba(0,0,0,0.4);\n      z-index:        9999;\n      display:        flex;\n      flex-direction: column;\n      overflow:       hidden;\n      font-family:    'Courier New', monospace;\n      min-width:      200px;\n      min-height:     40px;\n    }\n\n    .devtools-panel--minimized {\n      border-radius: 8px;\n    }\n\n    .devtools-header {\n      display:         flex;\n      justify-content: space-between;\n      align-items:     center;\n      padding:         0.6rem 0.75rem;\n      background:      #1e293b;\n      border-bottom:   1px solid #334155;\n      cursor:          grab;\n      user-select:     none;\n    }\n    .devtools-header:active { cursor: grabbing; }\n\n    .devtools-title {\n      color:       #e2e8f0;\n      font-size:   0.82rem;\n      font-weight: 600;\n      font-family: system-ui;\n    }\n\n    .devtools-header-actions { display: flex; gap: 0.25rem; }\n\n    .btn-icon {\n      background:    transparent;\n      color:         #64748b;\n      border:        none;\n      cursor:        pointer;\n      font-size:     0.8rem;\n      padding:       0.15rem 0.35rem;\n      border-radius: 4px;\n      line-height:   1;\n    }\n    .btn-icon:hover { background: #334155; color: white; }\n\n    .devtools-resize {\n      position:  absolute;\n      bottom:    2px;\n      right:     4px;\n      color:     #334155;\n      font-size: 0.9rem;\n      cursor:    nwse-resize;\n      user-select: none;\n      line-height: 1;\n    }\n    .devtools-resize:hover { color: #64748b; }\n\n    .devtools-tabs {\n      display:       flex;\n      background:    #1e293b;\n      border-bottom: 1px solid #334155;\n    }\n    .tab {\n      padding:     0.4rem 0.75rem;\n      background:  transparent;\n      color:       #64748b;\n      border:      none;\n      cursor:      pointer;\n      font-size:   0.78rem;\n      font-family: system-ui;\n    }\n    .tab:hover    { color: #e2e8f0; }\n    .tab--active  { color: #3b82f6; border-bottom: 2px solid #3b82f6; }\n\n    .devtools-content {\n      overflow-y: auto;\n      flex:       1;\n      padding:    0.25rem 0;\n    }\n\n    .devtools-empty {\n      padding:     2rem;\n      text-align:  center;\n      color:       #475569;\n      font-size:   0.78rem;\n      font-family: system-ui;\n    }\n\n    .log-item {\n      display:         flex;\n      justify-content: space-between;\n      align-items:     center;\n      padding:         0.35rem 0.75rem;\n      cursor:          pointer;\n      border-bottom:   1px solid #1e293b;\n    }\n    .log-item:hover        { background: #1e293b; }\n    .log-item--error       { background: #1a0a0a; }\n\n    .log-item__left  { display: flex; align-items: center; gap: 0.4rem; overflow: hidden; }\n    .log-item__right { display: flex; align-items: center; gap: 0.4rem; flex-shrink: 0; }\n\n    .log-status  { font-size: 0.72rem; color: #22c55e; flex-shrink: 0; }\n    .log-item--error .log-status { color: #ef4444; }\n    .log-name    { color: #e2e8f0; font-size: 0.75rem; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }\n    .log-duration { color: #64748b; font-size: 0.7rem; }\n    .log-time    { color: #475569; font-size: 0.68rem; }\n\n    .log-error-badge {\n      background:    #7f1d1d;\n      color:         #fca5a5;\n      font-size:     0.68rem;\n      padding:       0.1rem 0.35rem;\n      border-radius: 4px;\n    }\n\n    .log-detail {\n      background:    #0a0f1a;\n      padding:       0.6rem 0.75rem;\n      border-left:   3px solid #3b82f6;\n      margin:        0 0.4rem 0.4rem;\n      border-radius: 0 4px 4px 0;\n    }\n    .log-detail__error   { color: #fca5a5; font-size: 0.72rem; margin-bottom: 0.4rem; }\n    .log-detail__section { margin-bottom: 0.4rem; }\n    .log-detail__label   {\n      color:         #64748b;\n      font-size:     0.68rem;\n      display:       block;\n      margin-bottom: 0.2rem;\n      font-family:   system-ui;\n    }\n    pre {\n      color:       #86efac;\n      font-size:   0.7rem;\n      margin:      0;\n      white-space: pre-wrap;\n      word-break:  break-all;\n      max-height:  140px;\n      overflow-y:  auto;\n    }\n    .state-view {\n      color:       #86efac;\n      font-size:   0.7rem;\n      padding:     0.75rem;\n      margin:      0;\n      white-space: pre-wrap;\n      word-break:  break-all;\n    }\n  `]\n})\nexport class StatoDevToolsComponent implements OnInit, OnDestroy {\n\n  private unsub?: () => void\n\n  // State UI\n  isOpen      = signal(false)\n  isMinimized = signal(false)\n  activeTab   = signal<'actions' | 'state'>('actions')\n  logs        = signal<ActionLog[]>([])\n  selectedLog = signal<ActionLog | null>(null)\n\n  // Position et taille\n  posX        = signal(24)\n  posY        = signal(window.innerHeight - 500)\n  panelWidth  = signal(420)\n  panelHeight = signal(460)\n\n  // Drag state\n  private isDragging  = false\n  private isResizing  = false\n  private dragOffsetX = 0\n  private dragOffsetY = 0\n  private startW      = 0\n  private startH      = 0\n  private startX      = 0\n  private startY      = 0\n\n  // Bound listeners\n  private boundMouseMove = this.onMouseMove.bind(this)\n  private boundMouseUp   = this.onMouseUp.bind(this)\n\n  ngOnInit() {\n    this.unsub = devTools.subscribe((state) => {\n      this.logs.set(state.logs)\n      this.isOpen.set(state.isOpen)\n    })\n\n    document.addEventListener('mousemove', this.boundMouseMove)\n    document.addEventListener('mouseup',   this.boundMouseUp)\n  }\n\n  ngOnDestroy() {\n    this.unsub?.()\n    document.removeEventListener('mousemove', this.boundMouseMove)\n    document.removeEventListener('mouseup',   this.boundMouseUp)\n  }\n\n  // ── Toggle ─────────────────────────────────────────\n  toggle()         { devTools.toggle() }\n  toggleMinimize() { this.isMinimized.update(v => !v) }\n  clear()          { devTools.clear(); this.selectedLog.set(null) }\n  selectLog(log: ActionLog) {\n    this.selectedLog.set(this.selectedLog()?.id === log.id ? null : log)\n  }\n  formatTime(iso: string): string {\n    return new Date(iso).toTimeString().slice(0, 8)\n  }\n\n  // ── Drag ───────────────────────────────────────────\n  onDragStart(e: MouseEvent) {\n    if ((e.target as HTMLElement).classList.contains('btn-icon')) return\n    this.isDragging  = true\n    this.dragOffsetX = e.clientX - this.posX()\n    this.dragOffsetY = e.clientY - this.posY()\n    e.preventDefault()\n  }\n\n  // ── Resize ─────────────────────────────────────────\n  onResizeStart(e: MouseEvent) {\n    this.isResizing = true\n    this.startW     = this.panelWidth()\n    this.startH     = this.panelHeight()\n    this.startX     = e.clientX\n    this.startY     = e.clientY\n    e.preventDefault()\n    e.stopPropagation()\n  }\n\n  // ── Mouse Move ─────────────────────────────────────\n  onMouseMove(e: MouseEvent) {\n    if (this.isDragging) {\n      this.posX.set(Math.max(0, e.clientX - this.dragOffsetX))\n      this.posY.set(Math.max(0, e.clientY - this.dragOffsetY))\n    }\n    if (this.isResizing) {\n      const newW = Math.max(300, this.startW + e.clientX - this.startX)\n      const newH = Math.max(200, this.startH + e.clientY - this.startY)\n      this.panelWidth.set(newW)\n      this.panelHeight.set(newH)\n    }\n  }\n\n  // ── Mouse Up ───────────────────────────────────────\n  onMouseUp() {\n    this.isDragging = false\n    this.isResizing = false\n  }\n}"]}
553
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"devtools.component.js","sourceRoot":"","sources":["../../src/devtools.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  OnDestroy,\n  signal,\n  computed\n}                        from '@angular/core'\nimport { devTools }                           from '@ngstato/core'\nimport type { ActionLog, DevToolsState }       from '@ngstato/core'\nimport { JsonPipe, KeyValuePipe } from '@angular/common'\n\n@Component({\n  selector:   'ngstato-devtools, stato-devtools',\n  standalone: true,\n  imports:    [JsonPipe, KeyValuePipe],\n  template: `\n    <!-- Floating button -->\n    @if (!isOpen()) {\n      <button class=\"devtools-fab\" (click)=\"toggle()\">\n        🛠 Stato\n      </button>\n    }\n\n    <!-- Panel -->\n    @if (isOpen()) {\n      <div\n        class=\"devtools-panel\"\n        [class.devtools-panel--minimized]=\"isMinimized()\"\n        [style.left.px]=\"posX()\"\n        [style.top.px]=\"posY()\"\n        [style.width.px]=\"isMinimized() ? 220 : panelWidth()\"\n        [style.height]=\"isMinimized() ? 'auto' : panelHeight() + 'px'\"\n      >\n\n        <!-- Header — draggable -->\n        <div\n          class=\"devtools-header\"\n          (mousedown)=\"onDragStart($event)\"\n        >\n          <span class=\"devtools-title\">\n            🛠 Stato\n            @if (isTimeTraveling()) {\n              <span class=\"tt-badge\">TIME-TRAVEL</span>\n            }\n          </span>\n          <div class=\"devtools-header-actions\">\n            @if (!isMinimized()) {\n              <button class=\"btn-icon\" (click)=\"clear()\" title=\"Clear\">🗑</button>\n            }\n            <button class=\"btn-icon\" (click)=\"toggleMinimize()\" title=\"Minimize\">\n              {{ isMinimized() ? '▲' : '▼' }}\n            </button>\n            <button class=\"btn-icon\" (click)=\"toggle()\" title=\"Close\">✕</button>\n          </div>\n        </div>\n\n        <!-- Resize handle -->\n        @if (!isMinimized()) {\n          <div\n            class=\"devtools-resize\"\n            (mousedown)=\"onResizeStart($event)\"\n          >⊿</div>\n        }\n\n        @if (!isMinimized()) {\n\n          <!-- Tabs -->\n          <div class=\"devtools-tabs\">\n            <button\n              class=\"tab\"\n              [class.tab--active]=\"activeTab() === 'actions'\"\n              (click)=\"activeTab.set('actions')\"\n            >\n              Actions ({{ logs().length }})\n            </button>\n            <button\n              class=\"tab\"\n              [class.tab--active]=\"activeTab() === 'state'\"\n              (click)=\"activeTab.set('state')\"\n            >\n              State\n            </button>\n          </div>\n\n          <!-- Time-travel toolbar -->\n          @if (activeTab() === 'actions' && logs().length) {\n            <div class=\"tt-toolbar\">\n              <button\n                class=\"tt-btn\"\n                (click)=\"onUndo()\"\n                [disabled]=\"!canUndo()\"\n                title=\"Undo (step back)\"\n              >⏪</button>\n              <button\n                class=\"tt-btn\"\n                (click)=\"onRedo()\"\n                [disabled]=\"!canRedo()\"\n                title=\"Redo (step forward)\"\n              >⏩</button>\n              @if (isTimeTraveling()) {\n                <button\n                  class=\"tt-btn tt-btn--resume\"\n                  (click)=\"onResume()\"\n                  title=\"Resume live state\"\n                >▶ Live</button>\n              }\n              <div class=\"tt-spacer\"></div>\n              <button\n                class=\"tt-btn tt-btn--export\"\n                (click)=\"onExport()\"\n                title=\"Export state snapshot (JSON)\"\n              >📤</button>\n              <button\n                class=\"tt-btn tt-btn--import\"\n                (click)=\"onImport()\"\n                title=\"Import state snapshot\"\n              >📥</button>\n            </div>\n          }\n\n          <!-- Tab: Actions -->\n          @if (activeTab() === 'actions') {\n            <div class=\"devtools-content\">\n              @if (!logs().length) {\n                <div class=\"devtools-empty\">No actions yet</div>\n              }\n              @for (log of logs(); track log.id) {\n                <div\n                  class=\"log-item\"\n                  [class.log-item--error]=\"log.status === 'error'\"\n                  [class.log-item--active]=\"activeLogId() === log.id\"\n                  [class.log-item--future]=\"isFutureLog(log)\"\n                  (click)=\"onTravelTo(log)\"\n                >\n                  <div class=\"log-item__left\">\n                    <span class=\"log-status\">{{ log.status === 'success' ? '✓' : '✗' }}</span>\n                    <span class=\"log-name\">{{ log.name }}</span>\n                  </div>\n                  <div class=\"log-item__right\">\n                    @if (log.status === 'error') {\n                      <span class=\"log-error-badge\">error</span>\n                    } @else {\n                      <span class=\"log-duration\">{{ log.duration }}ms</span>\n                    }\n                    <span class=\"log-time\">{{ formatTime(log.at) }}</span>\n                    <button\n                      class=\"btn-icon btn-replay\"\n                      (click)=\"onReplay(log, $event)\"\n                      title=\"Replay this action\"\n                    >🔄</button>\n                  </div>\n                </div>\n\n                @if (selectedLog()?.id === log.id) {\n                  <div class=\"log-detail\">\n                    @if (log.error) {\n                      <div class=\"log-detail__error\">{{ log.error }}</div>\n                    }\n                    <div class=\"log-detail__section\">\n                      <span class=\"log-detail__label\">Before</span>\n                      <pre>{{ log.prevState | json }}</pre>\n                    </div>\n                    <div class=\"log-detail__section\">\n                      <span class=\"log-detail__label\">After</span>\n                      <pre>{{ log.nextState | json }}</pre>\n                    </div>\n                  </div>\n                }\n              }\n            </div>\n          }\n\n          <!-- Tab: State -->\n          @if (activeTab() === 'state') {\n            <div class=\"devtools-content\">\n              @if (globalState().size) {\n                @for (entry of globalState() | keyvalue; track entry.key) {\n                  <div class=\"state-store-block\">\n                    <div class=\"state-store-name\">{{ entry.key }}</div>\n                    <pre class=\"state-view\">{{ entry.value | json }}</pre>\n                  </div>\n                }\n              } @else {\n                <div class=\"devtools-empty\">No state available</div>\n              }\n            </div>\n          }\n        }\n\n      </div>\n    }\n\n    <!-- Hidden file input for import -->\n    <input\n      #fileInput\n      type=\"file\"\n      accept=\".json\"\n      style=\"display: none\"\n      (change)=\"onFileSelected($event)\"\n    />\n  `,\n  styles: [`\n    :host { font-family: system-ui, -apple-system, sans-serif; }\n\n    .devtools-fab {\n      position:      fixed;\n      bottom:        1.5rem;\n      left:          1.5rem;\n      background:    #1e293b;\n      color:         white;\n      border:        none;\n      border-radius: 999px;\n      padding:       0.5rem 1rem;\n      font-size:     0.85rem;\n      font-weight:   600;\n      cursor:        pointer;\n      z-index:       9999;\n      box-shadow:    0 4px 12px rgba(0,0,0,0.3);\n      transition:    background 0.15s;\n    }\n    .devtools-fab:hover { background: #334155; }\n\n    .devtools-panel {\n      position:       fixed;\n      background:     #0f172a;\n      border-radius:  12px;\n      box-shadow:     0 8px 32px rgba(0,0,0,0.5);\n      z-index:        9999;\n      display:        flex;\n      flex-direction: column;\n      overflow:       hidden;\n      font-family:    'Courier New', monospace;\n      min-width:      200px;\n      min-height:     40px;\n      border:         1px solid #1e293b;\n    }\n\n    .devtools-panel--minimized {\n      border-radius: 8px;\n    }\n\n    .devtools-header {\n      display:         flex;\n      justify-content: space-between;\n      align-items:     center;\n      padding:         0.6rem 0.75rem;\n      background:      #1e293b;\n      border-bottom:   1px solid #334155;\n      cursor:          grab;\n      user-select:     none;\n    }\n    .devtools-header:active { cursor: grabbing; }\n\n    .devtools-title {\n      color:       #e2e8f0;\n      font-size:   0.82rem;\n      font-weight: 600;\n      font-family: system-ui;\n      display:     flex;\n      align-items: center;\n      gap:         0.4rem;\n    }\n\n    .tt-badge {\n      background:    #7c3aed;\n      color:         #fff;\n      font-size:     0.6rem;\n      padding:       0.12rem 0.4rem;\n      border-radius: 4px;\n      font-weight:   700;\n      letter-spacing: 0.05em;\n      animation:     tt-pulse 1.5s ease-in-out infinite;\n    }\n    @keyframes tt-pulse {\n      0%, 100% { opacity: 1; }\n      50%      { opacity: 0.6; }\n    }\n\n    .devtools-header-actions { display: flex; gap: 0.25rem; }\n\n    .btn-icon {\n      background:    transparent;\n      color:         #64748b;\n      border:        none;\n      cursor:        pointer;\n      font-size:     0.8rem;\n      padding:       0.15rem 0.35rem;\n      border-radius: 4px;\n      line-height:   1;\n    }\n    .btn-icon:hover { background: #334155; color: white; }\n\n    .devtools-resize {\n      position:  absolute;\n      bottom:    2px;\n      right:     4px;\n      color:     #334155;\n      font-size: 0.9rem;\n      cursor:    nwse-resize;\n      user-select: none;\n      line-height: 1;\n    }\n    .devtools-resize:hover { color: #64748b; }\n\n    /* ── Tabs ─────────────────────────────────────────── */\n    .devtools-tabs {\n      display:       flex;\n      background:    #1e293b;\n      border-bottom: 1px solid #334155;\n    }\n    .tab {\n      padding:     0.4rem 0.75rem;\n      background:  transparent;\n      color:       #64748b;\n      border:      none;\n      cursor:      pointer;\n      font-size:   0.78rem;\n      font-family: system-ui;\n      transition:  color 0.15s;\n    }\n    .tab:hover    { color: #e2e8f0; }\n    .tab--active  { color: #3b82f6; border-bottom: 2px solid #3b82f6; }\n\n    /* ── Time-travel toolbar ──────────────────────────── */\n    .tt-toolbar {\n      display:       flex;\n      align-items:   center;\n      gap:           0.2rem;\n      padding:       0.3rem 0.5rem;\n      background:    #0f172a;\n      border-bottom: 1px solid #1e293b;\n    }\n\n    .tt-btn {\n      background:    #1e293b;\n      border:        1px solid #334155;\n      color:         #94a3b8;\n      padding:       0.2rem 0.5rem;\n      border-radius: 4px;\n      font-size:     0.72rem;\n      cursor:        pointer;\n      transition:    all 0.15s;\n      font-family:   system-ui;\n    }\n    .tt-btn:hover:not(:disabled) {\n      background: #334155;\n      color:      #e2e8f0;\n      border-color: #475569;\n    }\n    .tt-btn:disabled {\n      opacity: 0.3;\n      cursor:  not-allowed;\n    }\n    .tt-btn--resume {\n      background: #7c3aed;\n      border-color: #7c3aed;\n      color: white;\n      font-weight: 600;\n    }\n    .tt-btn--resume:hover:not(:disabled) {\n      background: #6d28d9;\n    }\n    .tt-btn--export, .tt-btn--import {\n      font-size: 0.68rem;\n    }\n    .tt-spacer { flex: 1; }\n\n    /* ── Content ──────────────────────────────────────── */\n    .devtools-content {\n      overflow-y: auto;\n      flex:       1;\n      padding:    0.25rem 0;\n    }\n\n    .devtools-empty {\n      padding:     2rem;\n      text-align:  center;\n      color:       #475569;\n      font-size:   0.78rem;\n      font-family: system-ui;\n    }\n\n    /* ── Log items ────────────────────────────────────── */\n    .log-item {\n      display:         flex;\n      justify-content: space-between;\n      align-items:     center;\n      padding:         0.35rem 0.75rem;\n      cursor:          pointer;\n      border-bottom:   1px solid #1e293b;\n      transition:      background 0.1s;\n    }\n    .log-item:hover     { background: #1e293b; }\n    .log-item--error    { background: #1a0a0a; }\n    .log-item--active   {\n      background:  #1e1b4b !important;\n      border-left: 3px solid #7c3aed;\n    }\n    .log-item--future {\n      opacity: 0.35;\n    }\n\n    .log-item__left  { display: flex; align-items: center; gap: 0.4rem; overflow: hidden; flex: 1; }\n    .log-item__right { display: flex; align-items: center; gap: 0.35rem; flex-shrink: 0; }\n\n    .log-status  { font-size: 0.72rem; color: #22c55e; flex-shrink: 0; }\n    .log-item--error .log-status { color: #ef4444; }\n    .log-name    { color: #e2e8f0; font-size: 0.75rem; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }\n    .log-duration { color: #64748b; font-size: 0.7rem; }\n    .log-time    { color: #475569; font-size: 0.68rem; }\n\n    .log-error-badge {\n      background:    #7f1d1d;\n      color:         #fca5a5;\n      font-size:     0.68rem;\n      padding:       0.1rem 0.35rem;\n      border-radius: 4px;\n    }\n\n    .btn-replay {\n      font-size: 0.65rem;\n      opacity:   0.5;\n    }\n    .btn-replay:hover { opacity: 1; }\n\n    /* ── Log detail ───────────────────────────────────── */\n    .log-detail {\n      background:    #0a0f1a;\n      padding:       0.6rem 0.75rem;\n      border-left:   3px solid #3b82f6;\n      margin:        0 0.4rem 0.4rem;\n      border-radius: 0 4px 4px 0;\n    }\n    .log-detail__error   { color: #fca5a5; font-size: 0.72rem; margin-bottom: 0.4rem; }\n    .log-detail__section { margin-bottom: 0.4rem; }\n    .log-detail__label   {\n      color:         #64748b;\n      font-size:     0.68rem;\n      display:       block;\n      margin-bottom: 0.2rem;\n      font-family:   system-ui;\n    }\n    pre {\n      color:       #86efac;\n      font-size:   0.7rem;\n      margin:      0;\n      white-space: pre-wrap;\n      word-break:  break-all;\n      max-height:  140px;\n      overflow-y:  auto;\n    }\n\n    /* ── State tab ────────────────────────────────────── */\n    .state-store-block {\n      border-bottom: 1px solid #1e293b;\n    }\n    .state-store-block:last-child { border-bottom: none; }\n    .state-store-name {\n      padding:       0.4rem 0.75rem 0.2rem;\n      color:         #3b82f6;\n      font-size:     0.72rem;\n      font-family:   system-ui;\n      font-weight:   600;\n      letter-spacing: 0.03em;\n      text-transform: uppercase;\n    }\n    .state-view {\n      color:       #86efac;\n      font-size:   0.7rem;\n      padding:     0.25rem 0.75rem 0.75rem;\n      margin:      0;\n      white-space: pre-wrap;\n      word-break:  break-all;\n    }\n  `]\n})\nexport class StatoDevToolsComponent implements OnInit, OnDestroy {\n\n  private unsub?: () => void\n\n  // State UI\n  isOpen          = signal(false)\n  isMinimized     = signal(false)\n  activeTab       = signal<'actions' | 'state'>('actions')\n  logs            = signal<ActionLog[]>([])\n  selectedLog     = signal<ActionLog | null>(null)\n  activeLogId     = signal<number | null>(null)\n  isTimeTraveling = signal(false)\n\n  // Snapshot global — latest known state per store\n  globalState = computed(() => {\n    const seen = new Map<string, unknown>()\n    for (const log of this.logs()) {\n      if (!seen.has(log.storeName)) {\n        seen.set(log.storeName, log.nextState)\n      }\n    }\n    return seen\n  })\n\n  // Time-travel computed\n  canUndo = computed(() => this.logs().length > 0)\n  canRedo = computed(() => {\n    if (!this.isTimeTraveling()) return false\n    const id = this.activeLogId()\n    if (id === null) return false\n    if (id === -1) return this.logs().length > 0\n    const idx = this.logs().findIndex(l => l.id === id)\n    return idx > 0  // can go forward (lower index = newer)\n  })\n\n  // Position & size\n  posX        = signal(24)\n  posY        = signal(window.innerHeight - 520)\n  panelWidth  = signal(440)\n  panelHeight = signal(480)\n\n  // Drag state\n  private isDragging  = false\n  private isResizing  = false\n  private dragOffsetX = 0\n  private dragOffsetY = 0\n  private startW      = 0\n  private startH      = 0\n  private startX      = 0\n  private startY      = 0\n\n  // Bound listeners\n  private boundMouseMove = this.onMouseMove.bind(this)\n  private boundMouseUp   = this.onMouseUp.bind(this)\n\n  ngOnInit() {\n    this.unsub = devTools.subscribe((state: DevToolsState) => {\n      this.logs.set(state.logs)\n      this.isOpen.set(state.isOpen)\n      this.activeLogId.set(state.activeLogId)\n      this.isTimeTraveling.set(state.isTimeTraveling)\n    })\n\n    document.addEventListener('mousemove', this.boundMouseMove)\n    document.addEventListener('mouseup',   this.boundMouseUp)\n  }\n\n  ngOnDestroy() {\n    this.unsub?.()\n    document.removeEventListener('mousemove', this.boundMouseMove)\n    document.removeEventListener('mouseup',   this.boundMouseUp)\n  }\n\n  // ── Toggle ─────────────────────────────────────────\n  toggle()         { devTools.toggle() }\n  toggleMinimize() { this.isMinimized.update(v => !v) }\n  clear()          { devTools.clear(); this.selectedLog.set(null) }\n\n  selectLog(log: ActionLog) {\n    this.selectedLog.set(this.selectedLog()?.id === log.id ? null : log)\n  }\n\n  formatTime(iso: string): string {\n    return new Date(iso).toTimeString().slice(0, 8)\n  }\n\n  // ── Time-travel actions ────────────────────────────\n  onTravelTo(log: ActionLog) {\n    // Toggle detail view\n    this.selectLog(log)\n    // Jump to this action's state\n    devTools.travelTo(log.id)\n  }\n\n  onUndo()   { devTools.undo() }\n  onRedo()   { devTools.redo() }\n  onResume() { devTools.resume() }\n\n  onReplay(log: ActionLog, event: Event) {\n    event.stopPropagation()\n    devTools.replay(log.id)\n  }\n\n  isFutureLog(log: ActionLog): boolean {\n    if (!this.isTimeTraveling()) return false\n    const activeId = this.activeLogId()\n    if (activeId === null) return false\n    if (activeId === -1) return true  // all logs are \"future\"\n    const activeIdx = this.logs().findIndex(l => l.id === activeId)\n    const logIdx = this.logs().findIndex(l => l.id === log.id)\n    return logIdx < activeIdx  // newer logs (lower index) are \"future\"\n  }\n\n  // ── Export/Import ──────────────────────────────────\n  onExport() {\n    const snapshot = devTools.exportSnapshot()\n    const json = JSON.stringify(snapshot, null, 2)\n    const blob = new Blob([json], { type: 'application/json' })\n    const url = URL.createObjectURL(blob)\n    const a = document.createElement('a')\n    a.href = url\n    a.download = `ngstato-snapshot-${new Date().toISOString().slice(0, 19).replace(/:/g, '-')}.json`\n    a.click()\n    URL.revokeObjectURL(url)\n  }\n\n  onImport() {\n    const input = document.querySelector('input[type=\"file\"]') as HTMLInputElement\n    input?.click()\n  }\n\n  onFileSelected(event: Event) {\n    const file = (event.target as HTMLInputElement).files?.[0]\n    if (!file) return\n\n    const reader = new FileReader()\n    reader.onload = () => {\n      try {\n        const snapshot = JSON.parse(reader.result as string)\n        devTools.importSnapshot(snapshot)\n      } catch (e) {\n        console.error('[ngStato DevTools] Invalid snapshot file:', e)\n      }\n    }\n    reader.readAsText(file)\n    // Reset input\n    ;(event.target as HTMLInputElement).value = ''\n  }\n\n  // ── Drag ───────────────────────────────────────────\n  onDragStart(e: MouseEvent) {\n    if ((e.target as HTMLElement).classList.contains('btn-icon')) return\n    this.isDragging  = true\n    this.dragOffsetX = e.clientX - this.posX()\n    this.dragOffsetY = e.clientY - this.posY()\n    e.preventDefault()\n  }\n\n  // ── Resize ─────────────────────────────────────────\n  onResizeStart(e: MouseEvent) {\n    this.isResizing = true\n    this.startW     = this.panelWidth()\n    this.startH     = this.panelHeight()\n    this.startX     = e.clientX\n    this.startY     = e.clientY\n    e.preventDefault()\n    e.stopPropagation()\n  }\n\n  // ── Mouse Move ─────────────────────────────────────\n  onMouseMove(e: MouseEvent) {\n    if (this.isDragging) {\n      this.posX.set(Math.max(0, e.clientX - this.dragOffsetX))\n      this.posY.set(Math.max(0, e.clientY - this.dragOffsetY))\n    }\n    if (this.isResizing) {\n      const newW = Math.max(300, this.startW + e.clientX - this.startX)\n      const newH = Math.max(200, this.startH + e.clientY - this.startY)\n      this.panelWidth.set(newW)\n      this.panelHeight.set(newH)\n    }\n  }\n\n  // ── Mouse Up ───────────────────────────────────────\n  onMouseUp() {\n    this.isDragging = false\n    this.isResizing = false\n  }\n}"]}