expo-gaode-map 2.2.31-next.0 → 2.2.31

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 (56) hide show
  1. package/android/src/main/java/expo/modules/gaodemap/ExpoGaodeMapView.kt +2 -2
  2. package/android/src/main/java/expo/modules/gaodemap/ExpoGaodeMapViewModule.kt +4 -14
  3. package/android/src/main/java/expo/modules/gaodemap/managers/UIManager.kt +5 -5
  4. package/android/src/main/java/expo/modules/gaodemap/overlays/MarkerBitmapRenderer.kt +29 -1
  5. package/android/src/main/java/expo/modules/gaodemap/overlays/MarkerViewModule.kt +3 -3
  6. package/build/ExpoGaodeMapModule.d.ts.map +1 -1
  7. package/build/ExpoGaodeMapModule.js +9 -26
  8. package/build/ExpoGaodeMapModule.js.map +1 -1
  9. package/build/ExpoGaodeMapView.d.ts.map +1 -1
  10. package/build/ExpoGaodeMapView.js +12 -0
  11. package/build/ExpoGaodeMapView.js.map +1 -1
  12. package/build/components/AreaMaskOverlay.d.ts +5 -0
  13. package/build/components/AreaMaskOverlay.d.ts.map +1 -0
  14. package/build/components/AreaMaskOverlay.js +20 -0
  15. package/build/components/AreaMaskOverlay.js.map +1 -0
  16. package/build/components/RouteOverlay.d.ts +5 -0
  17. package/build/components/RouteOverlay.d.ts.map +1 -0
  18. package/build/components/RouteOverlay.js +20 -0
  19. package/build/components/RouteOverlay.js.map +1 -0
  20. package/build/components/overlays/Marker.d.ts.map +1 -1
  21. package/build/components/overlays/Marker.js +65 -2
  22. package/build/components/overlays/Marker.js.map +1 -1
  23. package/build/hooks/useRoutePlayback.d.ts +4 -0
  24. package/build/hooks/useRoutePlayback.d.ts.map +1 -0
  25. package/build/hooks/useRoutePlayback.js +310 -0
  26. package/build/hooks/useRoutePlayback.js.map +1 -0
  27. package/build/index.d.ts +4 -0
  28. package/build/index.d.ts.map +1 -1
  29. package/build/index.js +4 -0
  30. package/build/index.js.map +1 -1
  31. package/build/types/common.types.d.ts +5 -5
  32. package/build/types/common.types.js +5 -5
  33. package/build/types/common.types.js.map +1 -1
  34. package/build/types/index.d.ts +1 -0
  35. package/build/types/index.d.ts.map +1 -1
  36. package/build/types/index.js.map +1 -1
  37. package/build/types/map-view.types.d.ts +6 -0
  38. package/build/types/map-view.types.d.ts.map +1 -1
  39. package/build/types/map-view.types.js.map +1 -1
  40. package/build/types/overlays.types.d.ts +9 -0
  41. package/build/types/overlays.types.d.ts.map +1 -1
  42. package/build/types/overlays.types.js.map +1 -1
  43. package/build/types/route-playback.types.d.ts +118 -0
  44. package/build/types/route-playback.types.d.ts.map +1 -0
  45. package/build/types/route-playback.types.js +2 -0
  46. package/build/types/route-playback.types.js.map +1 -0
  47. package/build/utils/RouteUtils.d.ts +8 -0
  48. package/build/utils/RouteUtils.d.ts.map +1 -0
  49. package/build/utils/RouteUtils.js +140 -0
  50. package/build/utils/RouteUtils.js.map +1 -0
  51. package/ios/ExpoGaodeMapView.swift +10 -7
  52. package/ios/ExpoGaodeMapViewModule.swift +2 -10
  53. package/ios/managers/UIManager.swift +5 -4
  54. package/ios/overlays/MarkerView.swift +38 -3
  55. package/ios/overlays/MarkerViewModule.swift +1 -1
  56. package/package.json +13 -2
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AreaMaskOverlay.js","sourceRoot":"","sources":["../../src/components/AreaMaskOverlay.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAG/B,OAAO,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACxD,OAAO,EAAE,sBAAsB,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAE,OAAO,EAAE,MAAM,YAAY,CAAC;AAErC,MAAM,UAAU,eAAe,CAAC,EAAE,KAAK,EAAE,YAAY,EAAwB;IAC3E,kCAAkC;IAClC,6BAA6B;IAC7B,MAAM,gBAAgB,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE;QAC1C,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,OAAO,sBAAsB,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC;QAC7C,CAAC;QAED,OAAO,mBAAmB,CAAC,KAAK,CAAC,CAAC;IACpC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAEZ,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE,CAAC;QAC7B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,OAAO,CACL,CAAC,OAAO,CACN,MAAM,CAAC,CAAC,gBAAgB,CAAC,CACzB,SAAS,CAAC,CAAC,YAAY,EAAE,SAAS,IAAI,wBAAwB,CAAC,CAC/D,WAAW,CAAC,CAAC,YAAY,EAAE,WAAW,IAAI,uBAAuB,CAAC,CAClE,WAAW,CAAC,CAAC,YAAY,EAAE,WAAW,IAAI,CAAC,CAAC,CAC5C,IAAI,YAAY,CAAC,EACjB,CACH,CAAC;AACJ,CAAC;AAED,eAAe,eAAe,CAAC","sourcesContent":["import * as React from 'react';\n\nimport type { AreaMaskOverlayProps } from '../types/route-playback.types';\nimport { normalizeLatLngList } from '../utils/GeoUtils';\nimport { parseMultiRingPolyline } from '../utils/RouteUtils';\nimport { Polygon } from './overlays';\n\nexport function AreaMaskOverlay({ rings, polygonProps }: AreaMaskOverlayProps) {\n // 支持直接传多环坐标,或传高德返回的 polyline 字符串。\n // 最终都落到 Polygon 的“带孔多边形”能力上。\n const normalizedPoints = React.useMemo(() => {\n if (typeof rings === 'string') {\n return parseMultiRingPolyline(rings).rings;\n }\n\n return normalizeLatLngList(rings);\n }, [rings]);\n\n if (!normalizedPoints.length) {\n return null;\n }\n\n return (\n <Polygon\n points={normalizedPoints}\n fillColor={polygonProps?.fillColor ?? 'rgba(15, 23, 42, 0.45)'}\n strokeColor={polygonProps?.strokeColor ?? 'rgba(15, 23, 42, 0.8)'}\n strokeWidth={polygonProps?.strokeWidth ?? 1}\n {...polygonProps}\n />\n );\n}\n\nexport default AreaMaskOverlay;\n"]}
@@ -0,0 +1,5 @@
1
+ import * as React from 'react';
2
+ import type { RouteOverlayProps } from '../types/route-playback.types';
3
+ export declare function RouteOverlay({ points, showStartMarker, showEndMarker, polylineProps, startMarkerProps, endMarkerProps, }: RouteOverlayProps): React.JSX.Element | null;
4
+ export default RouteOverlay;
5
+ //# sourceMappingURL=RouteOverlay.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"RouteOverlay.d.ts","sourceRoot":"","sources":["../../src/components/RouteOverlay.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAE/B,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,+BAA+B,CAAC;AAIvE,wBAAgB,YAAY,CAAC,EAC3B,MAAM,EACN,eAAsB,EACtB,aAAoB,EACpB,aAAa,EACb,gBAAgB,EAChB,cAAc,GACf,EAAE,iBAAiB,4BAuCnB;AAED,eAAe,YAAY,CAAC"}
@@ -0,0 +1,20 @@
1
+ import * as React from 'react';
2
+ import { normalizeLatLngList } from '../utils/GeoUtils';
3
+ import { Marker, Polyline } from './overlays';
4
+ export function RouteOverlay({ points, showStartMarker = true, showEndMarker = true, polylineProps, startMarkerProps, endMarkerProps, }) {
5
+ // 统一封装“主路径 + 起点 + 终点”的常见展示组合,
6
+ // 这样业务侧不需要每次手动拼装 3 个覆盖物。
7
+ const normalizedPoints = React.useMemo(() => normalizeLatLngList(points), [points]);
8
+ if (normalizedPoints.length === 0) {
9
+ return null;
10
+ }
11
+ const start = normalizedPoints[0];
12
+ const end = normalizedPoints[normalizedPoints.length - 1];
13
+ return (<>
14
+ <Polyline points={normalizedPoints} strokeWidth={polylineProps?.strokeWidth ?? 6} strokeColor={polylineProps?.strokeColor ?? '#2563eb'} {...polylineProps}/>
15
+ {showStartMarker ? (<Marker position={start} title={startMarkerProps?.title ?? '起点'} {...startMarkerProps}/>) : null}
16
+ {showEndMarker ? (<Marker position={end} title={endMarkerProps?.title ?? '终点'} {...endMarkerProps}/>) : null}
17
+ </>);
18
+ }
19
+ export default RouteOverlay;
20
+ //# sourceMappingURL=RouteOverlay.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"RouteOverlay.js","sourceRoot":"","sources":["../../src/components/RouteOverlay.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAG/B,OAAO,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACxD,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAC;AAE9C,MAAM,UAAU,YAAY,CAAC,EAC3B,MAAM,EACN,eAAe,GAAG,IAAI,EACtB,aAAa,GAAG,IAAI,EACpB,aAAa,EACb,gBAAgB,EAChB,cAAc,GACI;IAClB,8BAA8B;IAC9B,yBAAyB;IACzB,MAAM,gBAAgB,GAAG,KAAK,CAAC,OAAO,CACpC,GAAG,EAAE,CAAC,mBAAmB,CAAC,MAAM,CAAC,EACjC,CAAC,MAAM,CAAC,CACT,CAAC;IAEF,IAAI,gBAAgB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAClC,OAAO,IAAI,CAAC;IACd,CAAC;IAED,MAAM,KAAK,GAAG,gBAAgB,CAAC,CAAC,CAAC,CAAC;IAClC,MAAM,GAAG,GAAG,gBAAgB,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IAE1D,OAAO,CACL,EACE;MAAA,CAAC,QAAQ,CACP,MAAM,CAAC,CAAC,gBAAgB,CAAC,CACzB,WAAW,CAAC,CAAC,aAAa,EAAE,WAAW,IAAI,CAAC,CAAC,CAC7C,WAAW,CAAC,CAAC,aAAa,EAAE,WAAW,IAAI,SAAS,CAAC,CACrD,IAAI,aAAa,CAAC,EAEpB;MAAA,CAAC,eAAe,CAAC,CAAC,CAAC,CACjB,CAAC,MAAM,CACL,QAAQ,CAAC,CAAC,KAAK,CAAC,CAChB,KAAK,CAAC,CAAC,gBAAgB,EAAE,KAAK,IAAI,IAAI,CAAC,CACvC,IAAI,gBAAgB,CAAC,EACrB,CACH,CAAC,CAAC,CAAC,IAAI,CACR;MAAA,CAAC,aAAa,CAAC,CAAC,CAAC,CACf,CAAC,MAAM,CACL,QAAQ,CAAC,CAAC,GAAG,CAAC,CACd,KAAK,CAAC,CAAC,cAAc,EAAE,KAAK,IAAI,IAAI,CAAC,CACrC,IAAI,cAAc,CAAC,EACnB,CACH,CAAC,CAAC,CAAC,IAAI,CACV;IAAA,GAAG,CACJ,CAAC;AACJ,CAAC;AAED,eAAe,YAAY,CAAC","sourcesContent":["import * as React from 'react';\n\nimport type { RouteOverlayProps } from '../types/route-playback.types';\nimport { normalizeLatLngList } from '../utils/GeoUtils';\nimport { Marker, Polyline } from './overlays';\n\nexport function RouteOverlay({\n points,\n showStartMarker = true,\n showEndMarker = true,\n polylineProps,\n startMarkerProps,\n endMarkerProps,\n}: RouteOverlayProps) {\n // 统一封装“主路径 + 起点 + 终点”的常见展示组合,\n // 这样业务侧不需要每次手动拼装 3 个覆盖物。\n const normalizedPoints = React.useMemo(\n () => normalizeLatLngList(points),\n [points]\n );\n\n if (normalizedPoints.length === 0) {\n return null;\n }\n\n const start = normalizedPoints[0];\n const end = normalizedPoints[normalizedPoints.length - 1];\n\n return (\n <>\n <Polyline\n points={normalizedPoints}\n strokeWidth={polylineProps?.strokeWidth ?? 6}\n strokeColor={polylineProps?.strokeColor ?? '#2563eb'}\n {...polylineProps}\n />\n {showStartMarker ? (\n <Marker\n position={start}\n title={startMarkerProps?.title ?? '起点'}\n {...startMarkerProps}\n />\n ) : null}\n {showEndMarker ? (\n <Marker\n position={end}\n title={endMarkerProps?.title ?? '终点'}\n {...endMarkerProps}\n />\n ) : null}\n </>\n );\n}\n\nexport default RouteOverlay;\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"Marker.d.ts","sourceRoot":"","sources":["../../../src/components/overlays/Marker.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAG/B,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AA4C/C;;;;;;;;;GASG;AACH,iBAAS,MAAM,CAAC,KAAK,EAAE,WAAW,qBAsEjC;;AAsDD,wBAAiD"}
1
+ {"version":3,"file":"Marker.d.ts","sourceRoot":"","sources":["../../../src/components/overlays/Marker.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAG/B,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AA6C/C;;;;;;;;;GASG;AACH,iBAAS,MAAM,CAAC,KAAK,EAAE,WAAW,qBAyJjC;;AAsDD,wBAAiD"}
@@ -1,5 +1,6 @@
1
1
  import * as React from 'react';
2
2
  import { Platform, StyleSheet, View } from 'react-native';
3
+ import ExpoGaodeMapModule from '../../ExpoGaodeMapModule';
3
4
  import { normalizeLatLng, normalizeLatLngList } from '../../utils/GeoUtils';
4
5
  import { createLazyNativeViewManager } from '../../utils/lazyNativeViewManager';
5
6
  const getNativeMarkerView = createLazyNativeViewManager('MarkerView');
@@ -38,7 +39,7 @@ function Marker(props) {
38
39
  const NativeMarkerView = React.useMemo(() => getNativeMarkerView(), []);
39
40
  const [measuredSize, setMeasuredSize] = React.useState(AUTO_SIZE_FALLBACK);
40
41
  // 从 props 中排除 position 属性,避免传递到原生层
41
- const { position, customViewWidth, customViewHeight, iconWidth, iconHeight, children, smoothMovePath, ...restProps } = props;
42
+ const { position, customViewWidth, customViewHeight, iconWidth, iconHeight, children, smoothMovePath, cacheKey, ...restProps } = props;
42
43
  // 归一化坐标处理
43
44
  const normalizedPosition = normalizeLatLng(position);
44
45
  const normalizedSmoothMovePath = smoothMovePath ? normalizeLatLngList(smoothMovePath) : undefined;
@@ -52,6 +53,67 @@ function Marker(props) {
52
53
  const resolvedCustomViewHeight = customViewHeight && customViewHeight > 0
53
54
  ? customViewHeight
54
55
  : (shouldUseAutoMeasuredSize ? measuredSize.height : 0);
56
+ React.useEffect(() => {
57
+ if (!normalizedSmoothMovePath ||
58
+ normalizedSmoothMovePath.length < 2 ||
59
+ !props.smoothMoveDuration ||
60
+ props.smoothMoveDuration <= 0) {
61
+ return undefined;
62
+ }
63
+ const totalDistance = ExpoGaodeMapModule.calculatePathLength(normalizedSmoothMovePath);
64
+ if (totalDistance <= 0) {
65
+ props.onSmoothMoveEnd?.({
66
+ nativeEvent: {
67
+ position: normalizedSmoothMovePath[normalizedSmoothMovePath.length - 1],
68
+ angle: 0,
69
+ totalDistance,
70
+ },
71
+ });
72
+ return undefined;
73
+ }
74
+ const durationMs = props.smoothMoveDuration * 1000;
75
+ const startedAt = Date.now();
76
+ const tick = () => {
77
+ const progress = Math.min(1, (Date.now() - startedAt) / durationMs);
78
+ const distance = totalDistance * progress;
79
+ const pointInfo = ExpoGaodeMapModule.getPointAtDistance(normalizedSmoothMovePath, distance);
80
+ const point = pointInfo
81
+ ? { latitude: pointInfo.latitude, longitude: pointInfo.longitude }
82
+ : normalizedSmoothMovePath[normalizedSmoothMovePath.length - 1];
83
+ const angle = pointInfo?.angle ?? 0;
84
+ props.onSmoothMoveProgress?.({
85
+ nativeEvent: {
86
+ position: point,
87
+ angle,
88
+ progress,
89
+ distance,
90
+ totalDistance,
91
+ },
92
+ });
93
+ if (progress >= 1) {
94
+ props.onSmoothMoveEnd?.({
95
+ nativeEvent: {
96
+ position: point,
97
+ angle,
98
+ totalDistance,
99
+ },
100
+ });
101
+ }
102
+ };
103
+ tick();
104
+ const intervalId = setInterval(() => {
105
+ tick();
106
+ if (Date.now() - startedAt >= durationMs) {
107
+ clearInterval(intervalId);
108
+ }
109
+ }, 100);
110
+ return () => clearInterval(intervalId);
111
+ }, [
112
+ normalizedSmoothMovePath,
113
+ props.onSmoothMoveEnd,
114
+ props.onSmoothMoveProgress,
115
+ props.smoothMoveDuration,
116
+ ]);
55
117
  const handleAutoMeasure = (event) => {
56
118
  const nextWidth = customViewWidth && customViewWidth > 0
57
119
  ? customViewWidth
@@ -74,7 +136,8 @@ function Marker(props) {
74
136
  const finalIconHeight = hasChildren
75
137
  ? resolvedCustomViewHeight
76
138
  : (iconHeight && iconHeight > 0 ? iconHeight : 40);
77
- return (<NativeMarkerView latitude={normalizedPosition.latitude} longitude={normalizedPosition.longitude} iconWidth={finalIconWidth} iconHeight={finalIconHeight} customViewWidth={finalIconWidth} customViewHeight={finalIconHeight} smoothMovePath={normalizedSmoothMovePath} {...restProps}>
139
+ const optionalNativeProps = cacheKey != null ? { cacheKey } : undefined;
140
+ return (<NativeMarkerView latitude={normalizedPosition.latitude} longitude={normalizedPosition.longitude} iconWidth={finalIconWidth} iconHeight={finalIconHeight} customViewWidth={finalIconWidth} customViewHeight={finalIconHeight} smoothMovePath={normalizedSmoothMovePath} {...optionalNativeProps} {...restProps}>
78
141
  {hasChildren && shouldWrapChildrenForMeasurement ? (<View collapsable={false} onLayout={Platform.OS === 'ios' ? handleAutoMeasure : undefined} style={styles.measureContainer}>
79
142
  {children}
80
143
  </View>) : children}
@@ -1 +1 @@
1
- {"version":3,"file":"Marker.js","sourceRoot":"","sources":["../../../src/components/overlays/Marker.tsx"],"names":[],"mappings":"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","sourcesContent":["import * as React from 'react';\nimport type { LayoutChangeEvent } from 'react-native';\nimport { Platform, StyleSheet, View } from 'react-native';\nimport type { MarkerProps } from '../../types';\nimport { normalizeLatLng, normalizeLatLngList } from '../../utils/GeoUtils';\nimport { createLazyNativeViewManager } from '../../utils/lazyNativeViewManager';\n\ntype NativeMarkerViewProps = Omit<MarkerProps, 'position'> & {\n latitude: number;\n longitude: number;\n};\n\nconst getNativeMarkerView = createLazyNativeViewManager<NativeMarkerViewProps>('MarkerView');\n\nconst AUTO_SIZE_FALLBACK = { width: 0, height: 0 };\n\nfunction areSmoothMovePathsEqual(\n prevPath: MarkerProps['smoothMovePath'],\n nextPath: MarkerProps['smoothMovePath']\n): boolean {\n if (prevPath === nextPath) {\n return true;\n }\n\n if (!prevPath || !nextPath) {\n return prevPath === nextPath;\n }\n\n if (prevPath.length !== nextPath.length) {\n return false;\n }\n\n for (let index = 0; index < prevPath.length; index += 1) {\n const prevPoint = normalizeLatLng(prevPath[index]);\n const nextPoint = normalizeLatLng(nextPath[index]);\n\n if (\n prevPoint.latitude !== nextPoint.latitude ||\n prevPoint.longitude !== nextPoint.longitude\n ) {\n return false;\n }\n }\n\n return true;\n}\n\n/**\n * Marker 组件 - 完全声明式 API\n *\n * 支持:\n * - 自定义图标(icon)\n * - 自定义内容(children)- 自动测量尺寸\n * - 大头针样式(pinColor)\n * - 拖拽功能\n * - 所有事件回调\n */\nfunction Marker(props: MarkerProps) {\n const NativeMarkerView = React.useMemo(() => getNativeMarkerView(), []);\n const [measuredSize, setMeasuredSize] = React.useState(AUTO_SIZE_FALLBACK);\n // 从 props 中排除 position 属性,避免传递到原生层\n const { position, customViewWidth, customViewHeight, iconWidth, iconHeight, children, smoothMovePath, ...restProps } = props;\n \n // 归一化坐标处理\n const normalizedPosition = normalizeLatLng(position);\n const normalizedSmoothMovePath = smoothMovePath ? normalizeLatLngList(smoothMovePath) : undefined;\n\n // 根据是否有 children 来决定使用哪个尺寸属性\n const hasChildren = !!children;\n const shouldWrapChildrenForMeasurement = hasChildren;\n const shouldUseAutoMeasuredSize = Platform.OS === 'ios';\n const resolvedCustomViewWidth = customViewWidth && customViewWidth > 0\n ? customViewWidth\n : (shouldUseAutoMeasuredSize ? measuredSize.width : 0);\n const resolvedCustomViewHeight = customViewHeight && customViewHeight > 0\n ? customViewHeight\n : (shouldUseAutoMeasuredSize ? measuredSize.height : 0);\n\n const handleAutoMeasure = (event: LayoutChangeEvent) => {\n const nextWidth = customViewWidth && customViewWidth > 0\n ? customViewWidth\n : Math.ceil(event.nativeEvent.layout.width);\n const nextHeight = customViewHeight && customViewHeight > 0\n ? customViewHeight\n : Math.ceil(event.nativeEvent.layout.height);\n\n if (nextWidth === measuredSize.width && nextHeight === measuredSize.height) {\n return;\n }\n\n setMeasuredSize({\n width: nextWidth,\n height: nextHeight,\n });\n };\n \n // 智能尺寸计算\n const finalIconWidth = hasChildren\n ? resolvedCustomViewWidth\n : (iconWidth && iconWidth > 0 ? iconWidth : 40);\n \n const finalIconHeight = hasChildren\n ? resolvedCustomViewHeight\n : (iconHeight && iconHeight > 0 ? iconHeight : 40);\n \n return (\n <NativeMarkerView\n latitude={normalizedPosition.latitude}\n longitude={normalizedPosition.longitude}\n iconWidth={finalIconWidth}\n iconHeight={finalIconHeight}\n customViewWidth={finalIconWidth}\n customViewHeight={finalIconHeight}\n smoothMovePath={normalizedSmoothMovePath}\n {...restProps}\n >\n {hasChildren && shouldWrapChildrenForMeasurement ? (\n <View\n collapsable={false}\n onLayout={Platform.OS === 'ios' ? handleAutoMeasure : undefined}\n style={styles.measureContainer}\n >\n {children}\n </View>\n ) : children}\n </NativeMarkerView>\n );\n}\n\n/**\n * 🔑 性能优化:极简比较函数\n * 只检查最常变化的关键属性,减少 JS 线程开销\n */\nfunction arePropsEqual(prevProps: MarkerProps, nextProps: MarkerProps): boolean {\n // 快速路径:比较 position (最常变化)\n const prevPos = normalizeLatLng(prevProps.position);\n const nextPos = normalizeLatLng(nextProps.position);\n\n if (\n prevPos.latitude !== nextPos.latitude ||\n prevPos.longitude !== nextPos.longitude\n ) {\n return false;\n }\n \n // 比较 cacheKey (如果提供了 cacheKey,其他属性理论上不会变)\n if (prevProps.cacheKey !== nextProps.cacheKey) {\n return false;\n }\n \n // 比较 children (如果有 children)\n if (prevProps.children !== nextProps.children) {\n return false;\n }\n\n // 比较自定义内容尺寸和图标尺寸\n if (\n prevProps.customViewWidth !== nextProps.customViewWidth ||\n prevProps.customViewHeight !== nextProps.customViewHeight ||\n prevProps.icon !== nextProps.icon ||\n prevProps.iconWidth !== nextProps.iconWidth ||\n prevProps.iconHeight !== nextProps.iconHeight\n ) {\n return false;\n }\n \n // 比较 smoothMovePath (平滑移动路径)\n if (!areSmoothMovePathsEqual(prevProps.smoothMovePath, nextProps.smoothMovePath)) {\n return false;\n }\n \n // 比较 smoothMoveDuration (平滑移动时长)\n if (prevProps.smoothMoveDuration !== nextProps.smoothMoveDuration) {\n return false;\n }\n \n // 其他属性相同,不重新渲染\n return true;\n}\n\n// 导出优化后的组件\nexport default React.memo(Marker, arePropsEqual);\n\nconst styles = StyleSheet.create({\n measureContainer: {\n alignSelf: 'flex-start',\n alignItems: 'flex-start',\n flexShrink: 0,\n },\n});\n"]}
1
+ {"version":3,"file":"Marker.js","sourceRoot":"","sources":["../../../src/components/overlays/Marker.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAE/B,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,cAAc,CAAC;AAE1D,OAAO,kBAAkB,MAAM,0BAA0B,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,mBAAmB,EAAE,MAAM,sBAAsB,CAAC;AAC5E,OAAO,EAAE,2BAA2B,EAAE,MAAM,mCAAmC,CAAC;AAOhF,MAAM,mBAAmB,GAAG,2BAA2B,CAAwB,YAAY,CAAC,CAAC;AAE7F,MAAM,kBAAkB,GAAG,EAAE,KAAK,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,CAAC;AAEnD,SAAS,uBAAuB,CAC9B,QAAuC,EACvC,QAAuC;IAEvC,IAAI,QAAQ,KAAK,QAAQ,EAAE,CAAC;QAC1B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,IAAI,CAAC,QAAQ,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC3B,OAAO,QAAQ,KAAK,QAAQ,CAAC;IAC/B,CAAC;IAED,IAAI,QAAQ,CAAC,MAAM,KAAK,QAAQ,CAAC,MAAM,EAAE,CAAC;QACxC,OAAO,KAAK,CAAC;IACf,CAAC;IAED,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,QAAQ,CAAC,MAAM,EAAE,KAAK,IAAI,CAAC,EAAE,CAAC;QACxD,MAAM,SAAS,GAAG,eAAe,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;QACnD,MAAM,SAAS,GAAG,eAAe,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;QAEnD,IACE,SAAS,CAAC,QAAQ,KAAK,SAAS,CAAC,QAAQ;YACzC,SAAS,CAAC,SAAS,KAAK,SAAS,CAAC,SAAS,EAC3C,CAAC;YACD,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;;;;;;;GASG;AACH,SAAS,MAAM,CAAC,KAAkB;IAChC,MAAM,gBAAgB,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,mBAAmB,EAAE,EAAE,EAAE,CAAC,CAAC;IACxE,MAAM,CAAC,YAAY,EAAE,eAAe,CAAC,GAAG,KAAK,CAAC,QAAQ,CAAC,kBAAkB,CAAC,CAAC;IAC3E,mCAAmC;IACnC,MAAM,EACJ,QAAQ,EACR,eAAe,EACf,gBAAgB,EAChB,SAAS,EACT,UAAU,EACV,QAAQ,EACR,cAAc,EACd,QAAQ,EACR,GAAG,SAAS,EACb,GAAG,KAAK,CAAC;IAEV,UAAU;IACV,MAAM,kBAAkB,GAAG,eAAe,CAAC,QAAQ,CAAC,CAAC;IACrD,MAAM,wBAAwB,GAAG,cAAc,CAAC,CAAC,CAAC,mBAAmB,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAElG,6BAA6B;IAC7B,MAAM,WAAW,GAAG,CAAC,CAAC,QAAQ,CAAC;IAC/B,MAAM,gCAAgC,GAAG,WAAW,CAAC;IACrD,MAAM,yBAAyB,GAAG,QAAQ,CAAC,EAAE,KAAK,KAAK,CAAC;IACxD,MAAM,uBAAuB,GAAG,eAAe,IAAI,eAAe,GAAG,CAAC;QACpE,CAAC,CAAC,eAAe;QACjB,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACzD,MAAM,wBAAwB,GAAG,gBAAgB,IAAI,gBAAgB,GAAG,CAAC;QACvE,CAAC,CAAC,gBAAgB;QAClB,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAE1D,KAAK,CAAC,SAAS,CAAC,GAAG,EAAE;QACnB,IACE,CAAC,wBAAwB;YACzB,wBAAwB,CAAC,MAAM,GAAG,CAAC;YACnC,CAAC,KAAK,CAAC,kBAAkB;YACzB,KAAK,CAAC,kBAAkB,IAAI,CAAC,EAC7B,CAAC;YACD,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,MAAM,aAAa,GAAG,kBAAkB,CAAC,mBAAmB,CAAC,wBAAwB,CAAC,CAAC;QACvF,IAAI,aAAa,IAAI,CAAC,EAAE,CAAC;YACvB,KAAK,CAAC,eAAe,EAAE,CAAC;gBACtB,WAAW,EAAE;oBACX,QAAQ,EAAE,wBAAwB,CAAC,wBAAwB,CAAC,MAAM,GAAG,CAAC,CAAC;oBACvE,KAAK,EAAE,CAAC;oBACR,aAAa;iBACd;aACO,CAAC,CAAC;YACZ,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,MAAM,UAAU,GAAG,KAAK,CAAC,kBAAkB,GAAG,IAAI,CAAC;QACnD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,MAAM,IAAI,GAAG,GAAG,EAAE;YAChB,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC,GAAG,UAAU,CAAC,CAAC;YACpE,MAAM,QAAQ,GAAG,aAAa,GAAG,QAAQ,CAAC;YAC1C,MAAM,SAAS,GAAG,kBAAkB,CAAC,kBAAkB,CAAC,wBAAwB,EAAE,QAAQ,CAAC,CAAC;YAC5F,MAAM,KAAK,GAAG,SAAS;gBACrB,CAAC,CAAC,EAAE,QAAQ,EAAE,SAAS,CAAC,QAAQ,EAAE,SAAS,EAAE,SAAS,CAAC,SAAS,EAAE;gBAClE,CAAC,CAAC,wBAAwB,CAAC,wBAAwB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YAClE,MAAM,KAAK,GAAG,SAAS,EAAE,KAAK,IAAI,CAAC,CAAC;YAEpC,KAAK,CAAC,oBAAoB,EAAE,CAAC;gBAC3B,WAAW,EAAE;oBACX,QAAQ,EAAE,KAAK;oBACf,KAAK;oBACL,QAAQ;oBACR,QAAQ;oBACR,aAAa;iBACd;aACO,CAAC,CAAC;YAEZ,IAAI,QAAQ,IAAI,CAAC,EAAE,CAAC;gBAClB,KAAK,CAAC,eAAe,EAAE,CAAC;oBACtB,WAAW,EAAE;wBACX,QAAQ,EAAE,KAAK;wBACf,KAAK;wBACL,aAAa;qBACd;iBACO,CAAC,CAAC;YACd,CAAC;QACH,CAAC,CAAC;QAEF,IAAI,EAAE,CAAC;QACP,MAAM,UAAU,GAAG,WAAW,CAAC,GAAG,EAAE;YAClC,IAAI,EAAE,CAAC;YACP,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,IAAI,UAAU,EAAE,CAAC;gBACzC,aAAa,CAAC,UAAU,CAAC,CAAC;YAC5B,CAAC;QACH,CAAC,EAAE,GAAG,CAAC,CAAC;QAER,OAAO,GAAG,EAAE,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;IACzC,CAAC,EAAE;QACD,wBAAwB;QACxB,KAAK,CAAC,eAAe;QACrB,KAAK,CAAC,oBAAoB;QAC1B,KAAK,CAAC,kBAAkB;KACzB,CAAC,CAAC;IAEH,MAAM,iBAAiB,GAAG,CAAC,KAAwB,EAAE,EAAE;QACrD,MAAM,SAAS,GAAG,eAAe,IAAI,eAAe,GAAG,CAAC;YACtD,CAAC,CAAC,eAAe;YACjB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC9C,MAAM,UAAU,GAAG,gBAAgB,IAAI,gBAAgB,GAAG,CAAC;YACzD,CAAC,CAAC,gBAAgB;YAClB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAE/C,IAAI,SAAS,KAAK,YAAY,CAAC,KAAK,IAAI,UAAU,KAAK,YAAY,CAAC,MAAM,EAAE,CAAC;YAC3E,OAAO;QACT,CAAC;QAED,eAAe,CAAC;YACd,KAAK,EAAE,SAAS;YAChB,MAAM,EAAE,UAAU;SACnB,CAAC,CAAC;IACL,CAAC,CAAC;IAEF,SAAS;IACT,MAAM,cAAc,GAAG,WAAW;QAChC,CAAC,CAAC,uBAAuB;QACzB,CAAC,CAAC,CAAC,SAAS,IAAI,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAElD,MAAM,eAAe,GAAG,WAAW;QACjC,CAAC,CAAC,wBAAwB;QAC1B,CAAC,CAAC,CAAC,UAAU,IAAI,UAAU,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAErD,MAAM,mBAAmB,GAAG,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;IAExE,OAAO,CACL,CAAC,gBAAgB,CACf,QAAQ,CAAC,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CACtC,SAAS,CAAC,CAAC,kBAAkB,CAAC,SAAS,CAAC,CACxC,SAAS,CAAC,CAAC,cAAc,CAAC,CAC1B,UAAU,CAAC,CAAC,eAAe,CAAC,CAC5B,eAAe,CAAC,CAAC,cAAc,CAAC,CAChC,gBAAgB,CAAC,CAAC,eAAe,CAAC,CAClC,cAAc,CAAC,CAAC,wBAAwB,CAAC,CACzC,IAAI,mBAAmB,CAAC,CACxB,IAAI,SAAS,CAAC,CAEd;MAAA,CAAC,WAAW,IAAI,gCAAgC,CAAC,CAAC,CAAC,CACjD,CAAC,IAAI,CACH,WAAW,CAAC,CAAC,KAAK,CAAC,CACnB,QAAQ,CAAC,CAAC,QAAQ,CAAC,EAAE,KAAK,KAAK,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC,SAAS,CAAC,CAChE,KAAK,CAAC,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAE/B;UAAA,CAAC,QAAQ,CACX;QAAA,EAAE,IAAI,CAAC,CACR,CAAC,CAAC,CAAC,QAAQ,CACd;IAAA,EAAE,gBAAgB,CAAC,CACpB,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,SAAS,aAAa,CAAC,SAAsB,EAAE,SAAsB;IACnE,0BAA0B;IAC1B,MAAM,OAAO,GAAG,eAAe,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;IACpD,MAAM,OAAO,GAAG,eAAe,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;IAEpD,IACE,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC,QAAQ;QACrC,OAAO,CAAC,SAAS,KAAK,OAAO,CAAC,SAAS,EACvC,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,0CAA0C;IAC1C,IAAI,SAAS,CAAC,QAAQ,KAAK,SAAS,CAAC,QAAQ,EAAE,CAAC;QAC9C,OAAO,KAAK,CAAC;IACf,CAAC;IAED,6BAA6B;IAC7B,IAAI,SAAS,CAAC,QAAQ,KAAK,SAAS,CAAC,QAAQ,EAAE,CAAC;QAC9C,OAAO,KAAK,CAAC;IACf,CAAC;IAED,iBAAiB;IACjB,IACE,SAAS,CAAC,eAAe,KAAK,SAAS,CAAC,eAAe;QACvD,SAAS,CAAC,gBAAgB,KAAK,SAAS,CAAC,gBAAgB;QACzD,SAAS,CAAC,IAAI,KAAK,SAAS,CAAC,IAAI;QACjC,SAAS,CAAC,SAAS,KAAK,SAAS,CAAC,SAAS;QAC3C,SAAS,CAAC,UAAU,KAAK,SAAS,CAAC,UAAU,EAC7C,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,6BAA6B;IAC7B,IAAI,CAAC,uBAAuB,CAAC,SAAS,CAAC,cAAc,EAAE,SAAS,CAAC,cAAc,CAAC,EAAE,CAAC;QACjF,OAAO,KAAK,CAAC;IACf,CAAC;IAED,iCAAiC;IACjC,IAAI,SAAS,CAAC,kBAAkB,KAAK,SAAS,CAAC,kBAAkB,EAAE,CAAC;QAClE,OAAO,KAAK,CAAC;IACf,CAAC;IAED,eAAe;IACf,OAAO,IAAI,CAAC;AACd,CAAC;AAED,WAAW;AACX,eAAe,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;AAEjD,MAAM,MAAM,GAAG,UAAU,CAAC,MAAM,CAAC;IAC/B,gBAAgB,EAAE;QAChB,SAAS,EAAE,YAAY;QACvB,UAAU,EAAE,YAAY;QACxB,UAAU,EAAE,CAAC;KACd;CACF,CAAC,CAAC","sourcesContent":["import * as React from 'react';\nimport type { LayoutChangeEvent } from 'react-native';\nimport { Platform, StyleSheet, View } from 'react-native';\nimport type { MarkerProps } from '../../types';\nimport ExpoGaodeMapModule from '../../ExpoGaodeMapModule';\nimport { normalizeLatLng, normalizeLatLngList } from '../../utils/GeoUtils';\nimport { createLazyNativeViewManager } from '../../utils/lazyNativeViewManager';\n\ntype NativeMarkerViewProps = Omit<MarkerProps, 'position'> & {\n latitude: number;\n longitude: number;\n};\n\nconst getNativeMarkerView = createLazyNativeViewManager<NativeMarkerViewProps>('MarkerView');\n\nconst AUTO_SIZE_FALLBACK = { width: 0, height: 0 };\n\nfunction areSmoothMovePathsEqual(\n prevPath: MarkerProps['smoothMovePath'],\n nextPath: MarkerProps['smoothMovePath']\n): boolean {\n if (prevPath === nextPath) {\n return true;\n }\n\n if (!prevPath || !nextPath) {\n return prevPath === nextPath;\n }\n\n if (prevPath.length !== nextPath.length) {\n return false;\n }\n\n for (let index = 0; index < prevPath.length; index += 1) {\n const prevPoint = normalizeLatLng(prevPath[index]);\n const nextPoint = normalizeLatLng(nextPath[index]);\n\n if (\n prevPoint.latitude !== nextPoint.latitude ||\n prevPoint.longitude !== nextPoint.longitude\n ) {\n return false;\n }\n }\n\n return true;\n}\n\n/**\n * Marker 组件 - 完全声明式 API\n *\n * 支持:\n * - 自定义图标(icon)\n * - 自定义内容(children)- 自动测量尺寸\n * - 大头针样式(pinColor)\n * - 拖拽功能\n * - 所有事件回调\n */\nfunction Marker(props: MarkerProps) {\n const NativeMarkerView = React.useMemo(() => getNativeMarkerView(), []);\n const [measuredSize, setMeasuredSize] = React.useState(AUTO_SIZE_FALLBACK);\n // 从 props 中排除 position 属性,避免传递到原生层\n const {\n position,\n customViewWidth,\n customViewHeight,\n iconWidth,\n iconHeight,\n children,\n smoothMovePath,\n cacheKey,\n ...restProps\n } = props;\n \n // 归一化坐标处理\n const normalizedPosition = normalizeLatLng(position);\n const normalizedSmoothMovePath = smoothMovePath ? normalizeLatLngList(smoothMovePath) : undefined;\n\n // 根据是否有 children 来决定使用哪个尺寸属性\n const hasChildren = !!children;\n const shouldWrapChildrenForMeasurement = hasChildren;\n const shouldUseAutoMeasuredSize = Platform.OS === 'ios';\n const resolvedCustomViewWidth = customViewWidth && customViewWidth > 0\n ? customViewWidth\n : (shouldUseAutoMeasuredSize ? measuredSize.width : 0);\n const resolvedCustomViewHeight = customViewHeight && customViewHeight > 0\n ? customViewHeight\n : (shouldUseAutoMeasuredSize ? measuredSize.height : 0);\n\n React.useEffect(() => {\n if (\n !normalizedSmoothMovePath ||\n normalizedSmoothMovePath.length < 2 ||\n !props.smoothMoveDuration ||\n props.smoothMoveDuration <= 0\n ) {\n return undefined;\n }\n\n const totalDistance = ExpoGaodeMapModule.calculatePathLength(normalizedSmoothMovePath);\n if (totalDistance <= 0) {\n props.onSmoothMoveEnd?.({\n nativeEvent: {\n position: normalizedSmoothMovePath[normalizedSmoothMovePath.length - 1],\n angle: 0,\n totalDistance,\n },\n } as never);\n return undefined;\n }\n\n const durationMs = props.smoothMoveDuration * 1000;\n const startedAt = Date.now();\n const tick = () => {\n const progress = Math.min(1, (Date.now() - startedAt) / durationMs);\n const distance = totalDistance * progress;\n const pointInfo = ExpoGaodeMapModule.getPointAtDistance(normalizedSmoothMovePath, distance);\n const point = pointInfo\n ? { latitude: pointInfo.latitude, longitude: pointInfo.longitude }\n : normalizedSmoothMovePath[normalizedSmoothMovePath.length - 1];\n const angle = pointInfo?.angle ?? 0;\n\n props.onSmoothMoveProgress?.({\n nativeEvent: {\n position: point,\n angle,\n progress,\n distance,\n totalDistance,\n },\n } as never);\n\n if (progress >= 1) {\n props.onSmoothMoveEnd?.({\n nativeEvent: {\n position: point,\n angle,\n totalDistance,\n },\n } as never);\n }\n };\n\n tick();\n const intervalId = setInterval(() => {\n tick();\n if (Date.now() - startedAt >= durationMs) {\n clearInterval(intervalId);\n }\n }, 100);\n\n return () => clearInterval(intervalId);\n }, [\n normalizedSmoothMovePath,\n props.onSmoothMoveEnd,\n props.onSmoothMoveProgress,\n props.smoothMoveDuration,\n ]);\n\n const handleAutoMeasure = (event: LayoutChangeEvent) => {\n const nextWidth = customViewWidth && customViewWidth > 0\n ? customViewWidth\n : Math.ceil(event.nativeEvent.layout.width);\n const nextHeight = customViewHeight && customViewHeight > 0\n ? customViewHeight\n : Math.ceil(event.nativeEvent.layout.height);\n\n if (nextWidth === measuredSize.width && nextHeight === measuredSize.height) {\n return;\n }\n\n setMeasuredSize({\n width: nextWidth,\n height: nextHeight,\n });\n };\n \n // 智能尺寸计算\n const finalIconWidth = hasChildren\n ? resolvedCustomViewWidth\n : (iconWidth && iconWidth > 0 ? iconWidth : 40);\n \n const finalIconHeight = hasChildren\n ? resolvedCustomViewHeight\n : (iconHeight && iconHeight > 0 ? iconHeight : 40);\n\n const optionalNativeProps = cacheKey != null ? { cacheKey } : undefined;\n \n return (\n <NativeMarkerView\n latitude={normalizedPosition.latitude}\n longitude={normalizedPosition.longitude}\n iconWidth={finalIconWidth}\n iconHeight={finalIconHeight}\n customViewWidth={finalIconWidth}\n customViewHeight={finalIconHeight}\n smoothMovePath={normalizedSmoothMovePath}\n {...optionalNativeProps}\n {...restProps}\n >\n {hasChildren && shouldWrapChildrenForMeasurement ? (\n <View\n collapsable={false}\n onLayout={Platform.OS === 'ios' ? handleAutoMeasure : undefined}\n style={styles.measureContainer}\n >\n {children}\n </View>\n ) : children}\n </NativeMarkerView>\n );\n}\n\n/**\n * 🔑 性能优化:极简比较函数\n * 只检查最常变化的关键属性,减少 JS 线程开销\n */\nfunction arePropsEqual(prevProps: MarkerProps, nextProps: MarkerProps): boolean {\n // 快速路径:比较 position (最常变化)\n const prevPos = normalizeLatLng(prevProps.position);\n const nextPos = normalizeLatLng(nextProps.position);\n\n if (\n prevPos.latitude !== nextPos.latitude ||\n prevPos.longitude !== nextPos.longitude\n ) {\n return false;\n }\n \n // 比较 cacheKey (如果提供了 cacheKey,其他属性理论上不会变)\n if (prevProps.cacheKey !== nextProps.cacheKey) {\n return false;\n }\n \n // 比较 children (如果有 children)\n if (prevProps.children !== nextProps.children) {\n return false;\n }\n\n // 比较自定义内容尺寸和图标尺寸\n if (\n prevProps.customViewWidth !== nextProps.customViewWidth ||\n prevProps.customViewHeight !== nextProps.customViewHeight ||\n prevProps.icon !== nextProps.icon ||\n prevProps.iconWidth !== nextProps.iconWidth ||\n prevProps.iconHeight !== nextProps.iconHeight\n ) {\n return false;\n }\n \n // 比较 smoothMovePath (平滑移动路径)\n if (!areSmoothMovePathsEqual(prevProps.smoothMovePath, nextProps.smoothMovePath)) {\n return false;\n }\n \n // 比较 smoothMoveDuration (平滑移动时长)\n if (prevProps.smoothMoveDuration !== nextProps.smoothMoveDuration) {\n return false;\n }\n \n // 其他属性相同,不重新渲染\n return true;\n}\n\n// 导出优化后的组件\nexport default React.memo(Marker, arePropsEqual);\n\nconst styles = StyleSheet.create({\n measureContainer: {\n alignSelf: 'flex-start',\n alignItems: 'flex-start',\n flexShrink: 0,\n },\n});\n"]}
@@ -0,0 +1,4 @@
1
+ import type { LatLng } from '../types/common.types';
2
+ import type { RoutePlaybackController, RoutePlaybackOptions } from '../types/route-playback.types';
3
+ export declare function useRoutePlayback(points: Array<LatLng | [number, number] | number[]>, options?: RoutePlaybackOptions): RoutePlaybackController;
4
+ //# sourceMappingURL=useRoutePlayback.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useRoutePlayback.d.ts","sourceRoot":"","sources":["../../src/hooks/useRoutePlayback.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,uBAAuB,CAAC;AACpD,OAAO,KAAK,EACV,uBAAuB,EACvB,oBAAoB,EAErB,MAAM,+BAA+B,CAAC;AAyDvC,wBAAgB,gBAAgB,CAC9B,MAAM,EAAE,KAAK,CAAC,MAAM,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,EAAE,CAAC,EACnD,OAAO,GAAE,oBAAyB,GACjC,uBAAuB,CAkUzB"}
@@ -0,0 +1,310 @@
1
+ import * as React from 'react';
2
+ import ExpoGaodeMapModule from '../ExpoGaodeMapModule';
3
+ import { MapContext } from '../components/MapContext';
4
+ import { normalizeLatLngList } from '../utils/GeoUtils';
5
+ import { fitCameraToCoordinates } from '../utils/RouteUtils';
6
+ const DEFAULT_STATE = {
7
+ isPlaying: false,
8
+ isPaused: false,
9
+ progress: 0,
10
+ currentPosition: null,
11
+ currentAngle: 0,
12
+ totalDistance: 0,
13
+ traveledDistance: 0,
14
+ durationSeconds: 0,
15
+ };
16
+ function getMarkerAngle(path, totalDistance, targetDistance, lastAngle, options) {
17
+ // 先取当前距离上的点,再额外预读一个前瞻点,
18
+ // 让车辆朝向在转弯时更自然,不会出现生硬折角。
19
+ const pointInfo = ExpoGaodeMapModule.getPointAtDistance(path, targetDistance);
20
+ if (!pointInfo) {
21
+ return { angle: lastAngle, point: null };
22
+ }
23
+ const lookAheadDistance = options.lookAheadDistanceMeters ?? 5;
24
+ const futurePoint = ExpoGaodeMapModule.getPointAtDistance(path, Math.min(totalDistance, targetDistance + lookAheadDistance));
25
+ let targetAngle = pointInfo.angle;
26
+ if (futurePoint && targetDistance + lookAheadDistance < totalDistance) {
27
+ let diffNext = futurePoint.angle - pointInfo.angle;
28
+ if (diffNext > 180)
29
+ diffNext -= 360;
30
+ if (diffNext < -180)
31
+ diffNext += 360;
32
+ targetAngle = pointInfo.angle + diffNext * 0.4;
33
+ }
34
+ let diff = targetAngle - lastAngle;
35
+ if (diff > 180)
36
+ diff -= 360;
37
+ if (diff < -180)
38
+ diff += 360;
39
+ const bearingSmoothing = options.bearingSmoothing ?? 0.2;
40
+ return {
41
+ angle: lastAngle + diff * bearingSmoothing,
42
+ point: {
43
+ latitude: pointInfo.latitude,
44
+ longitude: pointInfo.longitude,
45
+ },
46
+ };
47
+ }
48
+ export function useRoutePlayback(points, options = {}) {
49
+ const map = React.useContext(MapContext);
50
+ const optionsRef = React.useRef(options);
51
+ const [state, setState] = React.useState(DEFAULT_STATE);
52
+ const stateRef = React.useRef(DEFAULT_STATE);
53
+ const [speedMultiplier, setSpeedMultiplierState] = React.useState(options.speedMultiplier ?? 1);
54
+ const timerRef = React.useRef(null);
55
+ const startAtRef = React.useRef(0);
56
+ const elapsedBeforePauseRef = React.useRef(0);
57
+ const lastAngleRef = React.useRef(0);
58
+ const previousPathRef = React.useRef(null);
59
+ const previousDurationRef = React.useRef(null);
60
+ React.useEffect(() => {
61
+ optionsRef.current = options;
62
+ }, [options]);
63
+ const normalizedPath = React.useMemo(() => {
64
+ // 长路径可选走一次抽稀,减少轨迹回放期间的计算量。
65
+ const normalized = normalizeLatLngList(points);
66
+ if (normalized.length <= 2 || !options.simplificationTolerance) {
67
+ return normalized;
68
+ }
69
+ return ExpoGaodeMapModule.simplifyPolyline(normalized, options.simplificationTolerance);
70
+ }, [options.simplificationTolerance, points]);
71
+ const totalDistance = React.useMemo(() => ExpoGaodeMapModule.calculatePathLength(normalizedPath), [normalizedPath]);
72
+ const durationSeconds = React.useMemo(() => {
73
+ if (options.durationSeconds) {
74
+ return options.durationSeconds;
75
+ }
76
+ const baseSpeed = options.baseSpeedMps ?? 15;
77
+ const minDuration = options.minDurationSeconds ?? 5;
78
+ if (totalDistance <= 0) {
79
+ return 0;
80
+ }
81
+ return Math.max(minDuration, totalDistance / (baseSpeed * Math.max(speedMultiplier, 0.1)));
82
+ }, [
83
+ options.baseSpeedMps,
84
+ options.durationSeconds,
85
+ options.minDurationSeconds,
86
+ speedMultiplier,
87
+ totalDistance,
88
+ ]);
89
+ const stopTimer = React.useCallback(() => {
90
+ if (timerRef.current) {
91
+ clearInterval(timerRef.current);
92
+ timerRef.current = null;
93
+ }
94
+ }, []);
95
+ const publishState = React.useCallback((nextState) => {
96
+ // 统一从这里下发状态,避免不同控制分支各自维护回调时机。
97
+ stateRef.current = nextState;
98
+ setState(nextState);
99
+ optionsRef.current.onProgress?.(nextState);
100
+ }, []);
101
+ const syncProgress = React.useCallback(async (progress, keepPlaying) => {
102
+ // 根据“当前进度 -> 已行驶距离 -> 路径上的点”推导出车辆位置,
103
+ // 再同步决定 smoothMovePath / 相机跟随 / 对外状态。
104
+ if (normalizedPath.length === 0 || totalDistance <= 0 || durationSeconds <= 0) {
105
+ const nextState = {
106
+ ...DEFAULT_STATE,
107
+ currentPosition: normalizedPath[0] ?? null,
108
+ totalDistance,
109
+ durationSeconds,
110
+ };
111
+ publishState(nextState);
112
+ return nextState;
113
+ }
114
+ const clampedProgress = Math.max(0, Math.min(1, progress));
115
+ const traveledDistance = totalDistance * clampedProgress;
116
+ const { angle, point } = getMarkerAngle(normalizedPath, totalDistance, traveledDistance, lastAngleRef.current, optionsRef.current);
117
+ lastAngleRef.current = angle;
118
+ const nextState = {
119
+ isPlaying: keepPlaying,
120
+ isPaused: !keepPlaying && clampedProgress > 0 && clampedProgress < 1,
121
+ progress: clampedProgress,
122
+ currentPosition: point ?? normalizedPath[0] ?? null,
123
+ currentAngle: angle,
124
+ totalDistance,
125
+ traveledDistance,
126
+ durationSeconds,
127
+ smoothMovePath: keepPlaying ? normalizedPath : undefined,
128
+ smoothMoveDuration: keepPlaying ? durationSeconds : undefined,
129
+ };
130
+ publishState(nextState);
131
+ if (optionsRef.current.followCamera === true && nextState.currentPosition && map) {
132
+ await map.moveCamera({
133
+ target: nextState.currentPosition,
134
+ zoom: optionsRef.current.followZoom ?? 17,
135
+ bearing: angle,
136
+ }, optionsRef.current.updateIntervalMs ?? 100);
137
+ }
138
+ return nextState;
139
+ }, [
140
+ durationSeconds,
141
+ map,
142
+ normalizedPath,
143
+ publishState,
144
+ totalDistance,
145
+ ]);
146
+ const runPlaybackTick = React.useCallback(async () => {
147
+ const elapsedMs = Date.now() - startAtRef.current + elapsedBeforePauseRef.current;
148
+ const progress = durationSeconds <= 0 ? 1 : elapsedMs / (durationSeconds * 1000);
149
+ if (progress >= 1) {
150
+ stopTimer();
151
+ elapsedBeforePauseRef.current = 0;
152
+ startAtRef.current = 0;
153
+ const completedState = await syncProgress(1, false);
154
+ optionsRef.current.onComplete?.(completedState);
155
+ return;
156
+ }
157
+ await syncProgress(progress, true);
158
+ }, [durationSeconds, stopTimer, syncProgress]);
159
+ const startTimer = React.useCallback(() => {
160
+ stopTimer();
161
+ const interval = Math.max(optionsRef.current.updateIntervalMs ?? 100, 16);
162
+ timerRef.current = setInterval(() => {
163
+ void runPlaybackTick();
164
+ }, interval);
165
+ }, [runPlaybackTick, stopTimer]);
166
+ const stop = React.useCallback(() => {
167
+ stopTimer();
168
+ elapsedBeforePauseRef.current = 0;
169
+ startAtRef.current = 0;
170
+ lastAngleRef.current = 0;
171
+ publishState({
172
+ ...DEFAULT_STATE,
173
+ currentPosition: normalizedPath[0] ?? null,
174
+ totalDistance,
175
+ durationSeconds,
176
+ });
177
+ }, [durationSeconds, normalizedPath, publishState, stopTimer, totalDistance]);
178
+ const start = React.useCallback(async () => {
179
+ stopTimer();
180
+ elapsedBeforePauseRef.current = 0;
181
+ startAtRef.current = Date.now();
182
+ lastAngleRef.current = 0;
183
+ if (optionsRef.current.autoFit === true && map && normalizedPath.length > 0) {
184
+ await fitCameraToCoordinates(map, normalizedPath, optionsRef.current.fitOptions);
185
+ }
186
+ await syncProgress(0, true);
187
+ startTimer();
188
+ }, [
189
+ map,
190
+ normalizedPath,
191
+ startTimer,
192
+ syncProgress,
193
+ ]);
194
+ const pause = React.useCallback(() => {
195
+ if (!timerRef.current || durationSeconds <= 0) {
196
+ return;
197
+ }
198
+ stopTimer();
199
+ elapsedBeforePauseRef.current += Date.now() - startAtRef.current;
200
+ startAtRef.current = 0;
201
+ const currentState = stateRef.current;
202
+ publishState({
203
+ ...currentState,
204
+ isPlaying: false,
205
+ isPaused: true,
206
+ smoothMovePath: undefined,
207
+ smoothMoveDuration: undefined,
208
+ });
209
+ }, [durationSeconds, publishState, stopTimer]);
210
+ const resume = React.useCallback(async () => {
211
+ const currentState = stateRef.current;
212
+ if (currentState.progress >= 1) {
213
+ await start();
214
+ return;
215
+ }
216
+ startAtRef.current = Date.now();
217
+ publishState({
218
+ ...currentState,
219
+ isPlaying: true,
220
+ isPaused: false,
221
+ smoothMovePath: normalizedPath,
222
+ smoothMoveDuration: durationSeconds,
223
+ });
224
+ startTimer();
225
+ }, [
226
+ durationSeconds,
227
+ normalizedPath,
228
+ publishState,
229
+ start,
230
+ startTimer,
231
+ ]);
232
+ const seek = React.useCallback((progress) => {
233
+ // seek 只重算“已经播放过的时间”,其余状态由 syncProgress 统一刷新。
234
+ const clamped = Math.max(0, Math.min(1, progress));
235
+ elapsedBeforePauseRef.current = clamped * durationSeconds * 1000;
236
+ startAtRef.current = Date.now();
237
+ void syncProgress(clamped, !!timerRef.current);
238
+ }, [durationSeconds, syncProgress]);
239
+ const setSpeedMultiplier = React.useCallback((multiplier) => {
240
+ setSpeedMultiplierState(Math.max(multiplier, 0.1));
241
+ }, []);
242
+ React.useEffect(() => () => stopTimer(), [stopTimer]);
243
+ React.useEffect(() => {
244
+ const pathChanged = previousPathRef.current !== normalizedPath;
245
+ const durationChanged = previousDurationRef.current !== null && previousDurationRef.current !== durationSeconds;
246
+ previousPathRef.current = normalizedPath;
247
+ previousDurationRef.current = durationSeconds;
248
+ if (pathChanged) {
249
+ stopTimer();
250
+ elapsedBeforePauseRef.current = 0;
251
+ startAtRef.current = 0;
252
+ lastAngleRef.current = 0;
253
+ publishState({
254
+ ...DEFAULT_STATE,
255
+ currentPosition: normalizedPath[0] ?? null,
256
+ totalDistance,
257
+ durationSeconds,
258
+ });
259
+ return;
260
+ }
261
+ if (!durationChanged) {
262
+ return;
263
+ }
264
+ const currentState = stateRef.current;
265
+ if (currentState.isPlaying) {
266
+ elapsedBeforePauseRef.current = currentState.progress * durationSeconds * 1000;
267
+ startAtRef.current = Date.now();
268
+ publishState({
269
+ ...currentState,
270
+ totalDistance,
271
+ durationSeconds,
272
+ smoothMoveDuration: durationSeconds,
273
+ });
274
+ startTimer();
275
+ return;
276
+ }
277
+ if (currentState.isPaused) {
278
+ elapsedBeforePauseRef.current = currentState.progress * durationSeconds * 1000;
279
+ startAtRef.current = 0;
280
+ publishState({
281
+ ...currentState,
282
+ totalDistance,
283
+ durationSeconds,
284
+ });
285
+ return;
286
+ }
287
+ publishState({
288
+ ...currentState,
289
+ currentPosition: currentState.progress > 0 ? currentState.currentPosition : normalizedPath[0] ?? null,
290
+ totalDistance,
291
+ durationSeconds,
292
+ });
293
+ }, [durationSeconds, normalizedPath, publishState, startTimer, stopTimer, totalDistance]);
294
+ return {
295
+ ...state,
296
+ totalDistance,
297
+ durationSeconds,
298
+ start: () => {
299
+ void start();
300
+ },
301
+ pause,
302
+ resume: () => {
303
+ void resume();
304
+ },
305
+ stop,
306
+ seek,
307
+ setSpeedMultiplier,
308
+ };
309
+ }
310
+ //# sourceMappingURL=useRoutePlayback.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useRoutePlayback.js","sourceRoot":"","sources":["../../src/hooks/useRoutePlayback.ts"],"names":[],"mappings":"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","sourcesContent":["import * as React from 'react';\n\nimport ExpoGaodeMapModule from '../ExpoGaodeMapModule';\nimport { MapContext } from '../components/MapContext';\nimport type { LatLng } from '../types/common.types';\nimport type {\n RoutePlaybackController,\n RoutePlaybackOptions,\n RoutePlaybackState,\n} from '../types/route-playback.types';\nimport { normalizeLatLngList } from '../utils/GeoUtils';\nimport { fitCameraToCoordinates } from '../utils/RouteUtils';\n\nconst DEFAULT_STATE: RoutePlaybackState = {\n isPlaying: false,\n isPaused: false,\n progress: 0,\n currentPosition: null,\n currentAngle: 0,\n totalDistance: 0,\n traveledDistance: 0,\n durationSeconds: 0,\n};\n\nfunction getMarkerAngle(\n path: LatLng[],\n totalDistance: number,\n targetDistance: number,\n lastAngle: number,\n options: RoutePlaybackOptions\n): { angle: number; point: LatLng | null } {\n // 先取当前距离上的点,再额外预读一个前瞻点,\n // 让车辆朝向在转弯时更自然,不会出现生硬折角。\n const pointInfo = ExpoGaodeMapModule.getPointAtDistance(path, targetDistance);\n if (!pointInfo) {\n return { angle: lastAngle, point: null };\n }\n\n const lookAheadDistance = options.lookAheadDistanceMeters ?? 5;\n const futurePoint = ExpoGaodeMapModule.getPointAtDistance(\n path,\n Math.min(totalDistance, targetDistance + lookAheadDistance)\n );\n\n let targetAngle = pointInfo.angle;\n if (futurePoint && targetDistance + lookAheadDistance < totalDistance) {\n let diffNext = futurePoint.angle - pointInfo.angle;\n if (diffNext > 180) diffNext -= 360;\n if (diffNext < -180) diffNext += 360;\n targetAngle = pointInfo.angle + diffNext * 0.4;\n }\n\n let diff = targetAngle - lastAngle;\n if (diff > 180) diff -= 360;\n if (diff < -180) diff += 360;\n\n const bearingSmoothing = options.bearingSmoothing ?? 0.2;\n return {\n angle: lastAngle + diff * bearingSmoothing,\n point: {\n latitude: pointInfo.latitude,\n longitude: pointInfo.longitude,\n },\n };\n}\n\nexport function useRoutePlayback(\n points: Array<LatLng | [number, number] | number[]>,\n options: RoutePlaybackOptions = {}\n): RoutePlaybackController {\n const map = React.useContext(MapContext);\n const optionsRef = React.useRef(options);\n const [state, setState] = React.useState<RoutePlaybackState>(DEFAULT_STATE);\n const stateRef = React.useRef<RoutePlaybackState>(DEFAULT_STATE);\n const [speedMultiplier, setSpeedMultiplierState] = React.useState(options.speedMultiplier ?? 1);\n const timerRef = React.useRef<ReturnType<typeof setInterval> | null>(null);\n const startAtRef = React.useRef(0);\n const elapsedBeforePauseRef = React.useRef(0);\n const lastAngleRef = React.useRef(0);\n const previousPathRef = React.useRef<LatLng[] | null>(null);\n const previousDurationRef = React.useRef<number | null>(null);\n\n React.useEffect(() => {\n optionsRef.current = options;\n }, [options]);\n\n const normalizedPath = React.useMemo(() => {\n // 长路径可选走一次抽稀,减少轨迹回放期间的计算量。\n const normalized = normalizeLatLngList(points) as LatLng[];\n if (normalized.length <= 2 || !options.simplificationTolerance) {\n return normalized;\n }\n return ExpoGaodeMapModule.simplifyPolyline(normalized, options.simplificationTolerance);\n }, [options.simplificationTolerance, points]);\n\n const totalDistance = React.useMemo(\n () => ExpoGaodeMapModule.calculatePathLength(normalizedPath),\n [normalizedPath]\n );\n\n const durationSeconds = React.useMemo(() => {\n if (options.durationSeconds) {\n return options.durationSeconds;\n }\n\n const baseSpeed = options.baseSpeedMps ?? 15;\n const minDuration = options.minDurationSeconds ?? 5;\n if (totalDistance <= 0) {\n return 0;\n }\n\n return Math.max(minDuration, totalDistance / (baseSpeed * Math.max(speedMultiplier, 0.1)));\n }, [\n options.baseSpeedMps,\n options.durationSeconds,\n options.minDurationSeconds,\n speedMultiplier,\n totalDistance,\n ]);\n\n const stopTimer = React.useCallback(() => {\n if (timerRef.current) {\n clearInterval(timerRef.current);\n timerRef.current = null;\n }\n }, []);\n\n const publishState = React.useCallback(\n (nextState: RoutePlaybackState) => {\n // 统一从这里下发状态,避免不同控制分支各自维护回调时机。\n stateRef.current = nextState;\n setState(nextState);\n optionsRef.current.onProgress?.(nextState);\n },\n []\n );\n\n const syncProgress = React.useCallback(\n async (progress: number, keepPlaying: boolean) => {\n // 根据“当前进度 -> 已行驶距离 -> 路径上的点”推导出车辆位置,\n // 再同步决定 smoothMovePath / 相机跟随 / 对外状态。\n if (normalizedPath.length === 0 || totalDistance <= 0 || durationSeconds <= 0) {\n const nextState = {\n ...DEFAULT_STATE,\n currentPosition: normalizedPath[0] ?? null,\n totalDistance,\n durationSeconds,\n };\n publishState(nextState);\n return nextState;\n }\n\n const clampedProgress = Math.max(0, Math.min(1, progress));\n const traveledDistance = totalDistance * clampedProgress;\n const { angle, point } = getMarkerAngle(\n normalizedPath,\n totalDistance,\n traveledDistance,\n lastAngleRef.current,\n optionsRef.current\n );\n lastAngleRef.current = angle;\n\n const nextState: RoutePlaybackState = {\n isPlaying: keepPlaying,\n isPaused: !keepPlaying && clampedProgress > 0 && clampedProgress < 1,\n progress: clampedProgress,\n currentPosition: point ?? normalizedPath[0] ?? null,\n currentAngle: angle,\n totalDistance,\n traveledDistance,\n durationSeconds,\n smoothMovePath: keepPlaying ? normalizedPath : undefined,\n smoothMoveDuration: keepPlaying ? durationSeconds : undefined,\n };\n\n publishState(nextState);\n\n if (optionsRef.current.followCamera === true && nextState.currentPosition && map) {\n await map.moveCamera(\n {\n target: nextState.currentPosition,\n zoom: optionsRef.current.followZoom ?? 17,\n bearing: angle,\n },\n optionsRef.current.updateIntervalMs ?? 100\n );\n }\n\n return nextState;\n },\n [\n durationSeconds,\n map,\n normalizedPath,\n publishState,\n totalDistance,\n ]\n );\n\n const runPlaybackTick = React.useCallback(async () => {\n const elapsedMs = Date.now() - startAtRef.current + elapsedBeforePauseRef.current;\n const progress = durationSeconds <= 0 ? 1 : elapsedMs / (durationSeconds * 1000);\n\n if (progress >= 1) {\n stopTimer();\n elapsedBeforePauseRef.current = 0;\n startAtRef.current = 0;\n const completedState = await syncProgress(1, false);\n optionsRef.current.onComplete?.(completedState);\n return;\n }\n\n await syncProgress(progress, true);\n }, [durationSeconds, stopTimer, syncProgress]);\n\n const startTimer = React.useCallback(() => {\n stopTimer();\n const interval = Math.max(optionsRef.current.updateIntervalMs ?? 100, 16);\n timerRef.current = setInterval(() => {\n void runPlaybackTick();\n }, interval);\n }, [runPlaybackTick, stopTimer]);\n\n const stop = React.useCallback(() => {\n stopTimer();\n elapsedBeforePauseRef.current = 0;\n startAtRef.current = 0;\n lastAngleRef.current = 0;\n publishState({\n ...DEFAULT_STATE,\n currentPosition: normalizedPath[0] ?? null,\n totalDistance,\n durationSeconds,\n });\n }, [durationSeconds, normalizedPath, publishState, stopTimer, totalDistance]);\n\n const start = React.useCallback(async () => {\n stopTimer();\n elapsedBeforePauseRef.current = 0;\n startAtRef.current = Date.now();\n lastAngleRef.current = 0;\n\n if (optionsRef.current.autoFit === true && map && normalizedPath.length > 0) {\n await fitCameraToCoordinates(map, normalizedPath, optionsRef.current.fitOptions);\n }\n\n await syncProgress(0, true);\n startTimer();\n }, [\n map,\n normalizedPath,\n startTimer,\n syncProgress,\n ]);\n\n const pause = React.useCallback(() => {\n if (!timerRef.current || durationSeconds <= 0) {\n return;\n }\n\n stopTimer();\n elapsedBeforePauseRef.current += Date.now() - startAtRef.current;\n startAtRef.current = 0;\n const currentState = stateRef.current;\n publishState({\n ...currentState,\n isPlaying: false,\n isPaused: true,\n smoothMovePath: undefined,\n smoothMoveDuration: undefined,\n });\n }, [durationSeconds, publishState, stopTimer]);\n\n const resume = React.useCallback(async () => {\n const currentState = stateRef.current;\n if (currentState.progress >= 1) {\n await start();\n return;\n }\n\n startAtRef.current = Date.now();\n publishState({\n ...currentState,\n isPlaying: true,\n isPaused: false,\n smoothMovePath: normalizedPath,\n smoothMoveDuration: durationSeconds,\n });\n startTimer();\n }, [\n durationSeconds,\n normalizedPath,\n publishState,\n start,\n startTimer,\n ]);\n\n const seek = React.useCallback(\n (progress: number) => {\n // seek 只重算“已经播放过的时间”,其余状态由 syncProgress 统一刷新。\n const clamped = Math.max(0, Math.min(1, progress));\n elapsedBeforePauseRef.current = clamped * durationSeconds * 1000;\n startAtRef.current = Date.now();\n void syncProgress(clamped, !!timerRef.current);\n },\n [durationSeconds, syncProgress]\n );\n\n const setSpeedMultiplier = React.useCallback((multiplier: number) => {\n setSpeedMultiplierState(Math.max(multiplier, 0.1));\n }, []);\n\n React.useEffect(() => () => stopTimer(), [stopTimer]);\n\n React.useEffect(() => {\n const pathChanged = previousPathRef.current !== normalizedPath;\n const durationChanged =\n previousDurationRef.current !== null && previousDurationRef.current !== durationSeconds;\n\n previousPathRef.current = normalizedPath;\n previousDurationRef.current = durationSeconds;\n\n if (pathChanged) {\n stopTimer();\n elapsedBeforePauseRef.current = 0;\n startAtRef.current = 0;\n lastAngleRef.current = 0;\n publishState({\n ...DEFAULT_STATE,\n currentPosition: normalizedPath[0] ?? null,\n totalDistance,\n durationSeconds,\n });\n return;\n }\n\n if (!durationChanged) {\n return;\n }\n\n const currentState = stateRef.current;\n\n if (currentState.isPlaying) {\n elapsedBeforePauseRef.current = currentState.progress * durationSeconds * 1000;\n startAtRef.current = Date.now();\n publishState({\n ...currentState,\n totalDistance,\n durationSeconds,\n smoothMoveDuration: durationSeconds,\n });\n startTimer();\n return;\n }\n\n if (currentState.isPaused) {\n elapsedBeforePauseRef.current = currentState.progress * durationSeconds * 1000;\n startAtRef.current = 0;\n publishState({\n ...currentState,\n totalDistance,\n durationSeconds,\n });\n return;\n }\n\n publishState({\n ...currentState,\n currentPosition:\n currentState.progress > 0 ? currentState.currentPosition : normalizedPath[0] ?? null,\n totalDistance,\n durationSeconds,\n });\n }, [durationSeconds, normalizedPath, publishState, startTimer, stopTimer, totalDistance]);\n\n return {\n ...state,\n totalDistance,\n durationSeconds,\n start: () => {\n void start();\n },\n pause,\n resume: () => {\n void resume();\n },\n stop,\n seek,\n setSpeedMultiplier,\n };\n}\n"]}
package/build/index.d.ts CHANGED
@@ -3,6 +3,9 @@ export { default as ExpoGaodeMapModule } from './ExpoGaodeMapModule';
3
3
  export { default as MapView } from './ExpoGaodeMapView';
4
4
  export { useMap } from './components/MapContext';
5
5
  export { MapUI } from './components/MapUI';
6
+ export { RouteOverlay } from './components/RouteOverlay';
7
+ export { AreaMaskOverlay } from './components/AreaMaskOverlay';
8
+ export { useRoutePlayback } from './hooks/useRoutePlayback';
6
9
  export { Marker, Polyline, Polygon, Circle, HeatMap, MultiPoint, Cluster, } from './components/overlays';
7
10
  export { ErrorHandler, ErrorLogger, GaodeMapError, ErrorType, } from './utils/ErrorHandler';
8
11
  export type { ErrorDetails } from './utils/ErrorHandler';
@@ -25,5 +28,6 @@ export type { OfflineMapInfo, OfflineMapStatus, OfflineMapDownloadConfig, Offlin
25
28
  */
26
29
  export declare const useLocationPermissions: (options?: import("expo-modules-core").PermissionHookOptions<object> | undefined) => [import("./types").PermissionStatus | null, () => Promise<import("./types").PermissionStatus>, () => Promise<import("./types").PermissionStatus>];
27
30
  export { getSDKConfig, getWebKey } from './ExpoGaodeMapModule';
31
+ export { buildLatLngBounds, fitCameraToCoordinates, getRouteBounds, parseMultiRingPolyline, } from './utils/RouteUtils';
28
32
  export { default } from './ExpoGaodeMapModule';
29
33
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAKA,cAAc,SAAS,CAAC;AAExB,OAAO,EAAE,OAAO,IAAI,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAGrE,OAAO,EAAE,OAAO,IAAI,OAAO,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAC;AACjD,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAG3C,OAAO,EACL,MAAM,EACN,QAAQ,EACR,OAAO,EACP,MAAM,EACN,OAAO,EACP,UAAU,EACV,OAAO,GACR,MAAM,uBAAuB,CAAC;AAG/B,OAAO,EACL,YAAY,EACZ,WAAW,EACX,aAAa,EACb,SAAS,GACV,MAAM,sBAAsB,CAAC;AAC9B,YAAY,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AAGzD,OAAO,EACL,gBAAgB,EAChB,UAAU,EACV,SAAS,EACT,eAAe,EACf,WAAW,EACX,QAAQ,EACR,UAAU,EACV,MAAM,GACP,MAAM,0BAA0B,CAAC;AAClC,YAAY,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AAG1E,OAAO,EACL,eAAe,EACf,iBAAiB,EAAE,UAAU;AAC7B,sBAAsB,GACvB,MAAM,yBAAyB,CAAC;AAGjC,OAAO,EACL,eAAe,EACf,cAAc,GACf,MAAM,8BAA8B,CAAC;AACtC,YAAY,EACV,oBAAoB,EACpB,cAAc,GACf,MAAM,8BAA8B,CAAC;AAGtC,OAAO,EAAE,OAAO,IAAI,yBAAyB,EAAE,MAAM,2BAA2B,CAAC;AAEjF,YAAY,EACV,cAAc,EACd,gBAAgB,EAChB,wBAAwB,EACxB,uBAAuB,EACvB,uBAAuB,EACvB,oBAAoB,EACpB,qBAAqB,EACrB,wBAAwB,EACxB,qBAAqB,EACrB,gBAAgB,GACjB,MAAM,uBAAuB,CAAC;AAM/B;;;;;;;;GAQG;AACH,eAAO,MAAM,sBAAsB,wOAGjC,CAAA;AAGF,OAAO,EAAE,YAAY,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AAG/D,OAAO,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAKA,cAAc,SAAS,CAAC;AAExB,OAAO,EAAE,OAAO,IAAI,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAGrE,OAAO,EAAE,OAAO,IAAI,OAAO,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAC;AACjD,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAC/D,OAAO,EAAE,gBAAgB,EAAE,MAAM,0BAA0B,CAAC;AAG5D,OAAO,EACL,MAAM,EACN,QAAQ,EACR,OAAO,EACP,MAAM,EACN,OAAO,EACP,UAAU,EACV,OAAO,GACR,MAAM,uBAAuB,CAAC;AAG/B,OAAO,EACL,YAAY,EACZ,WAAW,EACX,aAAa,EACb,SAAS,GACV,MAAM,sBAAsB,CAAC;AAC9B,YAAY,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AAGzD,OAAO,EACL,gBAAgB,EAChB,UAAU,EACV,SAAS,EACT,eAAe,EACf,WAAW,EACX,QAAQ,EACR,UAAU,EACV,MAAM,GACP,MAAM,0BAA0B,CAAC;AAClC,YAAY,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AAG1E,OAAO,EACL,eAAe,EACf,iBAAiB,EAAE,UAAU;AAC7B,sBAAsB,GACvB,MAAM,yBAAyB,CAAC;AAGjC,OAAO,EACL,eAAe,EACf,cAAc,GACf,MAAM,8BAA8B,CAAC;AACtC,YAAY,EACV,oBAAoB,EACpB,cAAc,GACf,MAAM,8BAA8B,CAAC;AAGtC,OAAO,EAAE,OAAO,IAAI,yBAAyB,EAAE,MAAM,2BAA2B,CAAC;AAEjF,YAAY,EACV,cAAc,EACd,gBAAgB,EAChB,wBAAwB,EACxB,uBAAuB,EACvB,uBAAuB,EACvB,oBAAoB,EACpB,qBAAqB,EACrB,wBAAwB,EACxB,qBAAqB,EACrB,gBAAgB,GACjB,MAAM,uBAAuB,CAAC;AAM/B;;;;;;;;GAQG;AACH,eAAO,MAAM,sBAAsB,wOAGjC,CAAA;AAGF,OAAO,EAAE,YAAY,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EACL,iBAAiB,EACjB,sBAAsB,EACtB,cAAc,EACd,sBAAsB,GACvB,MAAM,oBAAoB,CAAC;AAG5B,OAAO,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAC"}
package/build/index.js CHANGED
@@ -8,6 +8,9 @@ export { default as ExpoGaodeMapModule } from './ExpoGaodeMapModule';
8
8
  export { default as MapView } from './ExpoGaodeMapView';
9
9
  export { useMap } from './components/MapContext';
10
10
  export { MapUI } from './components/MapUI';
11
+ export { RouteOverlay } from './components/RouteOverlay';
12
+ export { AreaMaskOverlay } from './components/AreaMaskOverlay';
13
+ export { useRoutePlayback } from './hooks/useRoutePlayback';
11
14
  // 导出覆盖物组件
12
15
  export { Marker, Polyline, Polygon, Circle, HeatMap, MultiPoint, Cluster, } from './components/overlays';
13
16
  // 导出错误处理工具
@@ -38,6 +41,7 @@ export const useLocationPermissions = createPermissionHook({
38
41
  });
39
42
  // 导出便捷读取的 SDK 配置与 webKey
40
43
  export { getSDKConfig, getWebKey } from './ExpoGaodeMapModule';
44
+ export { buildLatLngBounds, fitCameraToCoordinates, getRouteBounds, parseMultiRingPolyline, } from './utils/RouteUtils';
41
45
  // 默认导出原生模块
42
46
  export { default } from './ExpoGaodeMapModule';
43
47
  //# sourceMappingURL=index.js.map