@dxos/react-ui-canvas 0.8.3 → 0.8.4-main.1068cf700f

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/dist/lib/browser/index.mjs +388 -390
  2. package/dist/lib/browser/index.mjs.map +4 -4
  3. package/dist/lib/browser/meta.json +1 -1
  4. package/dist/lib/node-esm/index.mjs +388 -390
  5. package/dist/lib/node-esm/index.mjs.map +4 -4
  6. package/dist/lib/node-esm/meta.json +1 -1
  7. package/dist/types/src/components/Canvas/Canvas.d.ts +1 -1
  8. package/dist/types/src/components/Canvas/Canvas.d.ts.map +1 -1
  9. package/dist/types/src/components/Canvas/Canvas.stories.d.ts +12 -4
  10. package/dist/types/src/components/Canvas/Canvas.stories.d.ts.map +1 -1
  11. package/dist/types/src/components/Grid/Grid.d.ts +2 -2
  12. package/dist/types/src/components/Grid/Grid.d.ts.map +1 -1
  13. package/dist/types/src/components/Grid/Grid.stories.d.ts +19 -4
  14. package/dist/types/src/components/Grid/Grid.stories.d.ts.map +1 -1
  15. package/dist/types/src/hooks/index.d.ts +1 -0
  16. package/dist/types/src/hooks/index.d.ts.map +1 -1
  17. package/dist/types/src/hooks/projection.d.ts +1 -1
  18. package/dist/types/src/hooks/projection.d.ts.map +1 -1
  19. package/dist/types/src/hooks/useDrag.d.ts +6 -0
  20. package/dist/types/src/hooks/useDrag.d.ts.map +1 -0
  21. package/dist/types/src/hooks/useWheel.d.ts.map +1 -1
  22. package/dist/types/src/types.d.ts +1 -1
  23. package/dist/types/src/types.d.ts.map +1 -1
  24. package/dist/types/src/util/svg.stories.d.ts +12 -4
  25. package/dist/types/src/util/svg.stories.d.ts.map +1 -1
  26. package/dist/types/tsconfig.tsbuildinfo +1 -1
  27. package/package.json +25 -22
  28. package/src/components/Canvas/Canvas.stories.tsx +17 -14
  29. package/src/components/Canvas/Canvas.tsx +11 -15
  30. package/src/components/FPS.tsx +3 -3
  31. package/src/components/Grid/Grid.stories.tsx +12 -10
  32. package/src/components/Grid/Grid.tsx +15 -13
  33. package/src/hooks/index.ts +1 -0
  34. package/src/hooks/projection.tsx +2 -2
  35. package/src/hooks/useDrag.tsx +96 -0
  36. package/src/hooks/useWheel.tsx +2 -28
  37. package/src/types.ts +1 -1
  38. package/src/util/svg.stories.tsx +9 -9
  39. package/src/util/svg.tsx +1 -1
  40. package/dist/lib/node/index.cjs +0 -691
  41. package/dist/lib/node/index.cjs.map +0 -7
  42. package/dist/lib/node/meta.json +0 -1
@@ -1,28 +1,27 @@
1
1
  import { createRequire } from 'node:module';const require = createRequire(import.meta.url);
2
2
 
3
- // packages/ui/react-ui-canvas/src/components/Canvas/Canvas.tsx
4
- import { useSignals as _useSignals2 } from "@preact-signals/safe-react/tracking";
5
- import React2, { forwardRef, useEffect as useEffect2, useImperativeHandle, useMemo, useState } from "react";
3
+ // src/components/Canvas/Canvas.tsx
4
+ import React2, { forwardRef, useEffect as useEffect3, useImperativeHandle, useMemo, useState } from "react";
6
5
  import { useResizeDetector } from "react-resize-detector";
7
- import { mx as mx2 } from "@dxos/react-ui-theme";
6
+ import { mx as mx2 } from "@dxos/ui-theme";
8
7
 
9
- // packages/ui/react-ui-canvas/src/hooks/projection.tsx
10
- import { interpolate, interpolateObject, transition, easeSinOut } from "d3";
8
+ // src/hooks/projection.tsx
9
+ import { easeSinOut, interpolate, interpolateObject, transition } from "d3";
11
10
  import { applyToPoints, compose, identity, inverse, scale as scaleMatrix, translate as translateMatrix } from "transformation-matrix";
12
11
  var defaultOrigin = {
13
12
  x: 0,
14
13
  y: 0
15
14
  };
16
15
  var ProjectionMapper = class {
16
+ _bounds = {
17
+ width: 0,
18
+ height: 0
19
+ };
20
+ _scale = 1;
21
+ _offset = defaultOrigin;
22
+ _toScreen = identity();
23
+ _toModel = identity();
17
24
  constructor(bounds, scale, offset) {
18
- this._bounds = {
19
- width: 0,
20
- height: 0
21
- };
22
- this._scale = 1;
23
- this._offset = defaultOrigin;
24
- this._toScreen = identity();
25
- this._toModel = identity();
26
25
  if (bounds && scale && offset) {
27
26
  this.update(bounds, scale, offset);
28
27
  }
@@ -98,7 +97,7 @@ var zoomTo = (setTransform, current, next, delay = 200, cb = noop) => {
98
97
  }).on("end", cb);
99
98
  };
100
99
 
101
- // packages/ui/react-ui-canvas/src/hooks/useCanvasContext.tsx
100
+ // src/hooks/useCanvasContext.tsx
102
101
  import { createContext, useContext } from "react";
103
102
  import { raise } from "@dxos/debug";
104
103
  var CanvasContext = /* @__PURE__ */ createContext(null);
@@ -106,14 +105,81 @@ var useCanvasContext = () => {
106
105
  return useContext(CanvasContext) ?? raise(new Error("Missing CanvasContext"));
107
106
  };
108
107
 
109
- // packages/ui/react-ui-canvas/src/hooks/useWheel.tsx
108
+ // src/hooks/useDrag.tsx
109
+ import { bind } from "bind-event-listener";
110
+ import { useEffect, useRef } from "react";
111
+ var useDrag = (_options = {}) => {
112
+ const { root, setProjection } = useCanvasContext();
113
+ const state = useRef({
114
+ panning: false,
115
+ x: 0,
116
+ y: 0
117
+ });
118
+ useEffect(() => {
119
+ if (!root) {
120
+ return;
121
+ }
122
+ return bind(root, {
123
+ type: "pointerdown",
124
+ listener: (ev) => {
125
+ if (ev.button !== 0) {
126
+ return;
127
+ }
128
+ if (ev.defaultPrevented) {
129
+ return;
130
+ }
131
+ if (ev.target !== root || ev.shiftKey) {
132
+ return;
133
+ }
134
+ ev.preventDefault();
135
+ root.setPointerCapture(ev.pointerId);
136
+ state.current = {
137
+ panning: true,
138
+ x: ev.clientX,
139
+ y: ev.clientY
140
+ };
141
+ const moveUnbind = bind(root, {
142
+ type: "pointermove",
143
+ listener: (ev2) => {
144
+ if (!state.current.panning) {
145
+ return;
146
+ }
147
+ const dx = ev2.clientX - state.current.x;
148
+ const dy = ev2.clientY - state.current.y;
149
+ state.current.x = ev2.clientX;
150
+ state.current.y = ev2.clientY;
151
+ setProjection((prev) => ({
152
+ ...prev,
153
+ offset: {
154
+ x: prev.offset.x + dx,
155
+ y: prev.offset.y + dy
156
+ }
157
+ }));
158
+ }
159
+ });
160
+ const upUnbind = bind(root, {
161
+ type: "pointerup",
162
+ listener: (ev2) => {
163
+ state.current.panning = false;
164
+ root.releasePointerCapture(ev2.pointerId);
165
+ moveUnbind();
166
+ upUnbind();
167
+ }
168
+ });
169
+ }
170
+ });
171
+ }, [
172
+ root
173
+ ]);
174
+ };
175
+
176
+ // src/hooks/useWheel.tsx
110
177
  import { bindAll } from "bind-event-listener";
111
- import { useEffect } from "react";
178
+ import { useEffect as useEffect2 } from "react";
112
179
 
113
- // packages/ui/react-ui-canvas/src/util/svg.tsx
114
- import { useSignals as _useSignals } from "@preact-signals/safe-react/tracking";
180
+ // src/util/svg.tsx
115
181
  import React from "react";
116
- import { mx } from "@dxos/react-ui-theme";
182
+ import { mx } from "@dxos/ui-theme";
117
183
  var createPath = (points, join = false) => {
118
184
  return [
119
185
  "M",
@@ -122,146 +188,120 @@ var createPath = (points, join = false) => {
122
188
  ].join(" ");
123
189
  };
124
190
  var Markers = ({ id = "dx-marker", classNames }) => {
125
- var _effect = _useSignals();
126
- try {
127
- return /* @__PURE__ */ React.createElement(React.Fragment, null, /* @__PURE__ */ React.createElement(Arrow, {
128
- id: `${id}-arrow-start`,
129
- dir: "start",
130
- classNames
131
- }), /* @__PURE__ */ React.createElement(Arrow, {
132
- id: `${id}-arrow-end`,
133
- dir: "end",
134
- classNames
135
- }), /* @__PURE__ */ React.createElement(Arrow, {
136
- id: `${id}-triangle-start`,
137
- dir: "start",
138
- closed: true,
139
- classNames
140
- }), /* @__PURE__ */ React.createElement(Arrow, {
141
- id: `${id}-triangle-end`,
142
- dir: "end",
143
- closed: true,
144
- classNames
145
- }), /* @__PURE__ */ React.createElement(Marker, {
146
- id: `${id}-circle`,
147
- pos: {
148
- x: 8,
149
- y: 8
150
- },
151
- size: {
152
- width: 16,
153
- height: 16
154
- }
155
- }, /* @__PURE__ */ React.createElement("circle", {
156
- cx: 8,
157
- cy: 8,
158
- r: 5,
159
- stroke: "context-stroke",
160
- className: mx(classNames)
161
- })));
162
- } finally {
163
- _effect.f();
164
- }
165
- };
166
- var Marker = ({ id, className, children, pos: { x: refX, y: refY }, size: { width: markerWidth, height: markerHeight }, fill, ...rest }) => {
167
- var _effect = _useSignals();
168
- try {
169
- return /* @__PURE__ */ React.createElement("marker", {
170
- id,
171
- className,
172
- refX,
173
- refY,
174
- markerWidth,
175
- markerHeight,
176
- markerUnits: "strokeWidth",
177
- orient: "auto",
178
- ...rest
179
- }, children);
180
- } finally {
181
- _effect.f();
182
- }
183
- };
184
- var Arrow = ({ classNames, id, size = 16, dir = "end", closed = false }) => {
185
- var _effect = _useSignals();
186
- try {
187
- return /* @__PURE__ */ React.createElement(Marker, {
188
- id,
189
- size: {
190
- width: size,
191
- height: size
192
- },
193
- pos: dir === "end" ? {
194
- x: size,
195
- y: size / 2
196
- } : {
197
- x: 0,
198
- y: size / 2
199
- }
200
- }, /* @__PURE__ */ React.createElement("path", {
201
- fill: closed ? void 0 : "none",
202
- stroke: "context-stroke",
203
- className: mx(classNames),
204
- d: createPath(dir === "end" ? [
205
- {
206
- x: 1,
207
- y: 1
208
- },
209
- {
210
- x: size,
211
- y: size / 2
212
- },
213
- {
214
- x: 1,
215
- y: size - 1
216
- }
217
- ] : [
218
- {
219
- x: size - 1,
220
- y: 1
221
- },
222
- {
223
- x: 0,
224
- y: size / 2
225
- },
226
- {
227
- x: size - 1,
228
- y: size - 1
229
- }
230
- ], closed)
231
- }));
232
- } finally {
233
- _effect.f();
234
- }
191
+ return /* @__PURE__ */ React.createElement(React.Fragment, null, /* @__PURE__ */ React.createElement(Arrow, {
192
+ id: `${id}-arrow-start`,
193
+ dir: "start",
194
+ classNames
195
+ }), /* @__PURE__ */ React.createElement(Arrow, {
196
+ id: `${id}-arrow-end`,
197
+ dir: "end",
198
+ classNames
199
+ }), /* @__PURE__ */ React.createElement(Arrow, {
200
+ id: `${id}-triangle-start`,
201
+ dir: "start",
202
+ closed: true,
203
+ classNames
204
+ }), /* @__PURE__ */ React.createElement(Arrow, {
205
+ id: `${id}-triangle-end`,
206
+ dir: "end",
207
+ closed: true,
208
+ classNames
209
+ }), /* @__PURE__ */ React.createElement(Marker, {
210
+ id: `${id}-circle`,
211
+ pos: {
212
+ x: 8,
213
+ y: 8
214
+ },
215
+ size: {
216
+ width: 16,
217
+ height: 16
218
+ }
219
+ }, /* @__PURE__ */ React.createElement("circle", {
220
+ cx: 8,
221
+ cy: 8,
222
+ r: 5,
223
+ stroke: "context-stroke",
224
+ className: mx(classNames)
225
+ })));
235
226
  };
236
- var GridPattern = ({ classNames, id, size, offset }) => {
237
- var _effect = _useSignals();
238
- try {
239
- return /* @__PURE__ */ React.createElement("pattern", {
240
- id,
241
- x: (size / 2 + offset.x) % size,
242
- y: (size / 2 + offset.y) % size,
243
- width: size,
244
- height: size,
245
- patternUnits: "userSpaceOnUse"
246
- }, /* @__PURE__ */ React.createElement("g", {
247
- className: mx(classNames)
248
- }, /* @__PURE__ */ React.createElement("line", {
249
- x1: 0,
250
- y1: size / 2,
251
- x2: size,
252
- y2: size / 2
253
- }), /* @__PURE__ */ React.createElement("line", {
254
- x1: size / 2,
255
- y1: 0,
256
- x2: size / 2,
257
- y2: size
258
- })));
259
- } finally {
260
- _effect.f();
227
+ var Marker = ({ id, className, children, pos: { x: refX, y: refY }, size: { width: markerWidth, height: markerHeight }, fill, ...rest }) => /* @__PURE__ */ React.createElement("marker", {
228
+ id,
229
+ className,
230
+ refX,
231
+ refY,
232
+ markerWidth,
233
+ markerHeight,
234
+ markerUnits: "strokeWidth",
235
+ orient: "auto",
236
+ ...rest
237
+ }, children);
238
+ var Arrow = ({ classNames, id, size = 16, dir = "end", closed = false }) => /* @__PURE__ */ React.createElement(Marker, {
239
+ id,
240
+ size: {
241
+ width: size,
242
+ height: size
243
+ },
244
+ pos: dir === "end" ? {
245
+ x: size,
246
+ y: size / 2
247
+ } : {
248
+ x: 0,
249
+ y: size / 2
261
250
  }
262
- };
251
+ }, /* @__PURE__ */ React.createElement("path", {
252
+ fill: closed ? void 0 : "none",
253
+ stroke: "context-stroke",
254
+ className: mx(classNames),
255
+ d: createPath(dir === "end" ? [
256
+ {
257
+ x: 1,
258
+ y: 1
259
+ },
260
+ {
261
+ x: size,
262
+ y: size / 2
263
+ },
264
+ {
265
+ x: 1,
266
+ y: size - 1
267
+ }
268
+ ] : [
269
+ {
270
+ x: size - 1,
271
+ y: 1
272
+ },
273
+ {
274
+ x: 0,
275
+ y: size / 2
276
+ },
277
+ {
278
+ x: size - 1,
279
+ y: size - 1
280
+ }
281
+ ], closed)
282
+ }));
283
+ var GridPattern = ({ classNames, id, size, offset }) => /* @__PURE__ */ React.createElement("pattern", {
284
+ id,
285
+ x: (size / 2 + offset.x) % size,
286
+ y: (size / 2 + offset.y) % size,
287
+ width: size,
288
+ height: size,
289
+ patternUnits: "userSpaceOnUse"
290
+ }, /* @__PURE__ */ React.createElement("g", {
291
+ className: mx(classNames)
292
+ }, /* @__PURE__ */ React.createElement("line", {
293
+ x1: 0,
294
+ y1: size / 2,
295
+ x2: size,
296
+ y2: size / 2
297
+ }), /* @__PURE__ */ React.createElement("line", {
298
+ x1: size / 2,
299
+ y1: 0,
300
+ x2: size / 2,
301
+ y2: size
302
+ })));
263
303
 
264
- // packages/ui/react-ui-canvas/src/util/util.ts
304
+ // src/util/util.ts
265
305
  var logged = false;
266
306
  var getRelativePoint = (el, ev) => {
267
307
  const rect = el.getBoundingClientRect();
@@ -296,13 +336,13 @@ var inspectElement = (el) => {
296
336
  };
297
337
  var DATA_TEST_ID = "data-test-id";
298
338
 
299
- // packages/ui/react-ui-canvas/src/hooks/useWheel.tsx
339
+ // src/hooks/useWheel.tsx
300
340
  var defaultOptions = {
301
341
  zoom: true
302
342
  };
303
343
  var useWheel = (options = defaultOptions) => {
304
344
  const { root, setProjection } = useCanvasContext();
305
- useEffect(() => {
345
+ useEffect2(() => {
306
346
  if (!root) {
307
347
  return;
308
348
  }
@@ -315,9 +355,6 @@ var useWheel = (options = defaultOptions) => {
315
355
  },
316
356
  listener: (ev) => {
317
357
  const zooming = isWheelZooming(ev);
318
- if (!hasFocus(root) && !zooming) {
319
- return;
320
- }
321
358
  ev.preventDefault();
322
359
  if (zooming && !options.zoom) {
323
360
  return;
@@ -361,191 +398,161 @@ var isWheelZooming = (ev) => {
361
398
  }
362
399
  return false;
363
400
  };
364
- var hasFocus = (element) => {
365
- const activeElement = document.activeElement;
366
- if (!activeElement) {
367
- return false;
368
- }
369
- let shadowActive = activeElement;
370
- while (shadowActive?.shadowRoot?.activeElement) {
371
- shadowActive = shadowActive.shadowRoot.activeElement;
372
- }
373
- let current = element;
374
- while (current) {
375
- if (current === activeElement || current === shadowActive) {
376
- return true;
377
- }
378
- current = current.parentElement;
379
- }
380
- return false;
381
- };
382
401
 
383
- // packages/ui/react-ui-canvas/src/components/Canvas/Canvas.tsx
384
- var Canvas = /* @__PURE__ */ forwardRef(({ children, classNames, scale: _scale = 1, offset: _offset = defaultOrigin, ...props }, forwardedRef) => {
385
- var _effect = _useSignals2();
386
- try {
387
- const { ref, width = 0, height = 0 } = useResizeDetector();
388
- const [ready, setReady] = useState(false);
389
- const [{ scale, offset }, setProjection] = useState({
390
- scale: _scale,
391
- offset: _offset
392
- });
393
- useEffect2(() => {
394
- if (width && height && offset === defaultOrigin) {
395
- setProjection({
396
- scale,
397
- offset: {
398
- x: width / 2,
399
- y: height / 2
400
- }
401
- });
402
+ // src/components/Canvas/Canvas.tsx
403
+ var Canvas = /* @__PURE__ */ forwardRef(({ children, classNames, scale: _scale = 1, offset: offsetProp = defaultOrigin, ...props }, forwardedRef) => {
404
+ const { ref, width = 0, height = 0 } = useResizeDetector();
405
+ const [ready, setReady] = useState(false);
406
+ const [{ scale, offset }, setProjection] = useState({
407
+ scale: _scale,
408
+ offset: offsetProp
409
+ });
410
+ useEffect3(() => {
411
+ if (width && height && offset === defaultOrigin) {
412
+ setProjection({
413
+ scale,
414
+ offset: {
415
+ x: width / 2,
416
+ y: height / 2
417
+ }
418
+ });
419
+ }
420
+ }, [
421
+ width,
422
+ height,
423
+ scale,
424
+ offset
425
+ ]);
426
+ const projection = useMemo(() => new ProjectionMapper(), []);
427
+ useEffect3(() => {
428
+ projection.update({
429
+ width,
430
+ height
431
+ }, scale, offset);
432
+ if (offset !== defaultOrigin) {
433
+ setReady(true);
434
+ }
435
+ }, [
436
+ projection,
437
+ scale,
438
+ offset,
439
+ width,
440
+ height
441
+ ]);
442
+ const styles = useMemo(() => {
443
+ return {
444
+ // NOTE: Order is important.
445
+ transform: `translate(${offset.x}px, ${offset.y}px) scale(${scale})`,
446
+ visibility: width && height ? "visible" : "hidden"
447
+ };
448
+ }, [
449
+ scale,
450
+ offset
451
+ ]);
452
+ useImperativeHandle(forwardedRef, () => {
453
+ return {
454
+ setProjection: async (projection2) => {
455
+ setProjection(projection2);
402
456
  }
403
- }, [
457
+ };
458
+ }, [
459
+ ref
460
+ ]);
461
+ return /* @__PURE__ */ React2.createElement(CanvasContext.Provider, {
462
+ value: {
463
+ root: ref.current,
464
+ ready,
404
465
  width,
405
466
  height,
406
467
  scale,
407
- offset
408
- ]);
409
- const projection = useMemo(() => new ProjectionMapper(), []);
410
- useEffect2(() => {
411
- projection.update({
412
- width,
413
- height
414
- }, scale, offset);
415
- if (offset !== defaultOrigin) {
416
- setReady(true);
417
- }
418
- }, [
419
- projection,
420
- scale,
421
468
  offset,
422
- width,
423
- height
424
- ]);
425
- const styles = useMemo(() => {
426
- return {
427
- // NOTE: Order is important.
428
- transform: `translate(${offset.x}px, ${offset.y}px) scale(${scale})`,
429
- visibility: width && height ? "visible" : "hidden"
430
- };
431
- }, [
432
- scale,
433
- offset
434
- ]);
435
- useImperativeHandle(forwardedRef, () => {
436
- return {
437
- setProjection: async (projection2) => {
438
- setProjection(projection2);
439
- }
440
- };
441
- }, [
442
- ref
443
- ]);
444
- return /* @__PURE__ */ React2.createElement(CanvasContext.Provider, {
445
- value: {
446
- root: ref.current,
447
- ready,
448
- width,
449
- height,
450
- scale,
451
- offset,
452
- styles,
453
- projection,
454
- setProjection
455
- }
456
- }, /* @__PURE__ */ React2.createElement("div", {
457
- role: "none",
458
- ...props,
459
- className: mx2("absolute inset-0 overflow-hidden", classNames),
460
- ref
461
- }, ready ? children : null));
462
- } finally {
463
- _effect.f();
464
- }
469
+ styles,
470
+ projection,
471
+ setProjection
472
+ }
473
+ }, /* @__PURE__ */ React2.createElement("div", {
474
+ role: "none",
475
+ ...props,
476
+ className: mx2("absolute inset-0 overflow-hidden", classNames),
477
+ ref
478
+ }, ready ? children : null));
465
479
  });
466
480
 
467
- // packages/ui/react-ui-canvas/src/components/FPS.tsx
468
- import { useSignals as _useSignals3 } from "@preact-signals/safe-react/tracking";
469
- import React3, { useEffect as useEffect3, useReducer, useRef } from "react";
470
- import { mx as mx3 } from "@dxos/react-ui-theme";
481
+ // src/components/FPS.tsx
482
+ import React3, { useEffect as useEffect4, useReducer, useRef as useRef2 } from "react";
483
+ import { mx as mx3 } from "@dxos/ui-theme";
471
484
  var SEC = 1e3;
472
485
  var FPS = ({ classNames, width = 60, height = 30, bar = "bg-cyan-500" }) => {
473
- var _effect = _useSignals3();
474
- try {
475
- const [{ fps, max, len }, dispatch] = useReducer((state) => {
476
- const currentTime = Date.now();
477
- if (currentTime > state.prevTime + SEC) {
478
- const nextFPS = [
479
- ...new Array(Math.floor((currentTime - state.prevTime - SEC) / SEC)).fill(0),
480
- Math.max(1, Math.round(state.frames * SEC / (currentTime - state.prevTime)))
481
- ];
482
- return {
483
- max: Math.max(state.max, ...nextFPS),
484
- len: Math.min(state.len + nextFPS.length, width),
485
- fps: [
486
- ...state.fps,
487
- ...nextFPS
488
- ].slice(-width),
489
- frames: 1,
490
- prevTime: currentTime
491
- };
492
- } else {
493
- return {
494
- ...state,
495
- frames: state.frames + 1
496
- };
497
- }
498
- }, {
499
- max: 0,
500
- len: 0,
501
- fps: [],
502
- frames: 0,
503
- prevTime: Date.now()
504
- });
505
- const requestRef = useRef();
506
- const tick = () => {
507
- dispatch();
508
- requestRef.current = requestAnimationFrame(tick);
509
- };
510
- useEffect3(() => {
511
- requestRef.current = requestAnimationFrame(tick);
512
- return () => {
513
- if (requestRef.current) {
514
- cancelAnimationFrame(requestRef.current);
515
- }
486
+ const [{ fps, max, len }, dispatch] = useReducer((state) => {
487
+ const currentTime = Date.now();
488
+ if (currentTime > state.prevTime + SEC) {
489
+ const nextFPS = [
490
+ ...new Array(Math.floor((currentTime - state.prevTime - SEC) / SEC)).fill(0),
491
+ Math.max(1, Math.round(state.frames * SEC / (currentTime - state.prevTime)))
492
+ ];
493
+ return {
494
+ max: Math.max(state.max, ...nextFPS),
495
+ len: Math.min(state.len + nextFPS.length, width),
496
+ fps: [
497
+ ...state.fps,
498
+ ...nextFPS
499
+ ].slice(-width),
500
+ frames: 1,
501
+ prevTime: currentTime
516
502
  };
517
- }, []);
518
- return /* @__PURE__ */ React3.createElement("div", {
519
- style: {
520
- width: width + 6
521
- },
522
- className: mx3("relative flex flex-col p-0.5", "bg-baseSurface text-xs text-subdued font-thin pointer-events-none border border-separator", classNames)
523
- }, /* @__PURE__ */ React3.createElement("div", null, fps[len - 1], " FPS"), /* @__PURE__ */ React3.createElement("div", {
524
- className: "w-full relative",
525
- style: {
526
- height
527
- }
528
- }, fps.map((frame, i) => /* @__PURE__ */ React3.createElement("div", {
529
- key: `fps-${i}`,
530
- className: bar,
531
- style: {
532
- position: "absolute",
533
- bottom: 0,
534
- right: `${len - 1 - i}px`,
535
- height: `${height * frame / max}px`,
536
- width: 1
503
+ } else {
504
+ return {
505
+ ...state,
506
+ frames: state.frames + 1
507
+ };
508
+ }
509
+ }, {
510
+ max: 0,
511
+ len: 0,
512
+ fps: [],
513
+ frames: 0,
514
+ prevTime: Date.now()
515
+ });
516
+ const requestRef = useRef2(null);
517
+ const tick = () => {
518
+ dispatch();
519
+ requestRef.current = requestAnimationFrame(tick);
520
+ };
521
+ useEffect4(() => {
522
+ requestRef.current = requestAnimationFrame(tick);
523
+ return () => {
524
+ if (requestRef.current) {
525
+ cancelAnimationFrame(requestRef.current);
537
526
  }
538
- }))));
539
- } finally {
540
- _effect.f();
541
- }
527
+ };
528
+ }, []);
529
+ return /* @__PURE__ */ React3.createElement("div", {
530
+ style: {
531
+ width: width + 6
532
+ },
533
+ className: mx3("relative flex flex-col p-0.5", "bg-baseSurface text-xs text-subdued font-thin pointer-events-none border border-separator", classNames)
534
+ }, /* @__PURE__ */ React3.createElement("div", null, fps[len - 1], " FPS"), /* @__PURE__ */ React3.createElement("div", {
535
+ className: "is-full relative",
536
+ style: {
537
+ height
538
+ }
539
+ }, fps.map((frame, i) => /* @__PURE__ */ React3.createElement("div", {
540
+ key: `fps-${i}`,
541
+ className: bar,
542
+ style: {
543
+ position: "absolute",
544
+ bottom: 0,
545
+ right: `${len - 1 - i}px`,
546
+ height: `${height * frame / max}px`,
547
+ width: 1
548
+ }
549
+ }))));
542
550
  };
543
551
 
544
- // packages/ui/react-ui-canvas/src/components/Grid/Grid.tsx
545
- import { useSignals as _useSignals4 } from "@preact-signals/safe-react/tracking";
546
- import React4, { forwardRef as forwardRef2, useMemo as useMemo2, useId } from "react";
552
+ // src/components/Grid/Grid.tsx
553
+ import React4, { forwardRef as forwardRef2, useId, useMemo as useMemo2 } from "react";
547
554
  import { useForwardedRef } from "@dxos/react-ui";
548
- import { mx as mx4 } from "@dxos/react-ui-theme";
555
+ import { mx as mx4 } from "@dxos/ui-theme";
549
556
  var gridRatios = [
550
557
  1 / 4,
551
558
  1,
@@ -558,67 +565,57 @@ var defaultOffset = {
558
565
  y: 0
559
566
  };
560
567
  var createId = (parent, grid) => `dx-canvas-grid-${parent}-${grid}`;
561
- var GridComponent = /* @__PURE__ */ forwardRef2(({ size: gridSize = defaultGridSize, scale = 1, offset = defaultOffset, showAxes = true, classNames }, forwardedRef) => {
562
- var _effect = _useSignals4();
563
- try {
564
- const svgRef = useForwardedRef(forwardedRef);
565
- const instanceId = useId();
566
- const grids = useMemo2(() => gridRatios.map((ratio) => ({
567
- id: ratio,
568
- size: ratio * gridSize * scale
569
- })).filter(({ size }) => size >= gridSize && size <= 256), [
570
- gridSize,
571
- scale
572
- ]);
573
- const { width = 0, height = 0 } = svgRef.current?.getBoundingClientRect() ?? {};
574
- return /* @__PURE__ */ React4.createElement("svg", {
575
- ...testId("dx-canvas-grid"),
576
- ref: svgRef,
577
- className: mx4("absolute inset-0 w-full h-full pointer-events-none touch-none select-none", "stroke-neutral-500", classNames)
578
- }, /* @__PURE__ */ React4.createElement("defs", null, grids.map(({ id, size }) => /* @__PURE__ */ React4.createElement(GridPattern, {
579
- key: id,
580
- id: createId(instanceId, id),
581
- offset,
582
- size
583
- }))), showAxes && /* @__PURE__ */ React4.createElement(React4.Fragment, null, /* @__PURE__ */ React4.createElement("line", {
584
- x1: 0,
585
- y1: offset.y,
586
- x2: width,
587
- y2: offset.y,
588
- className: "stroke-neutral-500 opacity-40"
589
- }), /* @__PURE__ */ React4.createElement("line", {
590
- x1: offset.x,
591
- y1: 0,
592
- x2: offset.x,
593
- y2: height,
594
- className: "stroke-neutral-500 opacity-40"
595
- })), /* @__PURE__ */ React4.createElement("g", null, grids.map(({ id }, i) => /* @__PURE__ */ React4.createElement("rect", {
596
- key: id,
597
- opacity: 0.1 + i * 0.05,
598
- fill: `url(#${createId(instanceId, id)})`,
599
- width: "100%",
600
- height: "100%"
601
- }))));
602
- } finally {
603
- _effect.f();
604
- }
605
- });
606
568
  var Grid = (props) => {
607
- var _effect = _useSignals4();
608
- try {
609
- const { scale, offset } = useCanvasContext();
610
- return /* @__PURE__ */ React4.createElement(GridComponent, {
611
- ...props,
612
- scale,
613
- offset
614
- });
615
- } finally {
616
- _effect.f();
617
- }
569
+ const { scale, offset } = useCanvasContext();
570
+ return /* @__PURE__ */ React4.createElement(GridComponent, {
571
+ ...props,
572
+ scale,
573
+ offset
574
+ });
618
575
  };
576
+ var GridComponent = /* @__PURE__ */ forwardRef2(({ size: gridSize = defaultGridSize, scale = 1, offset = defaultOffset, showAxes = true, classNames }, forwardedRef) => {
577
+ const svgRef = useForwardedRef(forwardedRef);
578
+ const { width = 0, height = 0 } = svgRef.current?.getBoundingClientRect() ?? {};
579
+ const instanceId = useId();
580
+ const grids = useMemo2(() => gridRatios.map((ratio) => ({
581
+ id: ratio,
582
+ size: ratio * gridSize * scale
583
+ })).filter(({ size }) => size >= gridSize && size <= 128), [
584
+ gridSize,
585
+ scale
586
+ ]);
587
+ return /* @__PURE__ */ React4.createElement("svg", {
588
+ ...testId("dx-canvas-grid"),
589
+ ref: svgRef,
590
+ className: mx4("absolute inset-0 is-full bs-full pointer-events-none touch-none select-none", "stroke-neutral-500", classNames)
591
+ }, /* @__PURE__ */ React4.createElement("defs", null, grids.map(({ id, size }) => /* @__PURE__ */ React4.createElement(GridPattern, {
592
+ key: id,
593
+ id: createId(instanceId, id),
594
+ offset,
595
+ size
596
+ }))), showAxes && /* @__PURE__ */ React4.createElement(React4.Fragment, null, /* @__PURE__ */ React4.createElement("line", {
597
+ x1: 0,
598
+ y1: offset.y,
599
+ x2: width,
600
+ y2: offset.y,
601
+ className: "stroke-neutral-500 opacity-40"
602
+ }), /* @__PURE__ */ React4.createElement("line", {
603
+ x1: offset.x,
604
+ y1: 0,
605
+ x2: offset.x,
606
+ y2: height,
607
+ className: "stroke-neutral-500 opacity-40"
608
+ })), /* @__PURE__ */ React4.createElement("g", null, grids.map(({ id }, i) => /* @__PURE__ */ React4.createElement("rect", {
609
+ key: id,
610
+ opacity: 0.1 + i * 0.05,
611
+ fill: `url(#${createId(instanceId, id)})`,
612
+ width: "100%",
613
+ height: "100%"
614
+ }))));
615
+ });
619
616
 
620
- // packages/ui/react-ui-canvas/src/types.ts
621
- import { Schema } from "effect";
617
+ // src/types.ts
618
+ import * as Schema from "effect/Schema";
622
619
  var Point = Schema.Struct({
623
620
  x: Schema.Number,
624
621
  y: Schema.Number
@@ -650,6 +647,7 @@ export {
650
647
  inspectElement,
651
648
  testId,
652
649
  useCanvasContext,
650
+ useDrag,
653
651
  useWheel,
654
652
  zoomInPlace,
655
653
  zoomTo