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

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.
@@ -1 +1 @@
1
- {"version":3,"file":"FoldableMapView.d.ts","sourceRoot":"","sources":["../../src/components/FoldableMapView.tsx"],"names":[],"mappings":"AAAA,OAAO,KAA+C,MAAM,OAAO,CAAC;AAIpE,OAAO,EAAoB,UAAU,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AACpF,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,UAAU,CAAC;AAEpD;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,oBAAoB;IACpB,cAAc,CAAC,EAAE,OAAO,CAAC;IACzB,iBAAiB;IACjB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,qBAAqB;IACrB,gBAAgB,CAAC,EAAE,OAAO,CAAC;IAC3B,eAAe;IACf,iBAAiB,CAAC,EAAE,CAAC,KAAK,EAAE,SAAS,EAAE,UAAU,EAAE,UAAU,KAAK,IAAI,CAAC;IACvE,eAAe;IACf,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB;AAED;;;;GAIG;AACH,MAAM,WAAW,oBAAqB,SAAQ,YAAY;IACxD,cAAc;IACd,cAAc,CAAC,EAAE,cAAc,CAAC;CACjC;AA2ED,eAAO,MAAM,eAAe,EAAE,KAAK,CAAC,EAAE,CAAC,oBAAoB,CA0F1D,CAAC;AAEF;;;;GAIG;AACH,wBAAgB,cAAc,CAC5B,MAAM,EAAE,KAAK,CAAC,SAAS,CAAC,UAAU,CAAC,EACnC,MAAM,CAAC,EAAE,cAAc;;;;EA8DxB"}
1
+ {"version":3,"file":"FoldableMapView.d.ts","sourceRoot":"","sources":["../../src/components/FoldableMapView.tsx"],"names":[],"mappings":"AAAA,OAAO,KAA+C,MAAM,OAAO,CAAC;AAKpE,OAAO,EAAoB,UAAU,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AACpF,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,UAAU,CAAC;AAEpD;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,oBAAoB;IACpB,cAAc,CAAC,EAAE,OAAO,CAAC;IACzB,iBAAiB;IACjB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,qBAAqB;IACrB,gBAAgB,CAAC,EAAE,OAAO,CAAC;IAC3B,eAAe;IACf,iBAAiB,CAAC,EAAE,CAAC,KAAK,EAAE,SAAS,EAAE,UAAU,EAAE,UAAU,KAAK,IAAI,CAAC;IACvE,eAAe;IACf,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB;AAED;;;;GAIG;AACH,MAAM,WAAW,oBAAqB,SAAQ,YAAY;IACxD,cAAc;IACd,cAAc,CAAC,EAAE,cAAc,CAAC;CACjC;AA2ED,eAAO,MAAM,eAAe,EAAE,KAAK,CAAC,EAAE,CAAC,oBAAoB,CA0F1D,CAAC;AAEF;;;;GAIG;AACH,wBAAgB,cAAc,CAC5B,MAAM,EAAE,KAAK,CAAC,SAAS,CAAC,UAAU,CAAC,EACnC,MAAM,CAAC,EAAE,cAAc;;;;EA8DxB"}
@@ -1,6 +1,7 @@
1
1
  import React, { useEffect, useMemo, useRef, useState } from 'react';
2
2
  import { Platform } from 'react-native';
3
3
  import ExpoGaodeMapView from '../ExpoGaodeMapView';
4
+ import { useEventCallback } from '../hooks/useEventCallback';
4
5
  import { PlatformDetector, FoldState } from '../utils/PlatformDetector';
5
6
  const DEFAULT_FOLDABLE_CONFIG = {
6
7
  autoAdjustZoom: true,
@@ -64,7 +65,7 @@ export const FoldableMapView = ({ foldableConfig, ...mapProps }) => {
64
65
  const [deviceInfo, setDeviceInfo] = useState(PlatformDetector.getDeviceInfo());
65
66
  const config = useMemo(() => createFoldableConfig(foldableConfig), [foldableConfig]);
66
67
  const foldStateRef = useRef(currentFoldState);
67
- const handleFoldStateChange = React.useEffectEvent(async (newInfo, debugPrefix) => {
68
+ const handleFoldStateChange = useEventCallback(async (newInfo, debugPrefix) => {
68
69
  const newFoldState = PlatformDetector.getFoldState();
69
70
  const previousFoldState = foldStateRef.current;
70
71
  if (config.debug) {
@@ -117,7 +118,7 @@ export const FoldableMapView = ({ foldableConfig, ...mapProps }) => {
117
118
  return () => {
118
119
  removeListener();
119
120
  };
120
- }, [config.debug, deviceInfo.isFoldable]);
121
+ }, [config.debug, deviceInfo.isFoldable, handleFoldStateChange]);
121
122
  return (<ExpoGaodeMapView ref={mapRef} {...mapProps}/>);
122
123
  };
123
124
  /**
@@ -133,7 +134,7 @@ export function useFoldableMap(mapRef, config) {
133
134
  useEffect(() => {
134
135
  foldStateRef.current = foldState;
135
136
  }, [foldState]);
136
- const handleFoldStateChange = React.useEffectEvent(async (newInfo) => {
137
+ const handleFoldStateChange = useEventCallback(async (newInfo) => {
137
138
  const newFoldState = PlatformDetector.getFoldState();
138
139
  const previousFoldState = foldStateRef.current;
139
140
  if (newFoldState !== previousFoldState && previousFoldState !== FoldState.UNKNOWN) {
@@ -164,7 +165,7 @@ export function useFoldableMap(mapRef, config) {
164
165
  return () => {
165
166
  removeListener();
166
167
  };
167
- }, [deviceInfo.isFoldable, mapRef]);
168
+ }, [deviceInfo.isFoldable, handleFoldStateChange, mapRef]);
168
169
  return {
169
170
  foldState,
170
171
  deviceInfo,
@@ -1 +1 @@
1
- {"version":3,"file":"FoldableMapView.js","sourceRoot":"","sources":["../../src/components/FoldableMapView.tsx"],"names":[],"mappings":"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","sourcesContent":["import React, { useEffect, useMemo, useRef, useState } from 'react';\nimport { Platform } from 'react-native';\nimport ExpoGaodeMapView from '../ExpoGaodeMapView';\n\nimport { PlatformDetector, DeviceInfo, FoldState } from '../utils/PlatformDetector';\nimport { MapViewProps, MapViewRef } from '../types';\n\n/**\n * 折叠屏适配配置\n */\nexport interface FoldableConfig {\n /** 折叠时是否自动调整缩放级别 */\n autoAdjustZoom?: boolean;\n /** 展开时的缩放级别增量 */\n unfoldedZoomDelta?: number;\n /** 是否在折叠/展开时保持中心点 */\n keepCenterOnFold?: boolean;\n /** 折叠状态变化回调 */\n onFoldStateChange?: (state: FoldState, deviceInfo: DeviceInfo) => void;\n /** 是否启用调试日志 */\n debug?: boolean;\n}\n\n/**\n * 折叠屏地图视图组件\n * \n * 自动适配折叠屏设备的展开/折叠状态变化\n */\nexport interface FoldableMapViewProps extends MapViewProps {\n /** 折叠屏适配配置 */\n foldableConfig?: FoldableConfig;\n}\n\nconst DEFAULT_FOLDABLE_CONFIG: Required<FoldableConfig> = {\n autoAdjustZoom: true,\n unfoldedZoomDelta: 1,\n keepCenterOnFold: true,\n onFoldStateChange: () => {},\n debug: false,\n};\n\nfunction clampZoomLevel(zoom: number): number {\n return Math.max(3, Math.min(20, zoom));\n}\n\nfunction createFoldableConfig(config?: FoldableConfig): Required<FoldableConfig> {\n return {\n ...DEFAULT_FOLDABLE_CONFIG,\n ...config,\n };\n}\n\nasync function applyFoldStateCameraAdjustment(\n mapRef: React.RefObject<MapViewRef | null>,\n oldState: FoldState,\n newState: FoldState,\n config: Required<FoldableConfig>,\n debugPrefix: 'FoldableMapView' | 'useFoldableMap'\n): Promise<void> {\n if (!mapRef.current || !config.autoAdjustZoom) {\n return;\n }\n\n const currentCamera = await mapRef.current.getCameraPosition();\n if (!currentCamera) {\n if (config.debug) {\n console.warn(`[${debugPrefix}] 无法获取相机位置`);\n }\n return;\n }\n\n const isUnfolding = newState === FoldState.UNFOLDED && oldState === FoldState.FOLDED;\n const isFolding = newState === FoldState.FOLDED && oldState === FoldState.UNFOLDED;\n\n if (config.debug) {\n console.log(`[${debugPrefix}] 折叠状态变化:`, {\n oldState,\n newState,\n isUnfolding,\n isFolding,\n currentZoom: currentCamera.zoom,\n });\n }\n\n if (!isUnfolding && !isFolding) {\n return;\n }\n\n const currentZoom = currentCamera.zoom ?? 15;\n const zoomDelta = isUnfolding ? config.unfoldedZoomDelta : -config.unfoldedZoomDelta;\n const nextZoom = clampZoomLevel(currentZoom + zoomDelta);\n\n if (config.debug) {\n console.log(`[${debugPrefix}] 调整缩放:`, {\n oldZoom: currentZoom,\n newZoom: nextZoom,\n delta: zoomDelta,\n });\n }\n\n await mapRef.current.moveCamera({\n target: config.keepCenterOnFold ? currentCamera.target : undefined,\n zoom: nextZoom,\n }, 300);\n}\n\nexport const FoldableMapView: React.FC<FoldableMapViewProps> = ({\n foldableConfig,\n ...mapProps\n}) => {\n const mapRef = useRef<MapViewRef>(null);\n const [currentFoldState, setCurrentFoldState] = useState<FoldState>(FoldState.UNKNOWN);\n const [deviceInfo, setDeviceInfo] = useState<DeviceInfo>(PlatformDetector.getDeviceInfo());\n const config = useMemo(() => createFoldableConfig(foldableConfig), [foldableConfig]);\n const foldStateRef = useRef(currentFoldState);\n\n const handleFoldStateChange = React.useEffectEvent(\n async (newInfo: DeviceInfo, debugPrefix: 'FoldableMapView' | 'useFoldableMap') => {\n const newFoldState = PlatformDetector.getFoldState();\n const previousFoldState = foldStateRef.current;\n\n if (config.debug) {\n console.log(`[${debugPrefix}] 屏幕尺寸变化`);\n console.log('新设备信息:', newInfo);\n console.log('新折叠状态:', newFoldState);\n }\n\n if (newFoldState !== previousFoldState && previousFoldState !== FoldState.UNKNOWN) {\n try {\n await applyFoldStateCameraAdjustment(\n mapRef,\n previousFoldState,\n newFoldState,\n config,\n debugPrefix\n );\n } catch (error) {\n if (config.debug) {\n console.error(`[${debugPrefix}] 处理折叠状态变化失败:`, error);\n }\n }\n }\n\n foldStateRef.current = newFoldState;\n setCurrentFoldState(newFoldState);\n setDeviceInfo(newInfo);\n config.onFoldStateChange(newFoldState, newInfo);\n }\n );\n\n useEffect(() => {\n foldStateRef.current = currentFoldState;\n }, [currentFoldState]);\n\n useEffect(() => {\n const latestDeviceInfo = PlatformDetector.getDeviceInfo();\n setDeviceInfo(latestDeviceInfo);\n }, []);\n\n useEffect(() => {\n // 仅在 Android 折叠屏设备上启用\n if (Platform.OS !== 'android' || !deviceInfo.isFoldable) {\n if (config.debug) {\n console.log('[FoldableMapView] 非折叠屏设备,跳过适配');\n }\n return;\n }\n\n const initialState = PlatformDetector.getFoldState();\n foldStateRef.current = initialState;\n setCurrentFoldState(initialState);\n\n if (config.debug) {\n console.log('[FoldableMapView] 初始化折叠屏适配');\n console.log('设备信息:', deviceInfo);\n console.log('初始折叠状态:', initialState);\n }\n\n // 监听屏幕尺寸变化\n const removeListener = PlatformDetector.addDimensionChangeListener(\n async (newInfo: DeviceInfo) => {\n await handleFoldStateChange(newInfo, 'FoldableMapView');\n }\n );\n\n return () => {\n removeListener();\n };\n }, [config.debug, deviceInfo.isFoldable]);\n\n return (\n <ExpoGaodeMapView\n ref={mapRef}\n {...mapProps}\n />\n );\n};\n\n/**\n * 折叠屏适配 Hook\n * \n * 用于在现有地图组件中添加折叠屏适配功能\n */\nexport function useFoldableMap(\n mapRef: React.RefObject<MapViewRef>,\n config?: FoldableConfig\n) {\n const [foldState, setFoldState] = useState<FoldState>(FoldState.UNKNOWN);\n const [deviceInfo, setDeviceInfo] = useState<DeviceInfo>(PlatformDetector.getDeviceInfo());\n const mergedConfig = useMemo(() => createFoldableConfig(config), [config]);\n const foldStateRef = useRef(foldState);\n\n useEffect(() => {\n foldStateRef.current = foldState;\n }, [foldState]);\n\n const handleFoldStateChange = React.useEffectEvent(async (newInfo: DeviceInfo) => {\n const newFoldState = PlatformDetector.getFoldState();\n const previousFoldState = foldStateRef.current;\n\n if (newFoldState !== previousFoldState && previousFoldState !== FoldState.UNKNOWN) {\n try {\n await applyFoldStateCameraAdjustment(\n mapRef,\n previousFoldState,\n newFoldState,\n mergedConfig,\n 'useFoldableMap'\n );\n } catch (error) {\n if (mergedConfig.debug) {\n console.error('[useFoldableMap] 调整失败:', error);\n }\n }\n }\n\n foldStateRef.current = newFoldState;\n setFoldState(newFoldState);\n setDeviceInfo(newInfo);\n mergedConfig.onFoldStateChange(newFoldState, newInfo);\n });\n\n useEffect(() => {\n if (Platform.OS !== 'android' || !deviceInfo.isFoldable) {\n return;\n }\n\n const initialState = PlatformDetector.getFoldState();\n foldStateRef.current = initialState;\n setFoldState(initialState);\n\n const removeListener = PlatformDetector.addDimensionChangeListener(\n async (newInfo: DeviceInfo) => {\n await handleFoldStateChange(newInfo);\n }\n );\n\n return () => {\n removeListener();\n };\n }, [deviceInfo.isFoldable, mapRef]);\n\n return {\n foldState,\n deviceInfo,\n isFoldable: deviceInfo.isFoldable,\n };\n}\n"]}
1
+ {"version":3,"file":"FoldableMapView.js","sourceRoot":"","sources":["../../src/components/FoldableMapView.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EAAE,SAAS,EAAE,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AACpE,OAAO,EAAE,QAAQ,EAAE,MAAM,cAAc,CAAC;AACxC,OAAO,gBAAgB,MAAM,qBAAqB,CAAC;AAEnD,OAAO,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC7D,OAAO,EAAE,gBAAgB,EAAc,SAAS,EAAE,MAAM,2BAA2B,CAAC;AA6BpF,MAAM,uBAAuB,GAA6B;IACxD,cAAc,EAAE,IAAI;IACpB,iBAAiB,EAAE,CAAC;IACpB,gBAAgB,EAAE,IAAI;IACtB,iBAAiB,EAAE,GAAG,EAAE,GAAE,CAAC;IAC3B,KAAK,EAAE,KAAK;CACb,CAAC;AAEF,SAAS,cAAc,CAAC,IAAY;IAClC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC,CAAC;AACzC,CAAC;AAED,SAAS,oBAAoB,CAAC,MAAuB;IACnD,OAAO;QACL,GAAG,uBAAuB;QAC1B,GAAG,MAAM;KACV,CAAC;AACJ,CAAC;AAED,KAAK,UAAU,8BAA8B,CAC3C,MAA0C,EAC1C,QAAmB,EACnB,QAAmB,EACnB,MAAgC,EAChC,WAAiD;IAEjD,IAAI,CAAC,MAAM,CAAC,OAAO,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC9C,OAAO;IACT,CAAC;IAED,MAAM,aAAa,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC,iBAAiB,EAAE,CAAC;IAC/D,IAAI,CAAC,aAAa,EAAE,CAAC;QACnB,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,OAAO,CAAC,IAAI,CAAC,IAAI,WAAW,YAAY,CAAC,CAAC;QAC5C,CAAC;QACD,OAAO;IACT,CAAC;IAED,MAAM,WAAW,GAAG,QAAQ,KAAK,SAAS,CAAC,QAAQ,IAAI,QAAQ,KAAK,SAAS,CAAC,MAAM,CAAC;IACrF,MAAM,SAAS,GAAG,QAAQ,KAAK,SAAS,CAAC,MAAM,IAAI,QAAQ,KAAK,SAAS,CAAC,QAAQ,CAAC;IAEnF,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;QACjB,OAAO,CAAC,GAAG,CAAC,IAAI,WAAW,WAAW,EAAE;YACtC,QAAQ;YACR,QAAQ;YACR,WAAW;YACX,SAAS;YACT,WAAW,EAAE,aAAa,CAAC,IAAI;SAChC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,CAAC,WAAW,IAAI,CAAC,SAAS,EAAE,CAAC;QAC/B,OAAO;IACT,CAAC;IAED,MAAM,WAAW,GAAG,aAAa,CAAC,IAAI,IAAI,EAAE,CAAC;IAC7C,MAAM,SAAS,GAAG,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,iBAAiB,CAAC;IACrF,MAAM,QAAQ,GAAG,cAAc,CAAC,WAAW,GAAG,SAAS,CAAC,CAAC;IAEzD,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;QACjB,OAAO,CAAC,GAAG,CAAC,IAAI,WAAW,SAAS,EAAE;YACpC,OAAO,EAAE,WAAW;YACpB,OAAO,EAAE,QAAQ;YACjB,KAAK,EAAE,SAAS;SACjB,CAAC,CAAC;IACL,CAAC;IAED,MAAM,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC;QAC9B,MAAM,EAAE,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS;QAClE,IAAI,EAAE,QAAQ;KACf,EAAE,GAAG,CAAC,CAAC;AACV,CAAC;AAED,MAAM,CAAC,MAAM,eAAe,GAAmC,CAAC,EAC9D,cAAc,EACd,GAAG,QAAQ,EACZ,EAAE,EAAE;IACH,MAAM,MAAM,GAAG,MAAM,CAAa,IAAI,CAAC,CAAC;IACxC,MAAM,CAAC,gBAAgB,EAAE,mBAAmB,CAAC,GAAG,QAAQ,CAAY,SAAS,CAAC,OAAO,CAAC,CAAC;IACvF,MAAM,CAAC,UAAU,EAAE,aAAa,CAAC,GAAG,QAAQ,CAAa,gBAAgB,CAAC,aAAa,EAAE,CAAC,CAAC;IAC3F,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,oBAAoB,CAAC,cAAc,CAAC,EAAE,CAAC,cAAc,CAAC,CAAC,CAAC;IACrF,MAAM,YAAY,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;IAE9C,MAAM,qBAAqB,GAAG,gBAAgB,CAC5C,KAAK,EAAE,OAAmB,EAAE,WAAiD,EAAE,EAAE;QAC/E,MAAM,YAAY,GAAG,gBAAgB,CAAC,YAAY,EAAE,CAAC;QACrD,MAAM,iBAAiB,GAAG,YAAY,CAAC,OAAO,CAAC;QAE/C,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,IAAI,WAAW,UAAU,CAAC,CAAC;YACvC,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YAC/B,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;QACtC,CAAC;QAED,IAAI,YAAY,KAAK,iBAAiB,IAAI,iBAAiB,KAAK,SAAS,CAAC,OAAO,EAAE,CAAC;YAClF,IAAI,CAAC;gBACH,MAAM,8BAA8B,CAClC,MAAM,EACN,iBAAiB,EACjB,YAAY,EACZ,MAAM,EACN,WAAW,CACZ,CAAC;YACJ,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;oBACjB,OAAO,CAAC,KAAK,CAAC,IAAI,WAAW,eAAe,EAAE,KAAK,CAAC,CAAC;gBACvD,CAAC;YACH,CAAC;QACH,CAAC;QAED,YAAY,CAAC,OAAO,GAAG,YAAY,CAAC;QACpC,mBAAmB,CAAC,YAAY,CAAC,CAAC;QAClC,aAAa,CAAC,OAAO,CAAC,CAAC;QACvB,MAAM,CAAC,iBAAiB,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;IAClD,CAAC,CACF,CAAC;IAEF,SAAS,CAAC,GAAG,EAAE;QACb,YAAY,CAAC,OAAO,GAAG,gBAAgB,CAAC;IAC1C,CAAC,EAAE,CAAC,gBAAgB,CAAC,CAAC,CAAC;IAEvB,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,gBAAgB,GAAG,gBAAgB,CAAC,aAAa,EAAE,CAAC;QAC1D,aAAa,CAAC,gBAAgB,CAAC,CAAC;IAClC,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,SAAS,CAAC,GAAG,EAAE;QACb,sBAAsB;QACtB,IAAI,QAAQ,CAAC,EAAE,KAAK,SAAS,IAAI,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;YACxD,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;gBACjB,OAAO,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAC;YAC/C,CAAC;YACD,OAAO;QACT,CAAC;QAED,MAAM,YAAY,GAAG,gBAAgB,CAAC,YAAY,EAAE,CAAC;QACrD,YAAY,CAAC,OAAO,GAAG,YAAY,CAAC;QACpC,mBAAmB,CAAC,YAAY,CAAC,CAAC;QAElC,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;YAC1C,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;YACjC,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QACvC,CAAC;QAED,WAAW;QACX,MAAM,cAAc,GAAG,gBAAgB,CAAC,0BAA0B,CAChE,KAAK,EAAE,OAAmB,EAAE,EAAE;YAC5B,MAAM,qBAAqB,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC;QAC1D,CAAC,CACF,CAAC;QAEF,OAAO,GAAG,EAAE;YACV,cAAc,EAAE,CAAC;QACnB,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE,UAAU,CAAC,UAAU,EAAE,qBAAqB,CAAC,CAAC,CAAC;IAEjE,OAAO,CACL,CAAC,gBAAgB,CACf,GAAG,CAAC,CAAC,MAAM,CAAC,CACZ,IAAI,QAAQ,CAAC,EACb,CACH,CAAC;AACJ,CAAC,CAAC;AAEF;;;;GAIG;AACH,MAAM,UAAU,cAAc,CAC5B,MAAmC,EACnC,MAAuB;IAEvB,MAAM,CAAC,SAAS,EAAE,YAAY,CAAC,GAAG,QAAQ,CAAY,SAAS,CAAC,OAAO,CAAC,CAAC;IACzE,MAAM,CAAC,UAAU,EAAE,aAAa,CAAC,GAAG,QAAQ,CAAa,gBAAgB,CAAC,aAAa,EAAE,CAAC,CAAC;IAC3F,MAAM,YAAY,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,oBAAoB,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC;IAC3E,MAAM,YAAY,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC;IAEvC,SAAS,CAAC,GAAG,EAAE;QACb,YAAY,CAAC,OAAO,GAAG,SAAS,CAAC;IACnC,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC;IAEhB,MAAM,qBAAqB,GAAG,gBAAgB,CAAC,KAAK,EAAE,OAAmB,EAAE,EAAE;QAC3E,MAAM,YAAY,GAAG,gBAAgB,CAAC,YAAY,EAAE,CAAC;QACrD,MAAM,iBAAiB,GAAG,YAAY,CAAC,OAAO,CAAC;QAE/C,IAAI,YAAY,KAAK,iBAAiB,IAAI,iBAAiB,KAAK,SAAS,CAAC,OAAO,EAAE,CAAC;YAClF,IAAI,CAAC;gBACH,MAAM,8BAA8B,CAClC,MAAM,EACN,iBAAiB,EACjB,YAAY,EACZ,YAAY,EACZ,gBAAgB,CACjB,CAAC;YACJ,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,IAAI,YAAY,CAAC,KAAK,EAAE,CAAC;oBACvB,OAAO,CAAC,KAAK,CAAC,wBAAwB,EAAE,KAAK,CAAC,CAAC;gBACjD,CAAC;YACH,CAAC;QACH,CAAC;QAED,YAAY,CAAC,OAAO,GAAG,YAAY,CAAC;QACpC,YAAY,CAAC,YAAY,CAAC,CAAC;QAC3B,aAAa,CAAC,OAAO,CAAC,CAAC;QACvB,YAAY,CAAC,iBAAiB,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;IACxD,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,QAAQ,CAAC,EAAE,KAAK,SAAS,IAAI,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;YACxD,OAAO;QACT,CAAC;QAED,MAAM,YAAY,GAAG,gBAAgB,CAAC,YAAY,EAAE,CAAC;QACrD,YAAY,CAAC,OAAO,GAAG,YAAY,CAAC;QACpC,YAAY,CAAC,YAAY,CAAC,CAAC;QAE3B,MAAM,cAAc,GAAG,gBAAgB,CAAC,0BAA0B,CAChE,KAAK,EAAE,OAAmB,EAAE,EAAE;YAC5B,MAAM,qBAAqB,CAAC,OAAO,CAAC,CAAC;QACvC,CAAC,CACF,CAAC;QAEF,OAAO,GAAG,EAAE;YACV,cAAc,EAAE,CAAC;QACnB,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,EAAE,qBAAqB,EAAE,MAAM,CAAC,CAAC,CAAC;IAE3D,OAAO;QACL,SAAS;QACT,UAAU;QACV,UAAU,EAAE,UAAU,CAAC,UAAU;KAClC,CAAC;AACJ,CAAC","sourcesContent":["import React, { useEffect, useMemo, useRef, useState } from 'react';\nimport { Platform } from 'react-native';\nimport ExpoGaodeMapView from '../ExpoGaodeMapView';\n\nimport { useEventCallback } from '../hooks/useEventCallback';\nimport { PlatformDetector, DeviceInfo, FoldState } from '../utils/PlatformDetector';\nimport { MapViewProps, MapViewRef } from '../types';\n\n/**\n * 折叠屏适配配置\n */\nexport interface FoldableConfig {\n /** 折叠时是否自动调整缩放级别 */\n autoAdjustZoom?: boolean;\n /** 展开时的缩放级别增量 */\n unfoldedZoomDelta?: number;\n /** 是否在折叠/展开时保持中心点 */\n keepCenterOnFold?: boolean;\n /** 折叠状态变化回调 */\n onFoldStateChange?: (state: FoldState, deviceInfo: DeviceInfo) => void;\n /** 是否启用调试日志 */\n debug?: boolean;\n}\n\n/**\n * 折叠屏地图视图组件\n * \n * 自动适配折叠屏设备的展开/折叠状态变化\n */\nexport interface FoldableMapViewProps extends MapViewProps {\n /** 折叠屏适配配置 */\n foldableConfig?: FoldableConfig;\n}\n\nconst DEFAULT_FOLDABLE_CONFIG: Required<FoldableConfig> = {\n autoAdjustZoom: true,\n unfoldedZoomDelta: 1,\n keepCenterOnFold: true,\n onFoldStateChange: () => {},\n debug: false,\n};\n\nfunction clampZoomLevel(zoom: number): number {\n return Math.max(3, Math.min(20, zoom));\n}\n\nfunction createFoldableConfig(config?: FoldableConfig): Required<FoldableConfig> {\n return {\n ...DEFAULT_FOLDABLE_CONFIG,\n ...config,\n };\n}\n\nasync function applyFoldStateCameraAdjustment(\n mapRef: React.RefObject<MapViewRef | null>,\n oldState: FoldState,\n newState: FoldState,\n config: Required<FoldableConfig>,\n debugPrefix: 'FoldableMapView' | 'useFoldableMap'\n): Promise<void> {\n if (!mapRef.current || !config.autoAdjustZoom) {\n return;\n }\n\n const currentCamera = await mapRef.current.getCameraPosition();\n if (!currentCamera) {\n if (config.debug) {\n console.warn(`[${debugPrefix}] 无法获取相机位置`);\n }\n return;\n }\n\n const isUnfolding = newState === FoldState.UNFOLDED && oldState === FoldState.FOLDED;\n const isFolding = newState === FoldState.FOLDED && oldState === FoldState.UNFOLDED;\n\n if (config.debug) {\n console.log(`[${debugPrefix}] 折叠状态变化:`, {\n oldState,\n newState,\n isUnfolding,\n isFolding,\n currentZoom: currentCamera.zoom,\n });\n }\n\n if (!isUnfolding && !isFolding) {\n return;\n }\n\n const currentZoom = currentCamera.zoom ?? 15;\n const zoomDelta = isUnfolding ? config.unfoldedZoomDelta : -config.unfoldedZoomDelta;\n const nextZoom = clampZoomLevel(currentZoom + zoomDelta);\n\n if (config.debug) {\n console.log(`[${debugPrefix}] 调整缩放:`, {\n oldZoom: currentZoom,\n newZoom: nextZoom,\n delta: zoomDelta,\n });\n }\n\n await mapRef.current.moveCamera({\n target: config.keepCenterOnFold ? currentCamera.target : undefined,\n zoom: nextZoom,\n }, 300);\n}\n\nexport const FoldableMapView: React.FC<FoldableMapViewProps> = ({\n foldableConfig,\n ...mapProps\n}) => {\n const mapRef = useRef<MapViewRef>(null);\n const [currentFoldState, setCurrentFoldState] = useState<FoldState>(FoldState.UNKNOWN);\n const [deviceInfo, setDeviceInfo] = useState<DeviceInfo>(PlatformDetector.getDeviceInfo());\n const config = useMemo(() => createFoldableConfig(foldableConfig), [foldableConfig]);\n const foldStateRef = useRef(currentFoldState);\n\n const handleFoldStateChange = useEventCallback(\n async (newInfo: DeviceInfo, debugPrefix: 'FoldableMapView' | 'useFoldableMap') => {\n const newFoldState = PlatformDetector.getFoldState();\n const previousFoldState = foldStateRef.current;\n\n if (config.debug) {\n console.log(`[${debugPrefix}] 屏幕尺寸变化`);\n console.log('新设备信息:', newInfo);\n console.log('新折叠状态:', newFoldState);\n }\n\n if (newFoldState !== previousFoldState && previousFoldState !== FoldState.UNKNOWN) {\n try {\n await applyFoldStateCameraAdjustment(\n mapRef,\n previousFoldState,\n newFoldState,\n config,\n debugPrefix\n );\n } catch (error) {\n if (config.debug) {\n console.error(`[${debugPrefix}] 处理折叠状态变化失败:`, error);\n }\n }\n }\n\n foldStateRef.current = newFoldState;\n setCurrentFoldState(newFoldState);\n setDeviceInfo(newInfo);\n config.onFoldStateChange(newFoldState, newInfo);\n }\n );\n\n useEffect(() => {\n foldStateRef.current = currentFoldState;\n }, [currentFoldState]);\n\n useEffect(() => {\n const latestDeviceInfo = PlatformDetector.getDeviceInfo();\n setDeviceInfo(latestDeviceInfo);\n }, []);\n\n useEffect(() => {\n // 仅在 Android 折叠屏设备上启用\n if (Platform.OS !== 'android' || !deviceInfo.isFoldable) {\n if (config.debug) {\n console.log('[FoldableMapView] 非折叠屏设备,跳过适配');\n }\n return;\n }\n\n const initialState = PlatformDetector.getFoldState();\n foldStateRef.current = initialState;\n setCurrentFoldState(initialState);\n\n if (config.debug) {\n console.log('[FoldableMapView] 初始化折叠屏适配');\n console.log('设备信息:', deviceInfo);\n console.log('初始折叠状态:', initialState);\n }\n\n // 监听屏幕尺寸变化\n const removeListener = PlatformDetector.addDimensionChangeListener(\n async (newInfo: DeviceInfo) => {\n await handleFoldStateChange(newInfo, 'FoldableMapView');\n }\n );\n\n return () => {\n removeListener();\n };\n }, [config.debug, deviceInfo.isFoldable, handleFoldStateChange]);\n\n return (\n <ExpoGaodeMapView\n ref={mapRef}\n {...mapProps}\n />\n );\n};\n\n/**\n * 折叠屏适配 Hook\n * \n * 用于在现有地图组件中添加折叠屏适配功能\n */\nexport function useFoldableMap(\n mapRef: React.RefObject<MapViewRef>,\n config?: FoldableConfig\n) {\n const [foldState, setFoldState] = useState<FoldState>(FoldState.UNKNOWN);\n const [deviceInfo, setDeviceInfo] = useState<DeviceInfo>(PlatformDetector.getDeviceInfo());\n const mergedConfig = useMemo(() => createFoldableConfig(config), [config]);\n const foldStateRef = useRef(foldState);\n\n useEffect(() => {\n foldStateRef.current = foldState;\n }, [foldState]);\n\n const handleFoldStateChange = useEventCallback(async (newInfo: DeviceInfo) => {\n const newFoldState = PlatformDetector.getFoldState();\n const previousFoldState = foldStateRef.current;\n\n if (newFoldState !== previousFoldState && previousFoldState !== FoldState.UNKNOWN) {\n try {\n await applyFoldStateCameraAdjustment(\n mapRef,\n previousFoldState,\n newFoldState,\n mergedConfig,\n 'useFoldableMap'\n );\n } catch (error) {\n if (mergedConfig.debug) {\n console.error('[useFoldableMap] 调整失败:', error);\n }\n }\n }\n\n foldStateRef.current = newFoldState;\n setFoldState(newFoldState);\n setDeviceInfo(newInfo);\n mergedConfig.onFoldStateChange(newFoldState, newInfo);\n });\n\n useEffect(() => {\n if (Platform.OS !== 'android' || !deviceInfo.isFoldable) {\n return;\n }\n\n const initialState = PlatformDetector.getFoldState();\n foldStateRef.current = initialState;\n setFoldState(initialState);\n\n const removeListener = PlatformDetector.addDimensionChangeListener(\n async (newInfo: DeviceInfo) => {\n await handleFoldStateChange(newInfo);\n }\n );\n\n return () => {\n removeListener();\n };\n }, [deviceInfo.isFoldable, handleFoldStateChange, mapRef]);\n\n return {\n foldState,\n deviceInfo,\n isFoldable: deviceInfo.isFoldable,\n };\n}\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;AAgB/C;;;;;;;;;GASG;AACH,iBAAS,MAAM,CAAC,KAAK,EAAE,WAAW,qBA6JjC;;AAED,wBAAkC"}
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;AAiB/C;;;;;;;;;GASG;AACH,iBAAS,MAAM,CAAC,KAAK,EAAE,WAAW,qBA6JjC;;AAED,wBAAkC"}
@@ -1,6 +1,7 @@
1
1
  import * as React from 'react';
2
2
  import { StyleSheet, View } from 'react-native';
3
3
  import ExpoGaodeMapModule from '../../ExpoGaodeMapModule';
4
+ import { useEventCallback } from '../../hooks/useEventCallback';
4
5
  import { normalizeLatLng, normalizeLatLngList } from '../../utils/GeoUtils';
5
6
  import { createLazyNativeViewManager } from '../../utils/lazyNativeViewManager';
6
7
  const getNativeMarkerView = createLazyNativeViewManager('MarkerView');
@@ -24,10 +25,10 @@ function Marker(props) {
24
25
  // 归一化坐标处理
25
26
  const normalizedPosition = React.useMemo(() => normalizeLatLng(position), [position]);
26
27
  const normalizedSmoothMovePath = React.useMemo(() => (smoothMovePath ? normalizeLatLngList(smoothMovePath) : undefined), [smoothMovePath]);
27
- const emitSmoothMoveProgress = React.useEffectEvent((nativeEvent) => {
28
+ const emitSmoothMoveProgress = useEventCallback((nativeEvent) => {
28
29
  props.onSmoothMoveProgress?.({ nativeEvent });
29
30
  });
30
- const emitSmoothMoveEnd = React.useEffectEvent((nativeEvent) => {
31
+ const emitSmoothMoveEnd = useEventCallback((nativeEvent) => {
31
32
  props.onSmoothMoveEnd?.({ nativeEvent });
32
33
  });
33
34
  // 根据是否有 children 来决定使用哪个尺寸属性
@@ -87,7 +88,7 @@ function Marker(props) {
87
88
  }
88
89
  }, 100);
89
90
  return () => clearInterval(intervalId);
90
- }, [normalizedSmoothMovePath, smoothMoveDuration]);
91
+ }, [emitSmoothMoveEnd, emitSmoothMoveProgress, normalizedSmoothMovePath, smoothMoveDuration]);
91
92
  const handleAutoMeasure = (event) => {
92
93
  const nextWidth = Math.ceil(event.nativeEvent.layout.width);
93
94
  const nextHeight = Math.ceil(event.nativeEvent.layout.height);
@@ -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 { 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 contentWidth?: number;\n contentHeight?: number;\n};\n\nconst getNativeMarkerView = createLazyNativeViewManager<NativeMarkerViewProps>('MarkerView');\n\nconst AUTO_SIZE_FALLBACK = { width: 0, height: 0 };\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 iconWidth,\n iconHeight,\n children,\n smoothMovePath,\n cacheKey,\n ...restProps\n } = props;\n\n const smoothMoveDuration = props.smoothMoveDuration;\n \n // 归一化坐标处理\n const normalizedPosition = React.useMemo(() => normalizeLatLng(position), [position]);\n const normalizedSmoothMovePath = React.useMemo(\n () => (smoothMovePath ? normalizeLatLngList(smoothMovePath) : undefined),\n [smoothMovePath]\n );\n\n const emitSmoothMoveProgress = React.useEffectEvent((nativeEvent: {\n position: { latitude: number; longitude: number };\n angle: number;\n progress: number;\n distance: number;\n totalDistance: number;\n }) => {\n props.onSmoothMoveProgress?.({ nativeEvent } as never);\n });\n\n const emitSmoothMoveEnd = React.useEffectEvent((nativeEvent: {\n position: { latitude: number; longitude: number };\n angle: number;\n totalDistance: number;\n }) => {\n props.onSmoothMoveEnd?.({ nativeEvent } as never);\n });\n\n // 根据是否有 children 来决定使用哪个尺寸属性\n const hasChildren = !!children;\n const shouldWrapChildrenForMeasurement = hasChildren;\n // Android 的 children marker 之前始终透传 0 尺寸,点击后重新快照时容易直接消失。\n // 统一走自动测量后,两个平台都会拿到稳定尺寸。\n const shouldUseAutoMeasuredSize = hasChildren;\n const resolvedContentWidth = shouldUseAutoMeasuredSize ? measuredSize.width : 0;\n const resolvedContentHeight = shouldUseAutoMeasuredSize ? measuredSize.height : 0;\n\n React.useEffect(() => {\n if (\n !normalizedSmoothMovePath ||\n normalizedSmoothMovePath.length < 2 ||\n !smoothMoveDuration ||\n smoothMoveDuration <= 0\n ) {\n return undefined;\n }\n\n const totalDistance = ExpoGaodeMapModule.calculatePathLength(normalizedSmoothMovePath);\n if (totalDistance <= 0) {\n emitSmoothMoveEnd({\n position: normalizedSmoothMovePath[normalizedSmoothMovePath.length - 1],\n angle: 0,\n totalDistance,\n });\n return undefined;\n }\n\n const durationMs = 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 emitSmoothMoveProgress({\n position: point,\n angle,\n progress,\n distance,\n totalDistance,\n });\n\n if (progress >= 1) {\n emitSmoothMoveEnd({\n position: point,\n angle,\n totalDistance,\n });\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 }, [normalizedSmoothMovePath, smoothMoveDuration]);\n\n const handleAutoMeasure = (event: LayoutChangeEvent) => {\n const nextWidth = Math.ceil(event.nativeEvent.layout.width);\n const nextHeight = 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 ? resolvedContentWidth\n : (iconWidth && iconWidth > 0 ? iconWidth : 40);\n \n const finalIconHeight = hasChildren\n ? resolvedContentHeight\n : (iconHeight && iconHeight > 0 ? iconHeight : 40);\n\n const optionalNativeProps = cacheKey != null ? { cacheKey } : undefined;\n \n return (\n <NativeMarkerView\n {...restProps}\n {...optionalNativeProps}\n latitude={normalizedPosition.latitude}\n longitude={normalizedPosition.longitude}\n iconWidth={finalIconWidth}\n iconHeight={finalIconHeight}\n contentWidth={hasChildren ? finalIconWidth : 0}\n contentHeight={hasChildren ? finalIconHeight : 0}\n smoothMovePath={normalizedSmoothMovePath}\n >\n {hasChildren && shouldWrapChildrenForMeasurement ? (\n <View\n collapsable={false}\n onLayout={handleAutoMeasure}\n style={styles.measureContainer}\n >\n {children}\n </View>\n ) : children}\n </NativeMarkerView>\n );\n}\n\nexport default React.memo(Marker);\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":"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","sourcesContent":["import * as React from 'react';\nimport type { LayoutChangeEvent } from 'react-native';\nimport { StyleSheet, View } from 'react-native';\nimport type { MarkerProps } from '../../types';\nimport ExpoGaodeMapModule from '../../ExpoGaodeMapModule';\nimport { useEventCallback } from '../../hooks/useEventCallback';\nimport { normalizeLatLng, normalizeLatLngList } from '../../utils/GeoUtils';\nimport { createLazyNativeViewManager } from '../../utils/lazyNativeViewManager';\n\ntype NativeMarkerViewProps = Omit<MarkerProps, 'position'> & {\n latitude: number;\n longitude: number;\n contentWidth?: number;\n contentHeight?: number;\n};\n\nconst getNativeMarkerView = createLazyNativeViewManager<NativeMarkerViewProps>('MarkerView');\n\nconst AUTO_SIZE_FALLBACK = { width: 0, height: 0 };\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 iconWidth,\n iconHeight,\n children,\n smoothMovePath,\n cacheKey,\n ...restProps\n } = props;\n\n const smoothMoveDuration = props.smoothMoveDuration;\n \n // 归一化坐标处理\n const normalizedPosition = React.useMemo(() => normalizeLatLng(position), [position]);\n const normalizedSmoothMovePath = React.useMemo(\n () => (smoothMovePath ? normalizeLatLngList(smoothMovePath) : undefined),\n [smoothMovePath]\n );\n\n const emitSmoothMoveProgress = useEventCallback((nativeEvent: {\n position: { latitude: number; longitude: number };\n angle: number;\n progress: number;\n distance: number;\n totalDistance: number;\n }) => {\n props.onSmoothMoveProgress?.({ nativeEvent } as never);\n });\n\n const emitSmoothMoveEnd = useEventCallback((nativeEvent: {\n position: { latitude: number; longitude: number };\n angle: number;\n totalDistance: number;\n }) => {\n props.onSmoothMoveEnd?.({ nativeEvent } as never);\n });\n\n // 根据是否有 children 来决定使用哪个尺寸属性\n const hasChildren = !!children;\n const shouldWrapChildrenForMeasurement = hasChildren;\n // Android 的 children marker 之前始终透传 0 尺寸,点击后重新快照时容易直接消失。\n // 统一走自动测量后,两个平台都会拿到稳定尺寸。\n const shouldUseAutoMeasuredSize = hasChildren;\n const resolvedContentWidth = shouldUseAutoMeasuredSize ? measuredSize.width : 0;\n const resolvedContentHeight = shouldUseAutoMeasuredSize ? measuredSize.height : 0;\n\n React.useEffect(() => {\n if (\n !normalizedSmoothMovePath ||\n normalizedSmoothMovePath.length < 2 ||\n !smoothMoveDuration ||\n smoothMoveDuration <= 0\n ) {\n return undefined;\n }\n\n const totalDistance = ExpoGaodeMapModule.calculatePathLength(normalizedSmoothMovePath);\n if (totalDistance <= 0) {\n emitSmoothMoveEnd({\n position: normalizedSmoothMovePath[normalizedSmoothMovePath.length - 1],\n angle: 0,\n totalDistance,\n });\n return undefined;\n }\n\n const durationMs = 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 emitSmoothMoveProgress({\n position: point,\n angle,\n progress,\n distance,\n totalDistance,\n });\n\n if (progress >= 1) {\n emitSmoothMoveEnd({\n position: point,\n angle,\n totalDistance,\n });\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 }, [emitSmoothMoveEnd, emitSmoothMoveProgress, normalizedSmoothMovePath, smoothMoveDuration]);\n\n const handleAutoMeasure = (event: LayoutChangeEvent) => {\n const nextWidth = Math.ceil(event.nativeEvent.layout.width);\n const nextHeight = 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 ? resolvedContentWidth\n : (iconWidth && iconWidth > 0 ? iconWidth : 40);\n \n const finalIconHeight = hasChildren\n ? resolvedContentHeight\n : (iconHeight && iconHeight > 0 ? iconHeight : 40);\n\n const optionalNativeProps = cacheKey != null ? { cacheKey } : undefined;\n \n return (\n <NativeMarkerView\n {...restProps}\n {...optionalNativeProps}\n latitude={normalizedPosition.latitude}\n longitude={normalizedPosition.longitude}\n iconWidth={finalIconWidth}\n iconHeight={finalIconHeight}\n contentWidth={hasChildren ? finalIconWidth : 0}\n contentHeight={hasChildren ? finalIconHeight : 0}\n smoothMovePath={normalizedSmoothMovePath}\n >\n {hasChildren && shouldWrapChildrenForMeasurement ? (\n <View\n collapsable={false}\n onLayout={handleAutoMeasure}\n style={styles.measureContainer}\n >\n {children}\n </View>\n ) : children}\n </NativeMarkerView>\n );\n}\n\nexport default React.memo(Marker);\n\nconst styles = StyleSheet.create({\n measureContainer: {\n alignSelf: 'flex-start',\n alignItems: 'flex-start',\n flexShrink: 0,\n },\n});\n"]}
@@ -0,0 +1,2 @@
1
+ export declare function useEventCallback<Args extends unknown[], Return>(callback: (...args: Args) => Return): (...args: Args) => Return;
2
+ //# sourceMappingURL=useEventCallback.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useEventCallback.d.ts","sourceRoot":"","sources":["../../src/hooks/useEventCallback.ts"],"names":[],"mappings":"AAEA,wBAAgB,gBAAgB,CAAC,IAAI,SAAS,OAAO,EAAE,EAAE,MAAM,EAC7D,QAAQ,EAAE,CAAC,GAAG,IAAI,EAAE,IAAI,KAAK,MAAM,GAClC,CAAC,GAAG,IAAI,EAAE,IAAI,KAAK,MAAM,CAQ3B"}
@@ -0,0 +1,9 @@
1
+ import * as React from 'react';
2
+ export function useEventCallback(callback) {
3
+ const callbackRef = React.useRef(callback);
4
+ React.useLayoutEffect(() => {
5
+ callbackRef.current = callback;
6
+ });
7
+ return React.useCallback((...args) => callbackRef.current(...args), []);
8
+ }
9
+ //# sourceMappingURL=useEventCallback.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useEventCallback.js","sourceRoot":"","sources":["../../src/hooks/useEventCallback.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAE/B,MAAM,UAAU,gBAAgB,CAC9B,QAAmC;IAEnC,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;IAE3C,KAAK,CAAC,eAAe,CAAC,GAAG,EAAE;QACzB,WAAW,CAAC,OAAO,GAAG,QAAQ,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC,WAAW,CAAC,CAAC,GAAG,IAAU,EAAE,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC;AAChF,CAAC","sourcesContent":["import * as React from 'react';\n\nexport function useEventCallback<Args extends unknown[], Return>(\n callback: (...args: Args) => Return\n): (...args: Args) => Return {\n const callbackRef = React.useRef(callback);\n\n React.useLayoutEffect(() => {\n callbackRef.current = callback;\n });\n\n return React.useCallback((...args: Args) => callbackRef.current(...args), []);\n}\n"]}
@@ -1 +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;AAmEvC;;;;;GAKG;AACH,wBAAgB,gBAAgB,CAC9B,MAAM,EAAE,KAAK,CAAC,MAAM,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,EAAE,CAAC,EACnD,OAAO,GAAE,oBAAyB,GACjC,uBAAuB,CA0UzB"}
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;AAoEvC;;;;;GAKG;AACH,wBAAgB,gBAAgB,CAC9B,MAAM,EAAE,KAAK,CAAC,MAAM,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,EAAE,CAAC,EACnD,OAAO,GAAE,oBAAyB,GACjC,uBAAuB,CA0UzB"}
@@ -3,6 +3,7 @@ import ExpoGaodeMapModule from '../ExpoGaodeMapModule';
3
3
  import { MapContext } from '../components/MapContext';
4
4
  import { normalizeLatLngList } from '../utils/GeoUtils';
5
5
  import { fitCameraToCoordinates } from '../utils/RouteUtils';
6
+ import { useEventCallback } from './useEventCallback';
6
7
  const DEFAULT_STATE = {
7
8
  isPlaying: false,
8
9
  isPaused: false,
@@ -75,10 +76,10 @@ export function useRoutePlayback(points, options = {}) {
75
76
  React.useEffect(() => {
76
77
  optionsRef.current = options;
77
78
  }, [options]);
78
- const emitProgress = React.useEffectEvent((nextState) => {
79
+ const emitProgress = useEventCallback((nextState) => {
79
80
  options.onProgress?.(nextState);
80
81
  });
81
- const emitComplete = React.useEffectEvent((nextState) => {
82
+ const emitComplete = useEventCallback((nextState) => {
82
83
  options.onComplete?.(nextState);
83
84
  });
84
85
  const normalizedPath = React.useMemo(() => {
@@ -118,7 +119,7 @@ export function useRoutePlayback(points, options = {}) {
118
119
  stateRef.current = nextState;
119
120
  setState(nextState);
120
121
  emitProgress(nextState);
121
- }, []);
122
+ }, [emitProgress]);
122
123
  const syncProgress = React.useCallback(async (progress, keepPlaying) => {
123
124
  // 根据“当前进度 -> 已行驶距离 -> 路径上的点”推导出车辆位置,
124
125
  // 再同步决定 smoothMovePath / 相机跟随 / 对外状态。
@@ -176,7 +177,7 @@ export function useRoutePlayback(points, options = {}) {
176
177
  return;
177
178
  }
178
179
  await syncProgress(progress, true);
179
- }, [durationSeconds, stopTimer, syncProgress]);
180
+ }, [durationSeconds, emitComplete, stopTimer, syncProgress]);
180
181
  const startTimer = React.useCallback(() => {
181
182
  stopTimer();
182
183
  const interval = Math.max(optionsRef.current.updateIntervalMs ?? 100, 16);
@@ -1 +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\n\n/**\n * 获取标记角度\n * @param path 路径点\n * @param totalDistance 总距离\n * @param targetDistance 目标距离\n * @param lastAngle 上一个角度\n * @param options 回放选项\n * @returns 角度和点\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\n/**\n * 路由回放\n * @param points 路径点\n * @param options 回放选项\n * @returns 回放控制器\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 emitProgress = React.useEffectEvent((nextState: RoutePlaybackState) => {\n options.onProgress?.(nextState);\n });\n\n const emitComplete = React.useEffectEvent((nextState: RoutePlaybackState) => {\n options.onComplete?.(nextState);\n });\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 emitProgress(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 emitComplete(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"]}
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';\nimport { useEventCallback } from './useEventCallback';\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\n\n/**\n * 获取标记角度\n * @param path 路径点\n * @param totalDistance 总距离\n * @param targetDistance 目标距离\n * @param lastAngle 上一个角度\n * @param options 回放选项\n * @returns 角度和点\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\n/**\n * 路由回放\n * @param points 路径点\n * @param options 回放选项\n * @returns 回放控制器\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 emitProgress = useEventCallback((nextState: RoutePlaybackState) => {\n options.onProgress?.(nextState);\n });\n\n const emitComplete = useEventCallback((nextState: RoutePlaybackState) => {\n options.onComplete?.(nextState);\n });\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 emitProgress(nextState);\n },\n [emitProgress]\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 emitComplete(completedState);\n return;\n }\n\n await syncProgress(progress, true);\n }, [durationSeconds, emitComplete, 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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "expo-gaode-map",
3
- "version": "2.2.37-next.0",
3
+ "version": "2.2.37",
4
4
  "description": "A full-featured React Native AMap (Gaode Map) library for Expo, including map display, location, overlays, offline maps, and geometry utilities.",
5
5
  "main": "build/index.js",
6
6
  "types": "build/index.d.ts",