@dxos/react-ui-geo 0.8.4-main.84f28bd → 0.8.4-main.a4bbb77

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 (74) hide show
  1. package/data/airports.ts +1 -1
  2. package/data/cities.ts +1 -1
  3. package/data/countries-110m.ts +1 -1
  4. package/data/countries-dots-3.ts +1 -1
  5. package/data/countries-dots-4.ts +1 -1
  6. package/dist/lib/browser/chunk-GMWLKTLN.mjs +9 -0
  7. package/dist/lib/browser/{countries-110m-37VAAFCK.mjs → countries-110m-ZM3ZIEFS.mjs} +1 -1
  8. package/dist/lib/browser/countries-110m-ZM3ZIEFS.mjs.map +7 -0
  9. package/dist/lib/browser/data.mjs +1 -1
  10. package/dist/lib/browser/index.mjs +174 -146
  11. package/dist/lib/browser/index.mjs.map +3 -3
  12. package/dist/lib/browser/meta.json +1 -1
  13. package/dist/lib/node-esm/{chunk-OPJPAAEK.mjs → chunk-JODBF4CC.mjs} +2 -2
  14. package/dist/lib/node-esm/{countries-110m-36TTKK5B.mjs → countries-110m-3SFASWVD.mjs} +1 -1
  15. package/dist/lib/node-esm/countries-110m-3SFASWVD.mjs.map +7 -0
  16. package/dist/lib/node-esm/data.mjs +1 -1
  17. package/dist/lib/node-esm/index.mjs +174 -146
  18. package/dist/lib/node-esm/index.mjs.map +3 -3
  19. package/dist/lib/node-esm/meta.json +1 -1
  20. package/dist/types/src/components/Globe/Globe.d.ts +1 -1
  21. package/dist/types/src/components/Globe/Globe.d.ts.map +1 -1
  22. package/dist/types/src/components/Globe/Globe.stories.d.ts +25 -9
  23. package/dist/types/src/components/Globe/Globe.stories.d.ts.map +1 -1
  24. package/dist/types/src/components/Map/Map.d.ts +25 -12
  25. package/dist/types/src/components/Map/Map.d.ts.map +1 -1
  26. package/dist/types/src/components/Map/Map.stories.d.ts +14 -8
  27. package/dist/types/src/components/Map/Map.stories.d.ts.map +1 -1
  28. package/dist/types/src/components/Toolbar/Controls.d.ts.map +1 -1
  29. package/dist/types/src/components/index.d.ts +0 -1
  30. package/dist/types/src/components/index.d.ts.map +1 -1
  31. package/dist/types/src/hooks/context.d.ts +7 -7
  32. package/dist/types/src/hooks/context.d.ts.map +1 -1
  33. package/dist/types/src/hooks/useGlobeZoomHandler.d.ts +1 -1
  34. package/dist/types/src/hooks/useGlobeZoomHandler.d.ts.map +1 -1
  35. package/dist/types/src/hooks/useMapZoomHandler.d.ts +1 -1
  36. package/dist/types/src/hooks/useMapZoomHandler.d.ts.map +1 -1
  37. package/dist/types/src/hooks/useSpinner.d.ts +1 -1
  38. package/dist/types/src/hooks/useSpinner.d.ts.map +1 -1
  39. package/dist/types/src/hooks/useTour.d.ts +4 -3
  40. package/dist/types/src/hooks/useTour.d.ts.map +1 -1
  41. package/dist/types/src/index.d.ts +1 -1
  42. package/dist/types/src/index.d.ts.map +1 -1
  43. package/dist/types/src/types.d.ts +2 -1
  44. package/dist/types/src/types.d.ts.map +1 -1
  45. package/dist/types/src/util/path.d.ts +5 -8
  46. package/dist/types/src/util/path.d.ts.map +1 -1
  47. package/dist/types/src/util/render.d.ts +4 -4
  48. package/dist/types/src/util/render.d.ts.map +1 -1
  49. package/dist/types/tsconfig.tsbuildinfo +1 -1
  50. package/package.json +22 -19
  51. package/src/components/Globe/Globe.stories.tsx +81 -33
  52. package/src/components/Globe/Globe.tsx +74 -58
  53. package/src/components/Map/Map.stories.tsx +25 -14
  54. package/src/components/Map/Map.tsx +206 -91
  55. package/src/components/Toolbar/Controls.tsx +2 -6
  56. package/src/components/index.ts +0 -2
  57. package/src/hooks/context.tsx +10 -10
  58. package/src/hooks/useGlobeZoomHandler.ts +3 -3
  59. package/src/hooks/useMapZoomHandler.ts +1 -1
  60. package/src/hooks/useSpinner.ts +2 -1
  61. package/src/hooks/useTour.ts +9 -8
  62. package/src/index.ts +1 -1
  63. package/src/types.ts +3 -1
  64. package/src/util/inertia.ts +1 -1
  65. package/src/util/path.ts +5 -6
  66. package/src/util/render.ts +5 -3
  67. package/dist/lib/browser/chunk-CYCBMCOP.mjs +0 -9
  68. package/dist/lib/browser/countries-110m-37VAAFCK.mjs.map +0 -7
  69. package/dist/lib/node-esm/countries-110m-36TTKK5B.mjs.map +0 -7
  70. package/dist/types/src/components/types.d.ts +0 -15
  71. package/dist/types/src/components/types.d.ts.map +0 -1
  72. package/src/components/types.ts +0 -19
  73. /package/dist/lib/browser/{chunk-CYCBMCOP.mjs.map → chunk-GMWLKTLN.mjs.map} +0 -0
  74. /package/dist/lib/node-esm/{chunk-OPJPAAEK.mjs.map → chunk-JODBF4CC.mjs.map} +0 -0
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dxos/react-ui-geo",
3
- "version": "0.8.4-main.84f28bd",
3
+ "version": "0.8.4-main.a4bbb77",
4
4
  "description": "Geo components.",
5
5
  "homepage": "https://github.com/dxos",
6
6
  "bugs": "https://github.com/dxos/issues",
@@ -10,11 +10,13 @@
10
10
  "type": "module",
11
11
  "exports": {
12
12
  "./data": {
13
+ "source": "./src/data.ts",
13
14
  "types": "./dist/types/src/data.d.ts",
14
15
  "browser": "./dist/lib/browser/data.mjs",
15
16
  "node": "./dist/lib/node-esm/data.mjs"
16
17
  },
17
18
  ".": {
19
+ "source": "./src/index.ts",
18
20
  "types": "./dist/types/src/index.d.ts",
19
21
  "browser": "./dist/lib/browser/index.mjs",
20
22
  "node": "./dist/lib/node-esm/index.mjs"
@@ -35,31 +37,32 @@
35
37
  ],
36
38
  "dependencies": {
37
39
  "@preact-signals/safe-react": "^0.9.0",
40
+ "@radix-ui/react-context": "^1.0.5",
38
41
  "d3": "^7.9.0",
39
42
  "d3-geo-projection": "^4.0.0",
40
43
  "d3-hexbin": "^0.2.2",
41
44
  "geojson": "^0.5.0",
42
45
  "leaflet": "^1.9.4",
43
46
  "lodash.defaultsdeep": "^4.6.1",
44
- "react-leaflet": "^4.2.1",
47
+ "react-leaflet": "^5.0.0",
45
48
  "react-resize-detector": "^11.0.1",
46
49
  "topojson-client": "^3.1.0",
47
50
  "topojson-simplify": "^3.0.3",
48
51
  "versor": "^0.2.0",
49
- "@dxos/async": "0.8.4-main.84f28bd",
50
- "@dxos/debug": "0.8.4-main.84f28bd",
51
- "@dxos/log": "0.8.4-main.84f28bd",
52
- "@dxos/node-std": "0.8.4-main.84f28bd",
53
- "@dxos/util": "0.8.4-main.84f28bd"
52
+ "@dxos/async": "0.8.4-main.a4bbb77",
53
+ "@dxos/log": "0.8.4-main.a4bbb77",
54
+ "@dxos/debug": "0.8.4-main.a4bbb77",
55
+ "@dxos/node-std": "0.8.4-main.a4bbb77",
56
+ "@dxos/util": "0.8.4-main.a4bbb77"
54
57
  },
55
58
  "devDependencies": {
56
59
  "@react-three/drei": "^9.99.0",
57
- "@react-three/fiber": "^8.15.0",
60
+ "@react-three/fiber": "^9.3.0",
58
61
  "@types/d3": "^7.4.3",
59
62
  "@types/geojson": "^7946.0.14",
60
63
  "@types/leaflet": "^1.9.16",
61
- "@types/react": "~18.2.0",
62
- "@types/react-dom": "~18.2.0",
64
+ "@types/react": "~19.2.0",
65
+ "@types/react-dom": "~19.2.0",
63
66
  "@types/three": "0.165.0",
64
67
  "@types/topojson-client": "^3.1.4",
65
68
  "@types/topojson-simplify": "^3.0.3",
@@ -67,18 +70,18 @@
67
70
  "JSONStream": "^1.3.5",
68
71
  "geojson2h3": "^1.2.0",
69
72
  "leva": "^0.9.35",
70
- "react": "~18.2.0",
71
- "react-dom": "~18.2.0",
73
+ "react": "~19.2.0",
74
+ "react-dom": "~19.2.0",
72
75
  "three": "0.165.0",
73
- "@dxos/react-ui": "0.8.4-main.84f28bd",
74
- "@dxos/react-ui-theme": "0.8.4-main.84f28bd",
75
- "@dxos/storybook-utils": "0.8.4-main.84f28bd"
76
+ "@dxos/react-ui": "0.8.4-main.a4bbb77",
77
+ "@dxos/react-ui-theme": "0.8.4-main.a4bbb77",
78
+ "@dxos/storybook-utils": "0.8.4-main.a4bbb77"
76
79
  },
77
80
  "peerDependencies": {
78
- "react": "~18.2.0",
79
- "react-dom": "~18.2.0",
80
- "@dxos/react-ui": "0.8.4-main.84f28bd",
81
- "@dxos/react-ui-theme": "0.8.4-main.84f28bd"
81
+ "react": "^19.0.0",
82
+ "react-dom": "^19.0.0",
83
+ "@dxos/react-ui": "0.8.4-main.a4bbb77",
84
+ "@dxos/react-ui-theme": "0.8.4-main.a4bbb77"
82
85
  },
83
86
  "publishConfig": {
84
87
  "access": "public"
@@ -2,22 +2,22 @@
2
2
  // Copyright 2018 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
7
- import { type Meta } from '@storybook/react-vite';
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
8
6
  import { type FeatureCollection, type Geometry, type Position } from 'geojson';
9
7
  import { Leva } from 'leva';
10
8
  import React, { useMemo, useRef, useState } from 'react';
11
9
  import { type Topology } from 'topojson-specification';
12
10
 
13
11
  import { useAsyncState } from '@dxos/react-ui';
14
- import { withTheme, withLayout } from '@dxos/storybook-utils';
12
+ import { withTheme } from '@dxos/react-ui/testing';
15
13
 
16
- import { Globe, type GlobeCanvasProps, type GlobeController, type GlobeRootProps } from './Globe';
17
- import { useDrag, useGlobeZoomHandler, useSpinner, useTour, type Vector } from '../../hooks';
18
- import { closestPoint, type LatLng, type StyleSet } from '../../util';
14
+ import { type Vector, useDrag, useGlobeZoomHandler, useSpinner, useTour } from '../../hooks';
15
+ import { type LatLngLiteral } from '../../types';
16
+ import { type StyleSet, closestPoint } from '../../util';
19
17
  import { type ControlProps } from '../Toolbar';
20
18
 
19
+ import { Globe, type GlobeCanvasProps, type GlobeController, type GlobeRootProps } from './Globe';
20
+
21
21
  // TODO(burdon): Load from JSON at runtime?
22
22
  const useTopology = () => {
23
23
  return useAsyncState(async () => (await import('../../../data/countries-110m.ts')).default);
@@ -95,8 +95,11 @@ const createTrip = (
95
95
  routes: Record<string, string[]>,
96
96
  points: Position[] = [],
97
97
  ) => {
98
- let previousHub: LatLng;
99
- return Object.entries(routes).reduce<{ points: LatLng[]; lines: { source: LatLng; target: LatLng }[] }>(
98
+ let previousHub: LatLngLiteral;
99
+ return Object.entries(routes).reduce<{
100
+ points: LatLngLiteral[];
101
+ lines: { source: LatLngLiteral; target: LatLngLiteral }[];
102
+ }>(
100
103
  (features, [hub, regional]) => {
101
104
  const hubAirport = airports.features.find(({ properties }) => properties.iata === hub);
102
105
  if (hubAirport) {
@@ -125,7 +128,7 @@ const createTrip = (
125
128
  );
126
129
  };
127
130
 
128
- type StoryProps = Pick<GlobeRootProps, 'scale' | 'translation' | 'rotation'> &
131
+ type StoryProps = Pick<GlobeRootProps, 'zoom' | 'translation' | 'rotation'> &
129
132
  Pick<GlobeCanvasProps, 'projection' | 'styles'> & {
130
133
  drag?: boolean;
131
134
  spin?: boolean;
@@ -133,8 +136,8 @@ type StoryProps = Pick<GlobeRootProps, 'scale' | 'translation' | 'rotation'> &
133
136
  xAxis?: boolean;
134
137
  };
135
138
 
136
- const Story = ({
137
- scale: _scale = 1,
139
+ const DefaultStory = ({
140
+ zoom: _zoom = 1,
138
141
  translation,
139
142
  rotation = [0, 0, 0],
140
143
  projection,
@@ -154,6 +157,7 @@ const Story = ({
154
157
  });
155
158
  const [topology] = useTopology();
156
159
  const [airports] = useAsyncState(async () => (await import('../../../data/airports.ts')).default);
160
+
157
161
  const features = useMemo(() => {
158
162
  return airports ? createTrip(airports, routes, (dots?.objects.dots as any)?.geometries[0].coordinates) : undefined;
159
163
  }, [airports, routes, dots]);
@@ -188,18 +192,18 @@ const Story = ({
188
192
  break;
189
193
  }
190
194
  case 'zoom-in': {
191
- controller.current.setScale((scale) => scale * 1.1);
195
+ controller.current.setZoom((scale) => scale * 1.1);
192
196
  break;
193
197
  }
194
198
  case 'zoom-out': {
195
- controller.current.setScale((scale) => scale * 0.9);
199
+ controller.current.setZoom((scale) => scale * 0.9);
196
200
  break;
197
201
  }
198
202
  }
199
203
  };
200
204
 
201
205
  return (
202
- <Globe.Root classNames='absolute inset-0' scale={_scale} translation={translation} rotation={rotation}>
206
+ <Globe.Root classNames='absolute inset-0' zoom={_zoom} translation={translation} rotation={rotation}>
203
207
  <Globe.Canvas
204
208
  ref={controller}
205
209
  topology={styles?.dots ? dots : topology}
@@ -219,11 +223,15 @@ const Story = ({
219
223
 
220
224
  const initialRotation: Vector = [0, -40, 0];
221
225
 
222
- const meta: Meta = {
226
+ const meta = {
223
227
  title: 'ui/react-ui-geo/Globe',
224
228
  component: Globe.Root,
225
- decorators: [withTheme, withLayout({ fullscreen: true, classNames: 'bg-[#000]' })],
226
- };
229
+ render: DefaultStory,
230
+ decorators: [withTheme],
231
+ parameters: {
232
+ layout: 'fullscreen',
233
+ },
234
+ } satisfies Meta;
227
235
 
228
236
  export default meta;
229
237
 
@@ -234,7 +242,7 @@ export const Earth1 = () => {
234
242
  useDrag(controller);
235
243
 
236
244
  return (
237
- <Globe.Root scale={1.2} rotation={[Math.random() * 360, 0, 0]}>
245
+ <Globe.Root zoom={1.2} rotation={[Math.random() * 360, 0, 0]}>
238
246
  <Globe.Canvas ref={setController} topology={topology} styles={defaultStyles} />
239
247
  <Globe.Zoom onAction={handleAction} />
240
248
  </Globe.Root>
@@ -249,7 +257,7 @@ export const Earth2 = () => {
249
257
 
250
258
  return (
251
259
  <div className='absolute bottom-0 left-0 right-0 '>
252
- <Globe.Root classNames='h-[400px]' scale={2.8} translation={{ x: 0, y: 400 }}>
260
+ <Globe.Root classNames='h-[400px]' zoom={2.8} translation={{ x: 0, y: 400 }}>
253
261
  <Globe.Canvas ref={setController} topology={topology} styles={defaultStyles} />
254
262
  <Globe.Zoom onAction={handleAction} />
255
263
  </Globe.Root>
@@ -283,33 +291,73 @@ export const Mercator = () => {
283
291
  useDrag(controller);
284
292
 
285
293
  return (
286
- <Globe.Root classNames='flex grow overflow-hidden' scale={0.7} rotation={initialRotation}>
294
+ <Globe.Root classNames='flex grow overflow-hidden' zoom={0.7} rotation={initialRotation}>
287
295
  <Globe.Canvas ref={setController} topology={topology} projection='mercator' styles={monochrome} />
288
296
  <Globe.Zoom onAction={handleAction} />
289
297
  </Globe.Root>
290
298
  );
291
299
  };
292
300
 
293
- export const Globe1 = () => {
294
- return <Story drag projection='mercator' scale={0.8} rotation={initialRotation} styles={defaultStyles} />;
301
+ type Story = StoryObj<typeof DefaultStory>;
302
+
303
+ export const Globe1: Story = {
304
+ args: {
305
+ drag: true,
306
+ projection: 'mercator',
307
+ zoom: 0.8,
308
+ rotation: initialRotation,
309
+ styles: defaultStyles,
310
+ },
295
311
  };
296
312
 
297
- export const Globe2 = () => {
298
- return <Story drag projection='transverse-mercator' scale={0.8} rotation={initialRotation} styles={defaultStyles} />;
313
+ export const Globe2: Story = {
314
+ args: {
315
+ drag: true,
316
+ projection: 'transverse-mercator',
317
+ zoom: 0.8,
318
+ rotation: initialRotation,
319
+ styles: defaultStyles,
320
+ },
299
321
  };
300
322
 
301
- export const Globe3 = () => {
302
- return <Story drag spin scale={1.5} rotation={initialRotation} styles={defaultStyles} />;
323
+ export const Globe3: Story = {
324
+ args: {
325
+ drag: true,
326
+ spin: true,
327
+ zoom: 1.5,
328
+ rotation: initialRotation,
329
+ styles: defaultStyles,
330
+ },
303
331
  };
304
332
 
305
- export const Globe4 = () => {
306
- return <Story drag tour scale={2} rotation={initialRotation} styles={defaultStyles} />;
333
+ export const Globe4: Story = {
334
+ args: {
335
+ drag: true,
336
+ tour: true,
337
+ zoom: 2,
338
+ rotation: initialRotation,
339
+ styles: defaultStyles,
340
+ },
307
341
  };
308
342
 
309
- export const Globe5 = () => {
310
- return <Story drag tour scale={0.9} rotation={initialRotation} styles={dotStyles} />;
343
+ export const Globe5: Story = {
344
+ args: {
345
+ drag: true,
346
+ tour: true,
347
+ zoom: 0.9,
348
+ rotation: initialRotation,
349
+ styles: dotStyles,
350
+ },
311
351
  };
312
352
 
313
- export const Globe6 = () => {
314
- return <Story drag xAxis tour scale={2} translation={{ x: 0, y: 600 }} rotation={[0, -20, 0]} styles={dotStyles} />;
353
+ export const Globe6: Story = {
354
+ args: {
355
+ drag: true,
356
+ xAxis: true,
357
+ tour: true,
358
+ zoom: 2,
359
+ translation: { x: 0, y: 600 },
360
+ rotation: [0, -20, 0],
361
+ styles: dotStyles,
362
+ },
315
363
  };
@@ -4,14 +4,14 @@
4
4
 
5
5
  import {
6
6
  type GeoProjection,
7
+ easeLinear,
8
+ easeSinOut,
7
9
  geoMercator,
8
10
  geoOrthographic,
9
11
  geoPath,
10
12
  geoTransverseMercator,
11
13
  interpolateNumber,
12
14
  transition,
13
- easeLinear,
14
- easeSinOut,
15
15
  } from 'd3';
16
16
  import { type ControlPosition } from 'leaflet';
17
17
  import React, {
@@ -26,7 +26,7 @@ import React, {
26
26
  import { useResizeDetector } from 'react-resize-detector';
27
27
  import { type Topology } from 'topojson-specification';
28
28
 
29
- import { type ThemedClassName, type ThemeMode, useDynamicRef, useThemeContext } from '@dxos/react-ui';
29
+ import { type ThemeMode, type ThemedClassName, useDynamicRef, useThemeContext } from '@dxos/react-ui';
30
30
  import { mx } from '@dxos/react-ui-theme';
31
31
 
32
32
  import {
@@ -44,7 +44,7 @@ import {
44
44
  renderLayers,
45
45
  timer,
46
46
  } from '../../util';
47
- import { ZoomControls, ActionControls, type ControlProps, controlPositions } from '../Toolbar';
47
+ import { ActionControls, type ControlProps, ZoomControls, controlPositions } from '../Toolbar';
48
48
 
49
49
  /**
50
50
  * https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute
@@ -103,7 +103,7 @@ const defaultStyles: Record<ThemeMode, StyleSet> = {
103
103
  export type GlobeController = {
104
104
  canvas: HTMLCanvasElement;
105
105
  projection: GeoProjection;
106
- } & Pick<GlobeContextType, 'scale' | 'translation' | 'rotation' | 'setScale' | 'setTranslation' | 'setRotation'>;
106
+ } & Pick<GlobeContextType, 'zoom' | 'translation' | 'rotation' | 'setZoom' | 'setTranslation' | 'setRotation'>;
107
107
 
108
108
  export type ProjectionType = 'orthographic' | 'mercator' | 'transverse-mercator';
109
109
 
@@ -154,6 +154,7 @@ type GlobeCanvasProps = {
154
154
  * Basic globe renderer.
155
155
  * https://github.com/topojson/world-atlas
156
156
  */
157
+ // TODO(burdon): Move controller to root.
157
158
  const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
158
159
  ({ projection: _projection, topology, features, styles: _styles }, forwardRef) => {
159
160
  const { themeMode } = useThemeContext();
@@ -173,55 +174,50 @@ const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
173
174
  }, [topology, features, styles]);
174
175
 
175
176
  // State.
176
- const { size, center, scale, translation, rotation, setCenter, setScale, setTranslation, setRotation } =
177
+ const { size, center, zoom, translation, rotation, setCenter, setZoom, setTranslation, setRotation } =
177
178
  useGlobeContext();
178
-
179
- const scaleRef = useDynamicRef(scale);
179
+ const zoomRef = useDynamicRef(zoom);
180
180
 
181
181
  // Update rotation.
182
182
  useEffect(() => {
183
183
  if (center) {
184
- setScale(1);
184
+ setZoom(1);
185
185
  setRotation(positionToRotation(geoToPosition(center)));
186
186
  }
187
187
  }, [center]);
188
188
 
189
189
  // External controller.
190
190
  const zooming = useRef(false);
191
- useImperativeHandle<GlobeController, GlobeController>(
192
- forwardRef,
193
- () => {
194
- return {
195
- canvas,
196
- projection,
197
- center,
198
- get scale() {
199
- return scaleRef.current;
200
- },
201
- translation,
202
- rotation,
203
- setCenter,
204
- setScale: (s) => {
205
- if (typeof s === 'function') {
206
- const is = interpolateNumber(scaleRef.current, s(scaleRef.current));
207
- // Stop easing if already zooming.
208
- transition()
209
- .ease(zooming.current ? easeLinear : easeSinOut)
210
- .duration(200)
211
- .tween('scale', () => (t) => setScale(is(t)))
212
- .on('end', () => {
213
- zooming.current = false;
214
- });
215
- } else {
216
- setScale(s);
217
- }
218
- },
219
- setTranslation,
220
- setRotation,
221
- };
222
- },
223
- [canvas],
224
- );
191
+ useImperativeHandle<GlobeController, GlobeController>(forwardRef, () => {
192
+ return {
193
+ canvas,
194
+ projection,
195
+ center,
196
+ get zoom() {
197
+ return zoomRef.current;
198
+ },
199
+ translation,
200
+ rotation,
201
+ setCenter,
202
+ setZoom: (s) => {
203
+ if (typeof s === 'function') {
204
+ const is = interpolateNumber(zoomRef.current, s(zoomRef.current));
205
+ // Stop easing if already zooming.
206
+ transition()
207
+ .ease(zooming.current ? easeLinear : easeSinOut)
208
+ .duration(200)
209
+ .tween('scale', () => (t) => setZoom(is(t)))
210
+ .on('end', () => {
211
+ zooming.current = false;
212
+ });
213
+ } else {
214
+ setZoom(s);
215
+ }
216
+ },
217
+ setTranslation,
218
+ setRotation,
219
+ };
220
+ }, [canvas]);
225
221
 
226
222
  // https://d3js.org/d3-geo/path#geoPath
227
223
  // https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/getContext
@@ -236,14 +232,14 @@ const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
236
232
  timer(() => {
237
233
  // https://d3js.org/d3-geo/projection
238
234
  projection
239
- .scale((Math.min(size.width, size.height) / 2) * scale)
235
+ .scale((Math.min(size.width, size.height) / 2) * zoom)
240
236
  .translate([size.width / 2 + (translation?.x ?? 0), size.height / 2 + (translation?.y ?? 0)])
241
237
  .rotate(rotation ?? [0, 0, 0]);
242
238
 
243
- renderLayers(generator, layers, scale, styles);
239
+ renderLayers(generator, layers, zoom, styles);
244
240
  });
245
241
  }
246
- }, [generator, size, scale, translation, rotation, layers]);
242
+ }, [generator, size, zoom, translation, rotation, layers]);
247
243
 
248
244
  if (!size.width || !size.height) {
249
245
  return null;
@@ -253,8 +249,12 @@ const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
253
249
  },
254
250
  );
255
251
 
252
+ //
253
+ // Debug
254
+ //
255
+
256
256
  const GlobeDebug = ({ position = 'topleft' }: { position?: ControlPosition }) => {
257
- const { size, scale, translation, rotation } = useGlobeContext();
257
+ const { size, zoom, translation, rotation } = useGlobeContext();
258
258
  return (
259
259
  <div
260
260
  className={mx(
@@ -263,12 +263,16 @@ const GlobeDebug = ({ position = 'topleft' }: { position?: ControlPosition }) =>
263
263
  )}
264
264
  >
265
265
  <pre className='font-mono text-xs text-green-700'>
266
- {JSON.stringify({ size, scale, translation, rotation }, null, 2)}
266
+ {JSON.stringify({ size, zoom, translation, rotation }, null, 2)}
267
267
  </pre>
268
268
  </div>
269
269
  );
270
270
  };
271
271
 
272
+ //
273
+ // Panel
274
+ //
275
+
272
276
  const GlobePanel = ({
273
277
  position,
274
278
  classNames,
@@ -277,25 +281,37 @@ const GlobePanel = ({
277
281
  return <div className={mx('z-10 absolute overflow-hidden', controlPositions[position], classNames)}>{children}</div>;
278
282
  };
279
283
 
284
+ //
285
+ // Controls
286
+ //
287
+
280
288
  const CustomControl = ({ position, children }: PropsWithChildren<{ position: ControlPosition }>) => {
281
289
  return <div className={mx('z-10 absolute overflow-hidden', controlPositions[position])}>{children}</div>;
282
290
  };
283
291
 
284
292
  type GlobeControlProps = { position?: ControlPosition } & Pick<ControlProps, 'onAction'>;
285
293
 
294
+ const GlobeZoom = ({ onAction, position = 'bottomleft', ...props }: GlobeControlProps) => (
295
+ <CustomControl position={position} {...props}>
296
+ <ZoomControls onAction={onAction} />
297
+ </CustomControl>
298
+ );
299
+
300
+ const GlobeAction = ({ onAction, position = 'bottomright', ...props }: GlobeControlProps) => (
301
+ <CustomControl position={position} {...props}>
302
+ <ActionControls onAction={onAction} />
303
+ </CustomControl>
304
+ );
305
+
306
+ //
307
+ // Globe
308
+ //
309
+
286
310
  export const Globe = {
287
311
  Root: GlobeRoot,
288
312
  Canvas: GlobeCanvas,
289
- Zoom: ({ onAction, position = 'bottomleft', ...props }: GlobeControlProps) => (
290
- <CustomControl position={position} {...props}>
291
- <ZoomControls onAction={onAction} />
292
- </CustomControl>
293
- ),
294
- Action: ({ onAction, position = 'bottomright', ...props }: GlobeControlProps) => (
295
- <CustomControl position={position} {...props}>
296
- <ActionControls onAction={onAction} />
297
- </CustomControl>
298
- ),
313
+ Zoom: GlobeZoom,
314
+ Action: GlobeAction,
299
315
  Debug: GlobeDebug,
300
316
  Panel: GlobePanel,
301
317
  };
@@ -2,45 +2,56 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
7
- import { type StoryObj, type Meta } from '@storybook/react-vite';
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
8
6
  import React, { useState } from 'react';
9
7
 
10
- import { withLayout, withTheme } from '@dxos/storybook-utils';
8
+ import { withTheme } from '@dxos/react-ui/testing';
11
9
 
12
- import { Map, type MapController } from './Map';
13
10
  import { useMapZoomHandler } from '../../hooks';
14
- import { type MapMarker } from '../../types';
11
+ import { type GeoMarker } from '../../types';
12
+
13
+ import { Map, type MapController } from './Map';
15
14
 
16
- const DefaultStory = ({ markers = [] }: { markers?: MapMarker[] }) => {
15
+ const DefaultStory = ({ markers = [] }: { markers?: GeoMarker[] }) => {
17
16
  const [controller, setController] = useState<MapController>();
18
17
  const handleZoomAction = useMapZoomHandler(controller);
19
18
 
20
19
  return (
21
- <Map.Root>
22
- <Map.Canvas ref={setController} markers={markers} />
20
+ <Map.Root ref={setController}>
21
+ <Map.Tiles />
22
+ <Map.Markers markers={markers} />
23
23
  <Map.Zoom position='bottomleft' onAction={handleZoomAction} />
24
24
  <Map.Action position='bottomright' />
25
25
  </Map.Root>
26
26
  );
27
27
  };
28
28
 
29
- const meta: Meta<typeof DefaultStory> = {
29
+ const meta = {
30
30
  title: 'ui/react-ui-geo/Map',
31
+ component: Map.Root as any,
31
32
  render: DefaultStory,
32
- decorators: [withTheme, withLayout({ fullscreen: true })],
33
- };
33
+ decorators: [withTheme],
34
+ parameters: {
35
+ layout: 'fullscreen',
36
+ },
37
+ } satisfies Meta<typeof DefaultStory>;
34
38
 
35
39
  export default meta;
36
40
 
37
- type Story = StoryObj<typeof DefaultStory>;
41
+ type Story = StoryObj<typeof meta>;
38
42
 
39
43
  export const Default: Story = {};
40
44
 
41
45
  export const WithMarkers: Story = {
42
46
  args: {
43
47
  markers: [
48
+ { id: 'los angeles', title: 'Los Angeles', location: { lat: 34.0522, lng: -118.2437 } },
49
+ { id: 'new york', title: 'New York', location: { lat: 40.7128, lng: -74.006 } },
50
+ { id: 'warsaw', title: 'Warsaw', location: { lat: 52.2297, lng: 21.0122 } },
51
+ { id: 'london', title: 'London', location: { lat: 51.5074, lng: -0.1278 } },
52
+ { id: 'toronto', title: 'Toronto', location: { lat: 43.6532, lng: -79.3832 } },
53
+ { id: 'seattle', title: 'Seattle', location: { lat: 47.6062, lng: -122.3321 } },
54
+ { id: 'barcelona', title: 'Barcelona', location: { lat: 41.3851, lng: 2.1734 } },
44
55
  { id: 'tokyo', title: 'Tokyo', location: { lat: 35.6762, lng: 139.6503 } },
45
56
  { id: 'sydney', title: 'Sydney', location: { lat: -33.8688, lng: 151.2093 } },
46
57
  { id: 'auckland', title: 'Auckland', location: { lat: -36.8509, lng: 174.7645 } },
@@ -56,6 +67,6 @@ export const WithMarkers: Story = {
56
67
  { id: 'phnom-penh', title: 'Phnom Penh', location: { lat: 11.5564, lng: 104.9282 } },
57
68
  { id: 'vientiane', title: 'Vientiane', location: { lat: 17.9757, lng: 102.6331 } },
58
69
  { id: 'yangon', title: 'Yangon', location: { lat: 16.8661, lng: 96.1951 } },
59
- ] as MapMarker[],
70
+ ] as GeoMarker[],
60
71
  },
61
72
  };