@dxos/react-ui-geo 0.8.4-main.b97322e → 0.8.4-main.bc674ce

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 (59) hide show
  1. package/dist/lib/browser/index.mjs +387 -442
  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 +387 -442
  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/Globe/Globe.d.ts +1 -1
  8. package/dist/types/src/components/Globe/Globe.d.ts.map +1 -1
  9. package/dist/types/src/components/Globe/Globe.stories.d.ts +25 -9
  10. package/dist/types/src/components/Globe/Globe.stories.d.ts.map +1 -1
  11. package/dist/types/src/components/Map/Map.d.ts +28 -18
  12. package/dist/types/src/components/Map/Map.d.ts.map +1 -1
  13. package/dist/types/src/components/Map/Map.stories.d.ts +14 -8
  14. package/dist/types/src/components/Map/Map.stories.d.ts.map +1 -1
  15. package/dist/types/src/components/Toolbar/Controls.d.ts.map +1 -1
  16. package/dist/types/src/components/index.d.ts +0 -1
  17. package/dist/types/src/components/index.d.ts.map +1 -1
  18. package/dist/types/src/hooks/context.d.ts +7 -7
  19. package/dist/types/src/hooks/context.d.ts.map +1 -1
  20. package/dist/types/src/hooks/useGlobeZoomHandler.d.ts +1 -1
  21. package/dist/types/src/hooks/useGlobeZoomHandler.d.ts.map +1 -1
  22. package/dist/types/src/hooks/useMapZoomHandler.d.ts +1 -1
  23. package/dist/types/src/hooks/useMapZoomHandler.d.ts.map +1 -1
  24. package/dist/types/src/hooks/useSpinner.d.ts +1 -1
  25. package/dist/types/src/hooks/useSpinner.d.ts.map +1 -1
  26. package/dist/types/src/hooks/useTour.d.ts +4 -3
  27. package/dist/types/src/hooks/useTour.d.ts.map +1 -1
  28. package/dist/types/src/index.d.ts +1 -0
  29. package/dist/types/src/index.d.ts.map +1 -1
  30. package/dist/types/src/translations.d.ts +12 -0
  31. package/dist/types/src/translations.d.ts.map +1 -0
  32. package/dist/types/src/types.d.ts +2 -1
  33. package/dist/types/src/types.d.ts.map +1 -1
  34. package/dist/types/src/util/path.d.ts +5 -8
  35. package/dist/types/src/util/path.d.ts.map +1 -1
  36. package/dist/types/src/util/render.d.ts +4 -4
  37. package/dist/types/src/util/render.d.ts.map +1 -1
  38. package/dist/types/tsconfig.tsbuildinfo +1 -1
  39. package/package.json +29 -23
  40. package/src/components/Globe/Globe.stories.tsx +82 -34
  41. package/src/components/Globe/Globe.tsx +56 -35
  42. package/src/components/Map/Map.stories.tsx +25 -14
  43. package/src/components/Map/Map.tsx +181 -96
  44. package/src/components/Toolbar/Controls.tsx +14 -20
  45. package/src/components/index.ts +0 -2
  46. package/src/hooks/context.tsx +22 -16
  47. package/src/hooks/useGlobeZoomHandler.ts +9 -3
  48. package/src/hooks/useMapZoomHandler.ts +1 -1
  49. package/src/hooks/useSpinner.ts +2 -1
  50. package/src/hooks/useTour.ts +10 -8
  51. package/src/index.ts +1 -0
  52. package/src/translations.ts +20 -0
  53. package/src/types.ts +3 -1
  54. package/src/util/inertia.ts +1 -1
  55. package/src/util/path.ts +5 -6
  56. package/src/util/render.ts +5 -3
  57. package/dist/types/src/components/types.d.ts +0 -15
  58. package/dist/types/src/components/types.d.ts.map +0 -1
  59. package/src/components/types.ts +0 -19
package/package.json CHANGED
@@ -1,20 +1,26 @@
1
1
  {
2
2
  "name": "@dxos/react-ui-geo",
3
- "version": "0.8.4-main.b97322e",
3
+ "version": "0.8.4-main.bc674ce",
4
4
  "description": "Geo components.",
5
5
  "homepage": "https://github.com/dxos",
6
6
  "bugs": "https://github.com/dxos/issues",
7
+ "repository": {
8
+ "type": "git",
9
+ "url": "https://github.com/dxos/dxos"
10
+ },
7
11
  "license": "MIT",
8
12
  "author": "DXOS.org",
9
13
  "sideEffects": true,
10
14
  "type": "module",
11
15
  "exports": {
12
16
  "./data": {
17
+ "source": "./src/data.ts",
13
18
  "types": "./dist/types/src/data.d.ts",
14
19
  "browser": "./dist/lib/browser/data.mjs",
15
20
  "node": "./dist/lib/node-esm/data.mjs"
16
21
  },
17
22
  ".": {
23
+ "source": "./src/index.ts",
18
24
  "types": "./dist/types/src/index.d.ts",
19
25
  "browser": "./dist/lib/browser/index.mjs",
20
26
  "node": "./dist/lib/node-esm/index.mjs"
@@ -34,51 +40,51 @@
34
40
  "src"
35
41
  ],
36
42
  "dependencies": {
37
- "@preact-signals/safe-react": "^0.9.0",
43
+ "@radix-ui/react-context": "1.1.1",
38
44
  "d3": "^7.9.0",
39
45
  "d3-geo-projection": "^4.0.0",
40
46
  "d3-hexbin": "^0.2.2",
41
47
  "geojson": "^0.5.0",
42
48
  "leaflet": "^1.9.4",
43
49
  "lodash.defaultsdeep": "^4.6.1",
44
- "react-leaflet": "^4.2.1",
50
+ "react-leaflet": "^5.0.0",
45
51
  "react-resize-detector": "^11.0.1",
46
52
  "topojson-client": "^3.1.0",
47
53
  "topojson-simplify": "^3.0.3",
48
54
  "versor": "^0.2.0",
49
- "@dxos/async": "0.8.4-main.b97322e",
50
- "@dxos/debug": "0.8.4-main.b97322e",
51
- "@dxos/log": "0.8.4-main.b97322e",
52
- "@dxos/node-std": "0.8.4-main.b97322e",
53
- "@dxos/util": "0.8.4-main.b97322e"
55
+ "@dxos/async": "0.8.4-main.bc674ce",
56
+ "@dxos/log": "0.8.4-main.bc674ce",
57
+ "@dxos/debug": "0.8.4-main.bc674ce",
58
+ "@dxos/node-std": "0.8.4-main.bc674ce",
59
+ "@dxos/util": "0.8.4-main.bc674ce"
54
60
  },
55
61
  "devDependencies": {
56
- "@react-three/drei": "^9.99.0",
57
- "@react-three/fiber": "^8.15.0",
62
+ "@react-three/drei": "^10.7.7",
63
+ "@react-three/fiber": "^9.5.0",
58
64
  "@types/d3": "^7.4.3",
59
65
  "@types/geojson": "^7946.0.14",
60
66
  "@types/leaflet": "^1.9.16",
61
- "@types/react": "~18.2.0",
62
- "@types/react-dom": "~18.2.0",
67
+ "@types/react": "~19.2.7",
68
+ "@types/react-dom": "~19.2.3",
63
69
  "@types/three": "0.165.0",
64
70
  "@types/topojson-client": "^3.1.4",
65
71
  "@types/topojson-simplify": "^3.0.3",
66
72
  "@types/topojson-specification": "^1.0.5",
67
73
  "JSONStream": "^1.3.5",
68
74
  "geojson2h3": "^1.2.0",
69
- "leva": "^0.9.35",
70
- "react": "~18.2.0",
71
- "react-dom": "~18.2.0",
72
- "three": "0.165.0",
73
- "@dxos/react-ui-theme": "0.8.4-main.b97322e",
74
- "@dxos/react-ui": "0.8.4-main.b97322e",
75
- "@dxos/storybook-utils": "0.8.4-main.b97322e"
75
+ "leva": "^0.10.1",
76
+ "react": "~19.2.3",
77
+ "react-dom": "~19.2.3",
78
+ "three": "^0.178.0",
79
+ "@dxos/react-ui": "0.8.4-main.bc674ce",
80
+ "@dxos/storybook-utils": "0.8.4-main.bc674ce",
81
+ "@dxos/ui-theme": "0.8.4-main.bc674ce"
76
82
  },
77
83
  "peerDependencies": {
78
- "react": "~18.2.0",
79
- "react-dom": "~18.2.0",
80
- "@dxos/react-ui": "0.8.4-main.b97322e",
81
- "@dxos/react-ui-theme": "0.8.4-main.b97322e"
84
+ "react": "~19.2.3",
85
+ "react-dom": "~19.2.3",
86
+ "@dxos/react-ui": "0.8.4-main.bc674ce",
87
+ "@dxos/ui-theme": "0.8.4-main.bc674ce"
82
88
  },
83
89
  "publishConfig": {
84
90
  "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}
@@ -210,7 +214,7 @@ const Story = ({
210
214
  <Globe.Zoom onAction={handleAction} />
211
215
  <Globe.Action onAction={handleAction} />
212
216
  <Globe.Debug />
213
- <Globe.Panel position='topright' classNames='w-20 h-20'>
217
+ <Globe.Panel position='topright' classNames='is-20 bs-20'>
214
218
  <Leva />
215
219
  </Globe.Panel>
216
220
  </Globe.Root>
@@ -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,8 +26,8 @@ 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';
30
- import { mx } from '@dxos/react-ui-theme';
29
+ import { type ThemeMode, type ThemedClassName, useDynamicRef, useThemeContext } from '@dxos/react-ui';
30
+ import { mx } from '@dxos/ui-theme';
31
31
 
32
32
  import {
33
33
  GlobeContextProvider,
@@ -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
 
@@ -130,6 +130,7 @@ type GlobeRootProps = PropsWithChildren<ThemedClassName<GlobeContextProviderProp
130
130
 
131
131
  const GlobeRoot = ({ classNames, children, ...props }: GlobeRootProps) => {
132
132
  const { ref, width, height } = useResizeDetector<HTMLDivElement>();
133
+
133
134
  return (
134
135
  <div ref={ref} className={mx('relative flex grow overflow-hidden', classNames)}>
135
136
  <GlobeContextProvider size={{ width, height }} {...props}>
@@ -154,17 +155,18 @@ type GlobeCanvasProps = {
154
155
  * Basic globe renderer.
155
156
  * https://github.com/topojson/world-atlas
156
157
  */
158
+ // TODO(burdon): Move controller to root.
157
159
  const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
158
- ({ projection: _projection, topology, features, styles: _styles }, forwardRef) => {
160
+ ({ projection: projectionProp, topology, features, styles: stylesProp }, forwardRef) => {
159
161
  const { themeMode } = useThemeContext();
160
- const styles = useMemo(() => _styles ?? defaultStyles[themeMode], [_styles, themeMode]);
162
+ const styles = useMemo(() => stylesProp ?? defaultStyles[themeMode], [stylesProp, themeMode]);
161
163
 
162
164
  // Canvas.
163
165
  const [canvas, setCanvas] = useState<HTMLCanvasElement>(null);
164
166
  const canvasRef = (canvas: HTMLCanvasElement) => setCanvas(canvas);
165
167
 
166
168
  // Projection.
167
- const projection = useMemo(() => getProjection(_projection), [_projection]);
169
+ const projection = useMemo(() => getProjection(projectionProp), [projectionProp]);
168
170
 
169
171
  // Layers.
170
172
  // TODO(burdon): Generate on the fly based on what is visible.
@@ -173,15 +175,14 @@ const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
173
175
  }, [topology, features, styles]);
174
176
 
175
177
  // State.
176
- const { size, center, scale, translation, rotation, setCenter, setScale, setTranslation, setRotation } =
178
+ const { size, center, zoom, translation, rotation, setCenter, setZoom, setTranslation, setRotation } =
177
179
  useGlobeContext();
178
-
179
- const scaleRef = useDynamicRef(scale);
180
+ const zoomRef = useDynamicRef(zoom);
180
181
 
181
182
  // Update rotation.
182
183
  useEffect(() => {
183
184
  if (center) {
184
- setScale(1);
185
+ setZoom(1);
185
186
  setRotation(positionToRotation(geoToPosition(center)));
186
187
  }
187
188
  }, [center]);
@@ -193,25 +194,25 @@ const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
193
194
  canvas,
194
195
  projection,
195
196
  center,
196
- get scale() {
197
- return scaleRef.current;
197
+ get zoom() {
198
+ return zoomRef.current;
198
199
  },
199
200
  translation,
200
201
  rotation,
201
202
  setCenter,
202
- setScale: (s) => {
203
+ setZoom: (s) => {
203
204
  if (typeof s === 'function') {
204
- const is = interpolateNumber(scaleRef.current, s(scaleRef.current));
205
+ const is = interpolateNumber(zoomRef.current, s(zoomRef.current));
205
206
  // Stop easing if already zooming.
206
207
  transition()
207
208
  .ease(zooming.current ? easeLinear : easeSinOut)
208
209
  .duration(200)
209
- .tween('scale', () => (t) => setScale(is(t)))
210
+ .tween('scale', () => (t) => setZoom(is(t)))
210
211
  .on('end', () => {
211
212
  zooming.current = false;
212
213
  });
213
214
  } else {
214
- setScale(s);
215
+ setZoom(s);
215
216
  }
216
217
  },
217
218
  setTranslation,
@@ -232,14 +233,14 @@ const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
232
233
  timer(() => {
233
234
  // https://d3js.org/d3-geo/projection
234
235
  projection
235
- .scale((Math.min(size.width, size.height) / 2) * scale)
236
+ .scale((Math.min(size.width, size.height) / 2) * zoom)
236
237
  .translate([size.width / 2 + (translation?.x ?? 0), size.height / 2 + (translation?.y ?? 0)])
237
238
  .rotate(rotation ?? [0, 0, 0]);
238
239
 
239
- renderLayers(generator, layers, scale, styles);
240
+ renderLayers(generator, layers, zoom, styles);
240
241
  });
241
242
  }
242
- }, [generator, size, scale, translation, rotation, layers]);
243
+ }, [generator, size, zoom, translation, rotation, layers]);
243
244
 
244
245
  if (!size.width || !size.height) {
245
246
  return null;
@@ -249,22 +250,30 @@ const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
249
250
  },
250
251
  );
251
252
 
253
+ //
254
+ // Debug
255
+ //
256
+
252
257
  const GlobeDebug = ({ position = 'topleft' }: { position?: ControlPosition }) => {
253
- const { size, scale, translation, rotation } = useGlobeContext();
258
+ const { size, zoom, translation, rotation } = useGlobeContext();
254
259
  return (
255
260
  <div
256
261
  className={mx(
257
- 'z-10 absolute w-96 p-2 overflow-hidden border border-green-700 rounded',
262
+ 'z-10 absolute is-96 p-2 overflow-hidden border border-green-700 rounded',
258
263
  controlPositions[position],
259
264
  )}
260
265
  >
261
266
  <pre className='font-mono text-xs text-green-700'>
262
- {JSON.stringify({ size, scale, translation, rotation }, null, 2)}
267
+ {JSON.stringify({ size, zoom, translation, rotation }, null, 2)}
263
268
  </pre>
264
269
  </div>
265
270
  );
266
271
  };
267
272
 
273
+ //
274
+ // Panel
275
+ //
276
+
268
277
  const GlobePanel = ({
269
278
  position,
270
279
  classNames,
@@ -273,25 +282,37 @@ const GlobePanel = ({
273
282
  return <div className={mx('z-10 absolute overflow-hidden', controlPositions[position], classNames)}>{children}</div>;
274
283
  };
275
284
 
285
+ //
286
+ // Controls
287
+ //
288
+
276
289
  const CustomControl = ({ position, children }: PropsWithChildren<{ position: ControlPosition }>) => {
277
290
  return <div className={mx('z-10 absolute overflow-hidden', controlPositions[position])}>{children}</div>;
278
291
  };
279
292
 
280
293
  type GlobeControlProps = { position?: ControlPosition } & Pick<ControlProps, 'onAction'>;
281
294
 
295
+ const GlobeZoom = ({ onAction, position = 'bottomleft', ...props }: GlobeControlProps) => (
296
+ <CustomControl position={position} {...props}>
297
+ <ZoomControls onAction={onAction} />
298
+ </CustomControl>
299
+ );
300
+
301
+ const GlobeAction = ({ onAction, position = 'bottomright', ...props }: GlobeControlProps) => (
302
+ <CustomControl position={position} {...props}>
303
+ <ActionControls onAction={onAction} />
304
+ </CustomControl>
305
+ );
306
+
307
+ //
308
+ // Globe
309
+ //
310
+
282
311
  export const Globe = {
283
312
  Root: GlobeRoot,
284
313
  Canvas: GlobeCanvas,
285
- Zoom: ({ onAction, position = 'bottomleft', ...props }: GlobeControlProps) => (
286
- <CustomControl position={position} {...props}>
287
- <ZoomControls onAction={onAction} />
288
- </CustomControl>
289
- ),
290
- Action: ({ onAction, position = 'bottomright', ...props }: GlobeControlProps) => (
291
- <CustomControl position={position} {...props}>
292
- <ActionControls onAction={onAction} />
293
- </CustomControl>
294
- ),
314
+ Zoom: GlobeZoom,
315
+ Action: GlobeAction,
295
316
  Debug: GlobeDebug,
296
317
  Panel: GlobePanel,
297
318
  };
@@ -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
  };