@dxos/react-ui-geo 0.8.4-main.dedc0f3 → 0.8.4-main.e8ec1fe
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/lib/browser/index.mjs +71 -62
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/node-esm/index.mjs +71 -62
- package/dist/lib/node-esm/index.mjs.map +4 -4
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/types/src/components/Globe/Globe.d.ts.map +1 -1
- package/dist/types/src/components/Globe/Globe.stories.d.ts +3 -1
- package/dist/types/src/components/Globe/Globe.stories.d.ts.map +1 -1
- package/dist/types/src/components/Map/Map.d.ts +6 -9
- package/dist/types/src/components/Map/Map.d.ts.map +1 -1
- package/dist/types/src/components/Map/Map.stories.d.ts +3 -1
- package/dist/types/src/components/Map/Map.stories.d.ts.map +1 -1
- package/dist/types/src/components/Toolbar/Controls.d.ts.map +1 -1
- package/dist/types/src/hooks/context.d.ts +1 -1
- package/dist/types/src/hooks/context.d.ts.map +1 -1
- package/dist/types/src/hooks/useGlobeZoomHandler.d.ts.map +1 -1
- package/dist/types/src/hooks/useTour.d.ts.map +1 -1
- package/dist/types/src/index.d.ts +1 -0
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/translations.d.ts +12 -0
- package/dist/types/src/translations.d.ts.map +1 -0
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +20 -20
- package/src/components/Globe/Globe.stories.tsx +6 -5
- package/src/components/Globe/Globe.tsx +5 -4
- package/src/components/Map/Map.stories.tsx +5 -4
- package/src/components/Map/Map.tsx +18 -48
- package/src/components/Toolbar/Controls.tsx +12 -14
- package/src/hooks/context.tsx +14 -8
- package/src/hooks/useGlobeZoomHandler.ts +8 -2
- package/src/hooks/useTour.ts +1 -0
- package/src/index.ts +1 -0
- package/src/translations.ts +20 -0
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dxos/react-ui-geo",
|
|
3
|
-
"version": "0.8.4-main.
|
|
3
|
+
"version": "0.8.4-main.e8ec1fe",
|
|
4
4
|
"description": "Geo components.",
|
|
5
5
|
"homepage": "https://github.com/dxos",
|
|
6
6
|
"bugs": "https://github.com/dxos/issues",
|
|
@@ -37,32 +37,32 @@
|
|
|
37
37
|
],
|
|
38
38
|
"dependencies": {
|
|
39
39
|
"@preact-signals/safe-react": "^0.9.0",
|
|
40
|
-
"@radix-ui/react-context": "^1.
|
|
40
|
+
"@radix-ui/react-context": "^1.1.1",
|
|
41
41
|
"d3": "^7.9.0",
|
|
42
42
|
"d3-geo-projection": "^4.0.0",
|
|
43
43
|
"d3-hexbin": "^0.2.2",
|
|
44
44
|
"geojson": "^0.5.0",
|
|
45
45
|
"leaflet": "^1.9.4",
|
|
46
46
|
"lodash.defaultsdeep": "^4.6.1",
|
|
47
|
-
"react-leaflet": "^
|
|
47
|
+
"react-leaflet": "^5.0.0",
|
|
48
48
|
"react-resize-detector": "^11.0.1",
|
|
49
49
|
"topojson-client": "^3.1.0",
|
|
50
50
|
"topojson-simplify": "^3.0.3",
|
|
51
51
|
"versor": "^0.2.0",
|
|
52
|
-
"@dxos/
|
|
53
|
-
"@dxos/
|
|
54
|
-
"@dxos/
|
|
55
|
-
"@dxos/
|
|
56
|
-
"@dxos/
|
|
52
|
+
"@dxos/debug": "0.8.4-main.e8ec1fe",
|
|
53
|
+
"@dxos/async": "0.8.4-main.e8ec1fe",
|
|
54
|
+
"@dxos/node-std": "0.8.4-main.e8ec1fe",
|
|
55
|
+
"@dxos/util": "0.8.4-main.e8ec1fe",
|
|
56
|
+
"@dxos/log": "0.8.4-main.e8ec1fe"
|
|
57
57
|
},
|
|
58
58
|
"devDependencies": {
|
|
59
59
|
"@react-three/drei": "^9.99.0",
|
|
60
|
-
"@react-three/fiber": "^
|
|
60
|
+
"@react-three/fiber": "^9.3.0",
|
|
61
61
|
"@types/d3": "^7.4.3",
|
|
62
62
|
"@types/geojson": "^7946.0.14",
|
|
63
63
|
"@types/leaflet": "^1.9.16",
|
|
64
|
-
"@types/react": "~
|
|
65
|
-
"@types/react-dom": "~
|
|
64
|
+
"@types/react": "~19.2.2",
|
|
65
|
+
"@types/react-dom": "~19.2.2",
|
|
66
66
|
"@types/three": "0.165.0",
|
|
67
67
|
"@types/topojson-client": "^3.1.4",
|
|
68
68
|
"@types/topojson-simplify": "^3.0.3",
|
|
@@ -70,18 +70,18 @@
|
|
|
70
70
|
"JSONStream": "^1.3.5",
|
|
71
71
|
"geojson2h3": "^1.2.0",
|
|
72
72
|
"leva": "^0.9.35",
|
|
73
|
-
"react": "~
|
|
74
|
-
"react-dom": "~
|
|
73
|
+
"react": "~19.2.0",
|
|
74
|
+
"react-dom": "~19.2.0",
|
|
75
75
|
"three": "0.165.0",
|
|
76
|
-
"@dxos/react-ui": "0.8.4-main.
|
|
77
|
-
"@dxos/
|
|
78
|
-
"@dxos/
|
|
76
|
+
"@dxos/react-ui": "0.8.4-main.e8ec1fe",
|
|
77
|
+
"@dxos/storybook-utils": "0.8.4-main.e8ec1fe",
|
|
78
|
+
"@dxos/react-ui-theme": "0.8.4-main.e8ec1fe"
|
|
79
79
|
},
|
|
80
80
|
"peerDependencies": {
|
|
81
|
-
"react": "
|
|
82
|
-
"react-dom": "
|
|
83
|
-
"@dxos/react-ui": "0.8.4-main.
|
|
84
|
-
"@dxos/react-ui-theme": "0.8.4-main.
|
|
81
|
+
"react": "^19.0.0",
|
|
82
|
+
"react-dom": "^19.0.0",
|
|
83
|
+
"@dxos/react-ui": "0.8.4-main.e8ec1fe",
|
|
84
|
+
"@dxos/react-ui-theme": "0.8.4-main.e8ec1fe"
|
|
85
85
|
},
|
|
86
86
|
"publishConfig": {
|
|
87
87
|
"access": "public"
|
|
@@ -2,8 +2,6 @@
|
|
|
2
2
|
// Copyright 2018 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import '@dxos-theme';
|
|
6
|
-
|
|
7
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';
|
|
@@ -11,7 +9,7 @@ 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 {
|
|
12
|
+
import { withTheme } from '@dxos/react-ui/testing';
|
|
15
13
|
|
|
16
14
|
import { type Vector, useDrag, useGlobeZoomHandler, useSpinner, useTour } from '../../hooks';
|
|
17
15
|
import { type LatLngLiteral } from '../../types';
|
|
@@ -216,7 +214,7 @@ const DefaultStory = ({
|
|
|
216
214
|
<Globe.Zoom onAction={handleAction} />
|
|
217
215
|
<Globe.Action onAction={handleAction} />
|
|
218
216
|
<Globe.Debug />
|
|
219
|
-
<Globe.Panel position='topright' classNames='
|
|
217
|
+
<Globe.Panel position='topright' classNames='is-20 bs-20'>
|
|
220
218
|
<Leva />
|
|
221
219
|
</Globe.Panel>
|
|
222
220
|
</Globe.Root>
|
|
@@ -229,7 +227,10 @@ const meta = {
|
|
|
229
227
|
title: 'ui/react-ui-geo/Globe',
|
|
230
228
|
component: Globe.Root,
|
|
231
229
|
render: DefaultStory,
|
|
232
|
-
decorators: [withTheme
|
|
230
|
+
decorators: [withTheme],
|
|
231
|
+
parameters: {
|
|
232
|
+
layout: 'fullscreen',
|
|
233
|
+
},
|
|
233
234
|
} satisfies Meta;
|
|
234
235
|
|
|
235
236
|
export default meta;
|
|
@@ -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}>
|
|
@@ -156,16 +157,16 @@ type GlobeCanvasProps = {
|
|
|
156
157
|
*/
|
|
157
158
|
// TODO(burdon): Move controller to root.
|
|
158
159
|
const GlobeCanvas = forwardRef<GlobeController, GlobeCanvasProps>(
|
|
159
|
-
({ projection:
|
|
160
|
+
({ projection: projectionParam, topology, features, styles: stylesParam }, forwardRef) => {
|
|
160
161
|
const { themeMode } = useThemeContext();
|
|
161
|
-
const styles = useMemo(() =>
|
|
162
|
+
const styles = useMemo(() => stylesParam ?? defaultStyles[themeMode], [stylesParam, themeMode]);
|
|
162
163
|
|
|
163
164
|
// Canvas.
|
|
164
165
|
const [canvas, setCanvas] = useState<HTMLCanvasElement>(null);
|
|
165
166
|
const canvasRef = (canvas: HTMLCanvasElement) => setCanvas(canvas);
|
|
166
167
|
|
|
167
168
|
// Projection.
|
|
168
|
-
const projection = useMemo(() => getProjection(
|
|
169
|
+
const projection = useMemo(() => getProjection(projectionParam), [projectionParam]);
|
|
169
170
|
|
|
170
171
|
// Layers.
|
|
171
172
|
// TODO(burdon): Generate on the fly based on what is visible.
|
|
@@ -258,7 +259,7 @@ const GlobeDebug = ({ position = 'topleft' }: { position?: ControlPosition }) =>
|
|
|
258
259
|
return (
|
|
259
260
|
<div
|
|
260
261
|
className={mx(
|
|
261
|
-
'z-10 absolute
|
|
262
|
+
'z-10 absolute is-96 p-2 overflow-hidden border border-green-700 rounded',
|
|
262
263
|
controlPositions[position],
|
|
263
264
|
)}
|
|
264
265
|
>
|
|
@@ -2,12 +2,10 @@
|
|
|
2
2
|
// Copyright 2024 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import '@dxos-theme';
|
|
6
|
-
|
|
7
5
|
import { type Meta, type StoryObj } from '@storybook/react-vite';
|
|
8
6
|
import React, { useState } from 'react';
|
|
9
7
|
|
|
10
|
-
import {
|
|
8
|
+
import { withTheme } from '@dxos/react-ui/testing';
|
|
11
9
|
|
|
12
10
|
import { useMapZoomHandler } from '../../hooks';
|
|
13
11
|
import { type GeoMarker } from '../../types';
|
|
@@ -32,7 +30,10 @@ const meta = {
|
|
|
32
30
|
title: 'ui/react-ui-geo/Map',
|
|
33
31
|
component: Map.Root as any,
|
|
34
32
|
render: DefaultStory,
|
|
35
|
-
decorators: [withTheme
|
|
33
|
+
decorators: [withTheme],
|
|
34
|
+
parameters: {
|
|
35
|
+
layout: 'fullscreen',
|
|
36
|
+
},
|
|
36
37
|
} satisfies Meta<typeof DefaultStory>;
|
|
37
38
|
|
|
38
39
|
export default meta;
|
|
@@ -8,10 +8,8 @@ import { createContext } from '@radix-ui/react-context';
|
|
|
8
8
|
import L, { Control, type ControlPosition, DomEvent, DomUtil, type LatLngLiteral, latLngBounds } from 'leaflet';
|
|
9
9
|
import React, { type PropsWithChildren, forwardRef, useEffect, useImperativeHandle, useRef, useState } from 'react';
|
|
10
10
|
import { createRoot } from 'react-dom/client';
|
|
11
|
-
import type
|
|
12
|
-
import { MapContainer, Marker, Popup, TileLayer, useMap } from 'react-leaflet';
|
|
11
|
+
import { MapContainer, type MapContainerProps, Marker, Popup, TileLayer, useMap, useMapEvents } from 'react-leaflet';
|
|
13
12
|
|
|
14
|
-
import { debounce } from '@dxos/async';
|
|
15
13
|
import { ThemeProvider, type ThemedClassName, Tooltip } from '@dxos/react-ui';
|
|
16
14
|
import { defaultTx, mx } from '@dxos/react-ui-theme';
|
|
17
15
|
|
|
@@ -25,7 +23,7 @@ import { ActionControls, type ControlProps, ZoomControls, controlPositions } fro
|
|
|
25
23
|
const defaults = {
|
|
26
24
|
center: { lat: 51, lng: 0 } as L.LatLngLiteral,
|
|
27
25
|
zoom: 4,
|
|
28
|
-
};
|
|
26
|
+
} as const;
|
|
29
27
|
|
|
30
28
|
//
|
|
31
29
|
// Controller
|
|
@@ -42,6 +40,7 @@ type MapController = {
|
|
|
42
40
|
|
|
43
41
|
type MapContextValue = {
|
|
44
42
|
attention?: boolean;
|
|
43
|
+
onChange?: (ev: { center: LatLngLiteral; zoom: number }) => void;
|
|
45
44
|
};
|
|
46
45
|
|
|
47
46
|
const [MapContextProvier, useMapContext] = createContext<MapContextValue>('Map');
|
|
@@ -50,27 +49,14 @@ const [MapContextProvier, useMapContext] = createContext<MapContextValue>('Map')
|
|
|
50
49
|
// Root
|
|
51
50
|
//
|
|
52
51
|
|
|
53
|
-
type MapRootProps = ThemedClassName<
|
|
54
|
-
MapContainerProps & {
|
|
55
|
-
onChange?: (ev: { center: LatLngLiteral; zoom: number }) => void;
|
|
56
|
-
}
|
|
57
|
-
>;
|
|
52
|
+
type MapRootProps = ThemedClassName<MapContainerProps & Pick<MapContextValue, 'onChange'>>;
|
|
58
53
|
|
|
59
54
|
/**
|
|
60
55
|
* https://react-leaflet.js.org/docs/api-map
|
|
61
56
|
*/
|
|
62
57
|
const MapRoot = forwardRef<MapController, MapRootProps>(
|
|
63
58
|
(
|
|
64
|
-
{
|
|
65
|
-
classNames,
|
|
66
|
-
scrollWheelZoom = true,
|
|
67
|
-
doubleClickZoom = true,
|
|
68
|
-
touchZoom = true,
|
|
69
|
-
center = defaults.center,
|
|
70
|
-
zoom = defaults.zoom,
|
|
71
|
-
onChange,
|
|
72
|
-
...props
|
|
73
|
-
},
|
|
59
|
+
{ classNames, scrollWheelZoom = true, doubleClickZoom = true, touchZoom = true, center, zoom, onChange, ...props },
|
|
74
60
|
forwardedRef,
|
|
75
61
|
) => {
|
|
76
62
|
const [attention, setAttention] = useState(false);
|
|
@@ -90,32 +76,6 @@ const MapRoot = forwardRef<MapController, MapRootProps>(
|
|
|
90
76
|
[],
|
|
91
77
|
);
|
|
92
78
|
|
|
93
|
-
// Events.
|
|
94
|
-
useEffect(() => {
|
|
95
|
-
if (!map) {
|
|
96
|
-
return;
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
const handler = debounce(() => {
|
|
100
|
-
setAttention(true);
|
|
101
|
-
onChange?.({
|
|
102
|
-
center: map.getCenter(),
|
|
103
|
-
zoom: map.getZoom(),
|
|
104
|
-
});
|
|
105
|
-
}, 100);
|
|
106
|
-
|
|
107
|
-
map.on('move', handler);
|
|
108
|
-
map.on('zoom', handler);
|
|
109
|
-
map.on('focus', () => setAttention(true));
|
|
110
|
-
map.on('blur', () => setAttention(false));
|
|
111
|
-
return () => {
|
|
112
|
-
map.off('move');
|
|
113
|
-
map.off('zoom');
|
|
114
|
-
map.off('focus');
|
|
115
|
-
map.off('blur');
|
|
116
|
-
};
|
|
117
|
-
}, [map, onChange]);
|
|
118
|
-
|
|
119
79
|
// Enable/disable scroll wheel zoom.
|
|
120
80
|
// TODO(burdon): Use attention:
|
|
121
81
|
// const {hasAttention} = useAttention(props.id);
|
|
@@ -132,7 +92,7 @@ const MapRoot = forwardRef<MapController, MapRootProps>(
|
|
|
132
92
|
}, [map, attention]);
|
|
133
93
|
|
|
134
94
|
return (
|
|
135
|
-
<MapContextProvier attention={attention}>
|
|
95
|
+
<MapContextProvier attention={attention} onChange={onChange}>
|
|
136
96
|
<MapContainer
|
|
137
97
|
{...props}
|
|
138
98
|
ref={mapRef}
|
|
@@ -142,8 +102,8 @@ const MapRoot = forwardRef<MapController, MapRootProps>(
|
|
|
142
102
|
scrollWheelZoom={scrollWheelZoom}
|
|
143
103
|
doubleClickZoom={doubleClickZoom}
|
|
144
104
|
touchZoom={touchZoom}
|
|
145
|
-
center={center}
|
|
146
|
-
zoom={zoom}
|
|
105
|
+
center={center ?? defaults.center}
|
|
106
|
+
zoom={zoom ?? defaults.zoom}
|
|
147
107
|
// whenReady={() => {}}
|
|
148
108
|
/>
|
|
149
109
|
</MapContextProvier>
|
|
@@ -162,6 +122,16 @@ type MapTilesProps = {};
|
|
|
162
122
|
|
|
163
123
|
const MapTiles = (_props: MapTilesProps) => {
|
|
164
124
|
const ref = useRef<L.TileLayer>(null);
|
|
125
|
+
const { onChange } = useMapContext(MapTiles.displayName);
|
|
126
|
+
|
|
127
|
+
useMapEvents({
|
|
128
|
+
zoomstart: (ev) => {
|
|
129
|
+
onChange?.({
|
|
130
|
+
center: ev.target.getCenter(),
|
|
131
|
+
zoom: ev.target.getZoom(),
|
|
132
|
+
});
|
|
133
|
+
},
|
|
134
|
+
});
|
|
165
135
|
|
|
166
136
|
// NOTE: Need to dynamically update data attribute since TileLayer doesn't update, but
|
|
167
137
|
// Tailwind requires setting the property for static analysis.
|
|
@@ -5,7 +5,9 @@
|
|
|
5
5
|
import { type ControlPosition } from 'leaflet';
|
|
6
6
|
import React from 'react';
|
|
7
7
|
|
|
8
|
-
import { IconButton, type ThemedClassName, Toolbar } from '@dxos/react-ui';
|
|
8
|
+
import { IconButton, type ThemedClassName, Toolbar, useTranslation } from '@dxos/react-ui';
|
|
9
|
+
|
|
10
|
+
import { translationKey } from '../../translations';
|
|
9
11
|
|
|
10
12
|
export type ControlAction = 'toggle' | 'start' | 'zoom-in' | 'zoom-out';
|
|
11
13
|
|
|
@@ -21,22 +23,20 @@ export const controlPositions: Record<ControlPosition, string> = {
|
|
|
21
23
|
};
|
|
22
24
|
|
|
23
25
|
export const ZoomControls = ({ classNames, onAction }: ControlProps) => {
|
|
26
|
+
const { t } = useTranslation(translationKey);
|
|
27
|
+
|
|
24
28
|
return (
|
|
25
29
|
<Toolbar.Root classNames={['gap-2', classNames]}>
|
|
26
30
|
<IconButton
|
|
27
31
|
icon='ph--plus--regular'
|
|
28
|
-
label='zoom in'
|
|
29
32
|
iconOnly
|
|
30
|
-
|
|
31
|
-
classNames='px-0 aspect-square'
|
|
33
|
+
label={t('zoom in icon button')}
|
|
32
34
|
onClick={() => onAction?.('zoom-in')}
|
|
33
35
|
/>
|
|
34
36
|
<IconButton
|
|
35
37
|
icon='ph--minus--regular'
|
|
36
|
-
label='zoom out'
|
|
37
38
|
iconOnly
|
|
38
|
-
|
|
39
|
-
classNames='px-0 aspect-square'
|
|
39
|
+
label={t('zoom out icon button')}
|
|
40
40
|
onClick={() => onAction?.('zoom-out')}
|
|
41
41
|
/>
|
|
42
42
|
</Toolbar.Root>
|
|
@@ -44,22 +44,20 @@ export const ZoomControls = ({ classNames, onAction }: ControlProps) => {
|
|
|
44
44
|
};
|
|
45
45
|
|
|
46
46
|
export const ActionControls = ({ classNames, onAction }: ControlProps) => {
|
|
47
|
+
const { t } = useTranslation(translationKey);
|
|
48
|
+
|
|
47
49
|
return (
|
|
48
50
|
<Toolbar.Root classNames={['gap-2', classNames]}>
|
|
49
51
|
<IconButton
|
|
50
|
-
icon='ph--
|
|
51
|
-
label='start'
|
|
52
|
+
icon='ph--path--regular'
|
|
52
53
|
iconOnly
|
|
53
|
-
|
|
54
|
-
classNames='px-0 aspect-square'
|
|
54
|
+
label={t('start icon button')}
|
|
55
55
|
onClick={() => onAction?.('start')}
|
|
56
56
|
/>
|
|
57
57
|
<IconButton
|
|
58
58
|
icon='ph--globe-hemisphere-west--regular'
|
|
59
|
-
label='toggle'
|
|
60
59
|
iconOnly
|
|
61
|
-
|
|
62
|
-
classNames='px-0 aspect-square'
|
|
60
|
+
label={t('toggle icon button')}
|
|
63
61
|
onClick={() => onAction?.('toggle')}
|
|
64
62
|
/>
|
|
65
63
|
</Toolbar.Root>
|
package/src/hooks/context.tsx
CHANGED
|
@@ -26,6 +26,12 @@ export type GlobeContextType = {
|
|
|
26
26
|
setRotation: Dispatch<SetStateAction<Vector>>;
|
|
27
27
|
};
|
|
28
28
|
|
|
29
|
+
const defaults = {
|
|
30
|
+
center: { lat: 51, lng: 0 } as LatLngLiteral,
|
|
31
|
+
zoom: 4,
|
|
32
|
+
} as const;
|
|
33
|
+
|
|
34
|
+
// TODO(burdon): Replace with radix.
|
|
29
35
|
const GlobeContext = createContext<GlobeContextType>(undefined);
|
|
30
36
|
|
|
31
37
|
export type GlobeContextProviderProps = PropsWithChildren<
|
|
@@ -35,15 +41,15 @@ export type GlobeContextProviderProps = PropsWithChildren<
|
|
|
35
41
|
export const GlobeContextProvider = ({
|
|
36
42
|
children,
|
|
37
43
|
size,
|
|
38
|
-
center:
|
|
39
|
-
zoom:
|
|
40
|
-
translation:
|
|
41
|
-
rotation:
|
|
44
|
+
center: centerParam = defaults.center,
|
|
45
|
+
zoom: zoomParam = defaults.zoom,
|
|
46
|
+
translation: translationParam,
|
|
47
|
+
rotation: rotationParam,
|
|
42
48
|
}: GlobeContextProviderProps) => {
|
|
43
|
-
const [center, setCenter] = useControlledState(
|
|
44
|
-
const [zoom, setZoom] = useControlledState(
|
|
45
|
-
const [translation, setTranslation] = useControlledState<Point>(
|
|
46
|
-
const [rotation, setRotation] = useControlledState<Vector>(
|
|
49
|
+
const [center, setCenter] = useControlledState(centerParam);
|
|
50
|
+
const [zoom, setZoom] = useControlledState(zoomParam);
|
|
51
|
+
const [translation, setTranslation] = useControlledState<Point>(translationParam);
|
|
52
|
+
const [rotation, setRotation] = useControlledState<Vector>(rotationParam);
|
|
47
53
|
|
|
48
54
|
return (
|
|
49
55
|
<GlobeContext.Provider
|
|
@@ -6,6 +6,8 @@ import { useCallback } from 'react';
|
|
|
6
6
|
|
|
7
7
|
import { type ControlProps, type GlobeController } from '../components';
|
|
8
8
|
|
|
9
|
+
const ZOOM_FACTOR = 0.1;
|
|
10
|
+
|
|
9
11
|
export const useGlobeZoomHandler = (controller: GlobeController | null | undefined): ControlProps['onAction'] => {
|
|
10
12
|
return useCallback<ControlProps['onAction']>(
|
|
11
13
|
(event) => {
|
|
@@ -15,11 +17,15 @@ export const useGlobeZoomHandler = (controller: GlobeController | null | undefin
|
|
|
15
17
|
|
|
16
18
|
switch (event) {
|
|
17
19
|
case 'zoom-in': {
|
|
18
|
-
controller.setZoom((zoom) =>
|
|
20
|
+
controller.setZoom((zoom) => {
|
|
21
|
+
return zoom * (1 + ZOOM_FACTOR);
|
|
22
|
+
});
|
|
19
23
|
break;
|
|
20
24
|
}
|
|
21
25
|
case 'zoom-out': {
|
|
22
|
-
controller.setZoom((zoom) =>
|
|
26
|
+
controller.setZoom((zoom) => {
|
|
27
|
+
return zoom * (1 - ZOOM_FACTOR);
|
|
28
|
+
});
|
|
23
29
|
break;
|
|
24
30
|
}
|
|
25
31
|
}
|
package/src/hooks/useTour.ts
CHANGED
|
@@ -34,6 +34,7 @@ export const useTour = (
|
|
|
34
34
|
options: TourOptions = {},
|
|
35
35
|
): [boolean, Dispatch<SetStateAction<boolean>>] => {
|
|
36
36
|
const selection = useMemo(() => d3Selection(), []);
|
|
37
|
+
// TODO(burdon): Redo controlled state.
|
|
37
38
|
const [running, setRunning] = useState(options.running ?? false);
|
|
38
39
|
useEffect(() => {
|
|
39
40
|
if (!running) {
|
package/src/index.ts
CHANGED
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2023 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { type Resource } from '@dxos/react-ui';
|
|
6
|
+
|
|
7
|
+
export const translationKey = 'react-ui-geo';
|
|
8
|
+
|
|
9
|
+
export const translations = [
|
|
10
|
+
{
|
|
11
|
+
'en-US': {
|
|
12
|
+
[translationKey]: {
|
|
13
|
+
'zoom in icon button': 'Zoom in',
|
|
14
|
+
'zoom out icon button': 'Zoom out',
|
|
15
|
+
'start icon button': 'Start',
|
|
16
|
+
'toggle icon button': 'Toggle',
|
|
17
|
+
},
|
|
18
|
+
},
|
|
19
|
+
},
|
|
20
|
+
] as const satisfies Resource[];
|