From 648f5f4d4907ad1ca1aa6a9c97bb5c252c310754 Mon Sep 17 00:00:00 2001 From: tangshaojian <63377964@qq.com> Date: Thu, 24 Apr 2025 22:52:12 +0800 Subject: [PATCH] =?UTF-8?q?tsj:=20CooglMap=20js=E6=9B=B4=E6=96=B0=E5=88=B0?= =?UTF-8?q?1.4.0=E7=89=88=E6=9C=AC=EF=BC=9B=E5=AE=9E=E7=8E=B0=E8=A1=8C?= =?UTF-8?q?=E6=94=BF=E5=8C=BA=E5=88=92=E6=8C=89=E9=92=AE=E7=BB=84=E6=95=B0?= =?UTF-8?q?=E6=8D=AE=E4=BB=8E=20regionTreeResult.json=E9=80=9A=E8=BF=87?= =?UTF-8?q?=E6=96=B9=E6=B3=95=E6=9D=A5=E8=8E=B7=E5=8F=96=E8=80=8C=E4=B8=8D?= =?UTF-8?q?=E6=98=AF=E9=9D=99=E6=80=81=EF=BC=9B=E5=AE=9E=E7=8E=B0=E8=A1=8C?= =?UTF-8?q?=E6=94=BF=E5=8C=BA=E5=88=92=E6=8C=89=E9=92=AE=E7=82=B9=E5=87=BB?= =?UTF-8?q?=E4=B8=8E=E5=9C=B0=E5=9B=BE=E7=9A=84=E8=81=94=E5=8A=A8=EF=BC=9B?= =?UTF-8?q?=E5=AE=9E=E7=8E=B0=E5=9C=B0=E5=9B=BE=E7=82=B9=E5=87=BB=E4=BA=8B?= =?UTF-8?q?=E4=BB=B6=EF=BC=9B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../MapContainer/AreaSelector/index.tsx | 221 +++++--- .../DistrictMapCustomized/index.tsx | 67 ++- .../DistrictMapCustomized/loadArea-1.4.0.js | 527 ++++++++++++++++++ .../components/MapContainer/index.tsx | 24 +- 4 files changed, 748 insertions(+), 91 deletions(-) create mode 100644 src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/loadArea-1.4.0.js diff --git a/src/pages/Home_v_2504/components/MapContainer/AreaSelector/index.tsx b/src/pages/Home_v_2504/components/MapContainer/AreaSelector/index.tsx index 6500fc36..de56f441 100644 --- a/src/pages/Home_v_2504/components/MapContainer/AreaSelector/index.tsx +++ b/src/pages/Home_v_2504/components/MapContainer/AreaSelector/index.tsx @@ -1,91 +1,147 @@ -import React, { useState } from 'react'; +import React, { useState, useEffect } from 'react'; import classNames from 'classnames'; import styles from './index.less'; import { useGlobalModalServices } from '@/pages/GlobalModalServices/provider'; +// 导入区域树数据 +import regionTreeResult from '../DistrictMapCustomized/regionTreeResult.json'; + +// 按钮数据类型定义 +interface ButtonItem { + key: string; + label: string; + hasChildren?: boolean; + regionCode?: string; + dataCode?: string; +} // 定义按钮组数据类型 interface ButtonGroupData { - level1: Array<{ - key: string; - label: string; - hasChildren?: boolean; - }>; - level2: Record>; - level3: Record>; + level1: ButtonItem[]; + level2: Record; + level3: Record; } -const AreaSelector: React.FC = () => { +interface AreaSelectorProps { + mapInstance?: any; // CooglLoadArea instance reference +} + +const AreaSelector: React.FC = ({ mapInstance }) => { // 获取全局弹窗服务 const { dispatch } = useGlobalModalServices(); - // 按钮组数据 - const buttonGroupData: ButtonGroupData = { + // 初始化按钮组数据的状态 + const [buttonGroupData, setButtonGroupData] = useState({ level1: [ - { key: 'chengdu', label: '成都市', hasChildren: true }, - { key: 'oilSmoke', label: '油烟事件' }, - { key: 'oilSmokeAlert', label: '油烟告警' }, - { key: 'restaurant', label: '餐饮企业' }, + { key: 'chengdu', label: '成都市', hasChildren: true, regionCode: '5101' }, + { key: 'oilSmoke', label: '油烟事件', dataCode: '20101' }, + { key: 'oilSmokeAlert', label: '油烟告警', dataCode: '20102' }, + { key: 'restaurant', label: '餐饮企业', dataCode: '20103' }, ], - level2: { - chengdu: [ - { key: 'gaoxin', label: '高新区' }, - { key: 'jinjiang', label: '锦江区', hasChildren: true }, - { key: 'chenghua', label: '成华区' }, - { key: 'qingyang', label: '青羊区', hasChildren: true }, - { key: 'jinniu', label: '金牛区' }, - { key: 'xindou', label: '新都区' }, - { key: 'wuhou', label: '武侯区', hasChildren: true }, - { key: 'longquanyi', label: '龙泉驿区' }, - { key: 'wenjiang', label: '温江区' }, - { key: 'pengzhou', label: '彭州市' }, - { key: 'xinjin', label: '新津区' }, - { key: 'dayi', label: '大邑县' }, - { key: 'jianyang', label: '简阳市' }, - ] - }, - level3: { - jinjiang: [ - { key: 'qingshuitang', label: '清水塘街道' }, - { key: 'sanlitun', label: '三里口街道' }, - { key: 'dafeng', label: '大丰街道' }, - { key: 'xindu', label: '新都街道' }, - { key: 'jinxing', label: '锦兴街道' }, - { key: 'zhujiang', label: '珠江街道' }, - ], - qingyang: [ - { key: 'sanhe', label: '三河街道' }, - { key: 'shibanqiao', label: '石板桥街道' }, - { key: 'zhupanzhen', label: '竹盘珍街道' }, - { key: 'xindu2', label: '新都街道' }, - ], - wuhou: [ - { key: 'wuhou1', label: '桂湖街道' }, - { key: 'wuhou2', label: '大丰街道' }, - { key: 'wuhou3', label: '三河街道' }, - { key: 'wuhou4', label: '石板桥街道' }, - { key: 'wuhou5', label: '竹盘珍街道' }, - { key: 'wuhou6', label: '新都街道' }, - ] - } - }; + level2: {}, + level3: {}, + }); // 状态管理 const [activeLevel1, setActiveLevel1] = useState(null); const [activeLevel2, setActiveLevel2] = useState(null); const [activeLevel3, setActiveLevel3] = useState(null); + // 控制第二层、第三层按钮组的显示 const showLevel2 = activeLevel1 && buttonGroupData.level2[activeLevel1]; const showLevel3 = activeLevel2 && buttonGroupData.level3[activeLevel2]; + // 初始化时从 JSON 加载区域树数据 + useEffect(() => { + if (!regionTreeResult?.data) return; + + try { + // 解析和处理区域树数据 + const processedData = processRegionTree(regionTreeResult.data); + + // 使用解析后的数据更新按钮组 + setButtonGroupData(prevData => ({ + ...prevData, + level2: { + chengdu: processedData.level2 + }, + level3: processedData.level3 + })); + } catch (error) { + console.error('处理区域树数据出错:', error); + } + }, []); + + // 处理区域树数据的函数 + const processRegionTree = (treeData: any[]) => { + const level2Data: ButtonItem[] = []; + const level3Data: Record = {}; + + // 查找成都市节点 + const chengduNode = treeData.find(node => node.id === 5101); + + if (chengduNode && chengduNode.children) { + // 处理区级 (level2) + chengduNode.children.forEach((district: any) => { + // 判断是否有子节点(街道) + const hasChildren = district.children && district.children.length > 0; + + level2Data.push({ + key: `district_${district.id}`, + label: district.gridName, + hasChildren, + regionCode: district.id.toString(), + }); + + // 如果有子节点,处理街道 (level3) + if (hasChildren) { + const streetData: ButtonItem[] = []; + + district.children.forEach((street: any) => { + streetData.push({ + key: `street_${street.id}`, + label: street.gridName, + regionCode: street.id.toString() + }); + }); + + level3Data[`district_${district.id}`] = streetData; + } + }); + } + + return { + level2: level2Data, + level3: level3Data + }; + }; + + // 更新地图区域和数据 + const updateMapArea = (regionCode: string, dataCode?: string) => { + if (!mapInstance) return; + + // 更新地图实例的参数 + if (regionCode) { + mapInstance.params.regionCode = regionCode; + } + + if (dataCode) { + mapInstance.params.dataCode = dataCode; + } + + // 判断是否为社区级别,设置不同的加载方法 + const isCommunityLevel = regionCode?.length === 9; + + if (isCommunityLevel) { + // 社区级别,加载点位数据 + mapInstance.setPoints(true); + } else { + // 区/街道级别,加载区域数据 + mapInstance.addArea(true); + } + }; + // 处理第一层按钮点击 - const handleLevel1Click = (key: string, hasChildren?: boolean) => { + const handleLevel1Click = (key: string, hasChildren?: boolean, regionCode?: string, dataCode?: string) => { if (key === 'oilSmokeAlert') { // 点击油烟告警按钮,显示弹窗 dispatch.push('RestaurantOilAlarm'); @@ -107,10 +163,15 @@ const AreaSelector: React.FC = () => { setActiveLevel1(key); setActiveLevel2(null); } + + // 更新地图区域展示 + if (regionCode) { + updateMapArea(regionCode, dataCode); + } }; // 处理第二层按钮点击 - const handleLevel2Click = (key: string, hasChildren?: boolean) => { + const handleLevel2Click = (key: string, hasChildren?: boolean, regionCode?: string) => { if (activeLevel2 === key) { // 如果点击当前已激活的按钮,则取消选中 setActiveLevel2(null); @@ -118,12 +179,31 @@ const AreaSelector: React.FC = () => { // 否则激活该按钮 setActiveLevel2(key); } + + // 更新地图区域展示 + if (regionCode) { + updateMapArea(regionCode); + } }; - const handleLevel3Click = (key: string) => { + // 处理第三层按钮点击 + const handleLevel3Click = (key: string, regionCode?: string) => { setActiveLevel3(key); + + // 更新地图区域展示 + if (regionCode) { + updateMapArea(regionCode); + } }; + // 默认展示成都市全貌 + useEffect(() => { + if (mapInstance) { + // 设置初始化参数为成都市 + updateMapArea('5101'); + } + }, [mapInstance]); + return (
{/* 第一层按钮组 */} @@ -134,7 +214,7 @@ const AreaSelector: React.FC = () => { className={classNames(styles.button, { [styles.buttonActive]: activeLevel1 === button.key })} - onClick={() => handleLevel1Click(button.key, button.hasChildren)} + onClick={() => handleLevel1Click(button.key, button.hasChildren, button.regionCode, button.dataCode)} > {button.label} {activeLevel1 === button.key && button.hasChildren && ( @@ -157,7 +237,7 @@ const AreaSelector: React.FC = () => { className={classNames(styles.button, { [styles.buttonActive]: activeLevel2 === button.key })} - onClick={() => handleLevel2Click(button.key, button.hasChildren)} + onClick={() => handleLevel2Click(button.key, button.hasChildren, button.regionCode)} > {button.label}
@@ -171,11 +251,10 @@ const AreaSelector: React.FC = () => { {buttonGroupData.level3[activeLevel2!].map((button) => (
handleLevel3Click(button.key)} + onClick={() => handleLevel3Click(button.key, button.regionCode)} > {button.label}
diff --git a/src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/index.tsx b/src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/index.tsx index 8e27d77e..163feec8 100644 --- a/src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/index.tsx +++ b/src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/index.tsx @@ -28,7 +28,9 @@ import styles from './index.less'; import PopoverCard from './PopoverCard'; import { useDashboard } from '../../context'; // import { CooglLoadArea } from './loadAreaNew'; -import { CooglLoadArea } from './loadArea-1.3.1'; +// import { CooglLoadArea } from './loadArea-1.3.1'; +import { CooglLoadArea } from './loadArea-1.4.0'; +import AreaSelector from '../AreaSelector'; export const colorConfig: Record< string, @@ -73,8 +75,14 @@ export const colorConfig: Record< tag: require('@/assets/images/SecurityServiceOverview/dtyjtag4.png'), }, }; + +// 添加组件属性接口 +interface DistrictMapCustomizedProps { + onMapInstanceChange?: (instance: any) => void; +} + /** 地图 */ -const DistrictMapCustomized: React.FC = () => { +const DistrictMapCustomized: React.FC = ({ onMapInstanceChange }) => { const { initialState } = useModel('@@initialState'); const { areaId } = initialState; const { dispatch } = useGlobalModalServices(); @@ -128,6 +136,16 @@ const DistrictMapCustomized: React.FC = () => { currentZoomRef.current = currentZoom; }, [currentZoom]); + // Add state to track the CooglLoadArea instance + const [areaMapInstance, setAreaMapInstance] = useState(null); + + // 在地图实例状态变化时,调用父组件传递的回调 + useEffect(() => { + if (areaMapInstance && onMapInstanceChange) { + onMapInstanceChange(areaMapInstance); + } + }, [areaMapInstance, onMapInstanceChange]); + const fetchVisiblePolygons = (bounds: any) => { // console.log('fetchVisiblePolygons bounds', bounds); // Only fetch if we don't have data yet @@ -398,25 +416,40 @@ const DistrictMapCustomized: React.FC = () => { console.log('[CooglMapDemo] 开始初始化CooglLoadArea', MapRef.current); let instance = new CooglLoadArea(viewer); - // 方便调试 - // window.mapInstance = mapRef.current; - // window.gridInstance = instanceRef.current; - - // console.log('[CooglMapDemo] 开始执行mapInit'); - instance.params ={ - regionCode: '5101', - gridType:'01', - startTime:'', - endTime:'', - dataCode:'20101', + // 设置实例参数 + instance.params = { + regionCode: '5101', // 默认显示成都市 + gridType: '01', // 城市部件 + startTime: '', + endTime: '', + dataCode: '20101', // 默认数据代码 hasChildren: 1, + }; + + // 保存实例到state,以便传递给父组件 + setAreaMapInstance(instance); + + // 使用类型断言解决 TypeScript 错误 + // @ts-ignore + instance.imageGraphicOfCommunityEvent = function(detail: any) { + // 处理点位点击事件 + console.log('点位详情:', detail); + + // 可以触发弹窗显示详情 + if (detail) { + dispatch.push('EventDetail', { + title: isHomePage ? '事件详情' : '餐饮油烟事件详情', + w: 'fit-content', + props: { + eventId: detail.eventId || detail.id, + }, + }); } - instance.mapInit(); + }; - // setStatus('网格基座初始化完成'); + instance.mapInit(); } catch (error) { console.error('[CooglMapDemo] 初始化网格基座失败:', error); - // setStatus(`初始化失败: ${error}`); } } @@ -503,6 +536,7 @@ const DistrictMapCustomized: React.FC = () => { {/* */} + {/* 状态栏 - 添加区域选择器并传递地图实例 */} {/*
{ setLeftTabKey={setLeftTabKey} setCheckedKeys={setCheckedKeys} /> +
*/}
diff --git a/src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/loadArea-1.4.0.js b/src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/loadArea-1.4.0.js new file mode 100644 index 00000000..7cebba68 --- /dev/null +++ b/src/pages/Home_v_2504/components/MapContainer/DistrictMapCustomized/loadArea-1.4.0.js @@ -0,0 +1,527 @@ +//这个是基于ceisum的二开,所以有些api不大相同 +// 区域划分的颜色配置 +import regionResult from './regionTreeResult.json' + +const CooGL = window.CooGL; + +// 调试开关 +const DEBUG = false; + +const IS_PROD = process.env.NODE_ENV === 'production' ? true : false; + +// API基础URL配置 +const API_BASE_URLS = { + // 直接地址 + DIRECT: { + REGION_API: 'https://10.1.174.34:13600/kelan/api', + GRID_RANGE_API: 'https://10.1.174.34:13600/grid-range' + }, + // 相对地址(代理) + PROXY: { + REGION_API: IS_PROD?'/cd-comprehensive/region/coogl/api':'/region/coogl/api', + GRID_RANGE_API: IS_PROD?'/cd-comprehensive/grid-range':'/grid-range' + } +}; + +// 是否使用直接地址(true: 使用直接地址, false: 使用代理地址) +const USE_DIRECT_URL = false; + +// 获取当前使用的API基础URL +const getApiBaseUrl = (type) => { + return USE_DIRECT_URL ? API_BASE_URLS.DIRECT[type] : API_BASE_URLS.PROXY[type]; +}; + +// 调试日志函数 +const logInfo = (...args) => { + if (DEBUG) { + console.log(...args); + } +}; +const logError = (...args) => { + if (DEBUG) { + console.error(...args); + } +}; + +export class CooglLoadArea { + params = { + regionCode: '', + gridType: '', + startTime: '', + endTime: '', + dataCode: '', + hasChildren: 1, + } + degreesArray = [] + levelOfMeter = { + 16: 20, + 14: 40, + 13: 75, + 12: 150, + // 11: 400, + } + level + opLevel + graphic + rightPickHandler + compressCancelFn + tinGraphic = [] + imageGraphicListOfCommunity = [] + imageGraphicList = [] + polygonGraphicList = [] + pickHandler = null + + //部件、事件、资源等拾取事件,返回拾取的详情 + imageGraphicOfCommunityEvent = null + + //设置在哪一级别加载当前区域的部件、事件、资源等数据 值为 6(区级) 9(街道) 12(社区、村),默认为12 + loadPointsLevel = 12 + constructor(viewer, token) { + this.viewer = viewer + this.regionResult = regionResult.data + this.token = token ?? CooGL.CooServer.defaultAccessToken + } + + async mapInit() { + if (this.pickHandler) { + this.pickHandler() + this.pickHandler = null + } + + //清除画布 + this.clear() + this.limitCamHeight() + //默认加载一次市级的标注上图 + this.pickHandler = this.viewer.event.pick(e => { + const { isCommunityImage, code: regionCode } = + e?.primitive?.description || {} + if (regionCode) { + this.params.regionCode = regionCode + //在这里判断 如果是社区、村的级别执行 setPoints + //示例 判断条件根据实际情况而定 + //4 6 9 12 + const isSameLevel = regionCode?.length === this.loadPointsLevel + if (isSameLevel) { + this.setPoints(true) + } else { + this.addArea(true) + } + } else if (isCommunityImage && this.imageGraphicOfCommunityEvent) { + this.imageGraphicOfCommunityEvent(e?.primitive?.description.detail) + } + }) + //增加右键功能,后退 + this.rightPickHandler = this.viewer.event.rightclick(e => { + if (this.params.regionCode == 5101) return + // 根据code查找父节点 + const parentCode = this.findParentCode(this.params.regionCode) + this.params.regionCode = parentCode + this.addArea(true) + }) + await this.addArea(true) + this.compressFn() + } + findParentCode(code) { + if (!code) return null + + // 递归查找函数 + const findParent = (nodes, targetCode, parent = null) => { + if (!nodes || !Array.isArray(nodes)) return null + + for (const node of nodes) { + // 找到目标节点,返回父节点id + if (node.id == targetCode) { + return parent?.id || null + } + + // 递归查找子节点 + if (node.children) { + const found = findParent(node.children, targetCode, node) + if (found !== null) return found + } + } + return null + } + + return findParent(this.regionResult, code) + } + /** + * 添加区域 + * @param {string} regionCode - 区域编码 + * @param {string} gridtype - 网格类型 + * 01:城市部件 + * 02:事件 + * 03:资源 + * 04:单元网格 + * 05:责任网格 + * 06:任意网格 + * 07:资源网格 + * 08:感知设备 + * @param {string} startTime - 开始时间 格式:yyyy-MM-DD HH:mm:ss 可选 + * @param {string} endTime - 结束时间 格式:yyyy-MM-DD HH:mm:ss 可选 + */ + + async addArea(clearData = false) { + const params = { ...this.params } + if (clearData) { + this.clear() + } else { + this.clearCurrentAreaData() + } + // const list = this.findRegionByCode(params.regionCode, true) + const resList = await Promise.all([ + this.getRigionDetailByCode(params.regionCode), + this.getAllRigionDetailByCode(params.regionCode), + ]) + this.moveViewToPolygon(resList[0]) + const sons = resList[1].features + for (const res of sons) { + this.addPolygon(res) + } + + const apiUrl = `${getApiBaseUrl('REGION_API')}/es/v1.0/bigdata_search_analysis`; + fetch(apiUrl, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + token: this.token, + }, + body: JSON.stringify(params), + }) + .then(res => res.json()) + .then(res => { + const obj = res.data + delete obj[params.regionCode] + Object.entries(obj).map(([key, value]) => { + const detailItem = sons.find(item => item.id === key) + if (!detailItem) return + const { _center, _name } = detailItem.properties + const center = _center.split(',') + const position = { + x: +center[0], + y: +center[1], + z: 0, + } + const item = { + num: value[0].num, + code: key, + position, + name: _name, + } + this.addImage(item) + }) + }) + } + + //获取指定regionCode的区域的经纬度以及中心点和下级区域的所有经纬度 + //FIXME: 这个接口需要自己实现 + async getRigionDetailByCode(regionCode) { + if (!regionCode) return Promise.reject('regionCode is required') + //参考已有代码 + // 获取geojson格式的区域边界数据 + // 获取txt格式的区域边界数据 + const apiUrl = `${getApiBaseUrl('GRID_RANGE_API')}/unit/${regionCode}`; + const response = await fetch(apiUrl) + return response.json() + } + async getAllRigionDetailByCode(regionCode) { + if (!regionCode) return Promise.reject('regionCode is required') + //参考已有代码 + // 获取geojson格式的区域边界数据 + // 获取txt格式的区域边界数据 + const apiUrl = `${getApiBaseUrl('GRID_RANGE_API')}/full/unit/${regionCode}`; + const response = await fetch(apiUrl) + return response.json() + } + + addImage({ num, code, position, name }) { + this.viewer + .addLayer( + CooGL.Layer.fromImageTextLabel({ + id: code, + description: { + code, + name, + }, + image: + '', + position, + scale: 0.5, + textContent: num, + font: '16px simsum', + pixelOffset: { x: 10, y: -100 }, + fillColor: '#fff', + style: CooGL.LabelStyle.FILL_AND_OUTLINE, + outlineWidth: 4, + outlineColor: '#000', + distanceDisplayCondition: new CooGL.DistanceDisplayCondition( + 0, + 100000, + ), + }), + ) + .then(res => { + this.imageGraphicList.push(res) + }) + } + addCommunityImage(obj, position) { + this.viewer + .addLayer( + CooGL.Layer.fromImageTextLabel({ + id: 'community' + '_' + obj.id, + description: { + isCommunityImage: true, + detail: obj, + }, + image: + '', + position, + textContent: obj.name, + font: '22px simsum', + pixelOffset: { x: 0, y: -150 }, + fillColor: '#fff', + style: CooGL.LabelStyle.FILL_AND_OUTLINE, + outlineWidth: 4, + distanceDisplayCondition: new CooGL.DistanceDisplayCondition( + 0, + 10000, + ), + outlineColor: '#000', + }), + ) + .then(res => { + this.imageGraphicListOfCommunity.push(res) + }) + } + addPolygon(polygonData) { + const id = 'polygon' + '_' + polygonData.id + this.viewer + .addLayer( + CooGL.Layer.fromGeoJsonGraphicResource({ + id, + resource: { + type: 'FeatureCollection', + features: [polygonData], + }, + customPolygonLabelOptions: function (positions, properties) { + const center = properties._center.split(',') + const textPosition = { x: center[0], y: center[1], z: 0 } + return { + positions, + material: '#0B304B', + textPosition, + outline: true, + outlineColor: '#36A6F5', + outlineWidth: 2, + fillColor: '#fff', + font: '12px simsum', + textContent: properties._name, + } + }, + }), + ) + .then(res => { + this.polygonGraphicList.push(res) + }) + } + + //获取村、社区下的特定网格类型点位并上图 + async setPoints(clearData = false) { + this.degreesArray = [] + if (clearData) { + this.clear() + } + const res = await this.getRigionDetailByCode(this.params.regionCode) + this.moveViewToPolygon(res) + const coordinates = res.features[0].geometry.coordinates[0] + .map(([x, y]) => `${x},${y}`) + .join(';') + // 构造请求参数 + const params = { + regionCode: this.params.regionCode, + dataCode: this.params.dataCode, + gridtype: this.params.gridType, + startTime: this.params.startTime, + endTime: this.params.endTime, + coordinates, + } + // 发送POST请求获取点位数据 + const apiUrl = `${getApiBaseUrl('REGION_API')}/es/v1.0/search`; + const response = await fetch(apiUrl, { + // const response = await fetch('/region/coogl/api/es/v1.0/search', { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + token: this.token, + }, + body: JSON.stringify(params), + }) + // 解析响应数据 + const res1 = await response.json() + const { component, event, resource } = res1.data + if (component.length > 100) { + } + for (const v of component) { + const position = { x: v.longitude, y: v.latitude, z: 0 } + this.degreesArray.push( + new CooGL.Vector3(position.x, position.y, position.z), + ) + this.addCommunityImage(v, position) + } + } + //计算多边形的包围盒,并将视角移动到包围盒的中心 + moveViewToPolygon(res) { + const positions = res.features[0].geometry.coordinates.flat(Infinity) + // 将扁平化的坐标数组重新组装成对象数组 + const positionObjects = [] + for (let i = 0; i < positions.length; i += 2) { + positionObjects.push({ + x: positions[i], + y: positions[i + 1], + z: 0, + }) + } + // 添加多边形图层 + const bs = CooGL.BoundingSphere.fromPoints( + CooGL.Vector3.degreesArray2cartesian(positionObjects), + ) + // 定位到包围盒中心 + this.viewer.camera.flyToBoundingSphere( + CooGL.Vector3.cartesian2degrees(bs.center), // 包围盒中心 + bs.radius, // 包围盒半径 + { + duration: 1.5, + offset: new CooGL.HeadingPitchRange(0, -Math.PI / 2, 0), + }, + ) + } + + /** + * 根据区域编码查找对应的区域信息及其子区域 + * @param {string} code - 区域编码(4/6/9/12位) + * @returns {object|null} 返回找到的区域信息对象,未找到返回null + */ + findRegionByCode(code, findSon = false) { + if (!code) return null + // 递归查找函数 + const findInTree = nodes => { + if (!nodes || !Array.isArray(nodes)) return null + + for (const node of nodes) { + // 匹配当前节点 + if (node.id == code) { + if (findSon) + return node.children.map(item => ({ ...item, children: null })) + return node + } + + // 递归查找子节点 + if (node.children) { + const found = findInTree(node.children) + if (found) return found + } + } + return null + } + return findInTree(this.regionResult) + } + + flyToArea([x, y], code) { + const z = this.levelHeight[code.length] + this.viewer.camera.flyTo({ + duration: 1.5, + destination: { + x, + y, + z, + }, + orientation: { + heading: 6.283185302776594, + pitch: -1.5707963267948966, + roll: 0, + }, + }) + } + compressFn() { + this.compressCancelFn = this.viewer.renderEvent.addEventListener(() => { + if (!this.degreesArray.length) return + const curlevel = + this.viewer.scene.globe._surface._debug.maxDepthVisited - 2 + if (this.level !== curlevel) { + this.level = curlevel + + if (curlevel > 16) { + if (this.opLevel < 17) { + this.opLevel = curlevel + for (const graphic of this.imageGraphicListOfCommunity) { + graphic.show = true + } + } + return + } + + const diff = this.levelOfMeter[this.level] + if (!diff) { + return + } + this.opLevel = this.level + const result = CooGL.PointsCompress.compute(this.degreesArray, diff) + const values = Object.values(result) + for (const graphic of this.imageGraphicListOfCommunity) { + const position = graphic._graphic.imageTextLabel.position + graphic.show = values.some( + ({ x, y }) => position.x === x && position.y === y, + ) + } + } + }) + } + //清除标注 + clearCommunity() { + for (const item of this.imageGraphicList) { + this.viewer.removeLayer(item) + } + this.imageGraphicList = [] + } + //清除画布 + clear() { + for (const item of this.imageGraphicList) { + this.viewer.removeLayer(item) + } + for (const item of this.polygonGraphicList) { + this.viewer.removeLayer(item) + } + for (const item of this.imageGraphicListOfCommunity) { + this.viewer.removeLayer(item) + } + this.imageGraphicListOfCommunity = [] + this.imageGraphicList = [] + this.polygonGraphicList = [] + this.degreesArray = [] + } + + clearCurrentAreaData() { + const id = 'layer-' + this.params.regionCode + const layer = this.viewer.getById(id) + const index = this.imageGraphicList.indexOf(layer) + if (layer) this.viewer.removeLayer(layer) + if (index > -1) this.imageGraphicList.splice(index, 1) + } + + limitCamHeight() { + this.viewer.scene.screenSpaceCameraController.maximumZoomDistance = 300000 + } + unlimitCamHeight() { + this.viewer.scene.screenSpaceCameraController.maximumZoomDistance = Infinity + } + destory() { + this.compressCancelFn?.() + this.rightPickHandler?.() + + this.clear() + this.clearCommunity() + this.pickHandler() + this.pickHandler = null + this.unlimitCamHeight() + } +} diff --git a/src/pages/Home_v_2504/components/MapContainer/index.tsx b/src/pages/Home_v_2504/components/MapContainer/index.tsx index ab59df3c..03c21a7a 100644 --- a/src/pages/Home_v_2504/components/MapContainer/index.tsx +++ b/src/pages/Home_v_2504/components/MapContainer/index.tsx @@ -10,7 +10,13 @@ import styles from './index.less'; import DistrictMapCustomized from './DistrictMapCustomized'; // import CooglMapDemo from './CooglMapDemo'; -const Map: Record = { +// 修改地图组件的属性类型 +interface MapComponentProps { + onMapInstanceChange?: (instance: any) => void; +} + +// 扩展地图组件类型,添加属性 +const Map: Record> = { // '1': CooglMapDemo, '1': DistrictMapCustomized, // '1': DistrictMap, @@ -37,10 +43,20 @@ const MapContainer = () => { const [topKey, setTopKey] = useState('1'); const { activePage } = useDashboard(); const isHomePage = activePage === 'home'; + + // 添加状态保存地图实例 + const [mapInstance, setMapInstance] = useState(null); + + // 处理地图实例变化 + const handleMapInstanceChange = (instance: any) => { + console.log('Map instance updated:', instance); + setMapInstance(instance); + }; const renderMap = useMemo(() => { const MapComponent = Map[topKey]; - return ; + // 传递 onMapInstanceChange 回调给地图组件 + return ; }, [topKey]); return ( @@ -61,8 +77,8 @@ const MapContainer = () => { {/* 地图上的最新事件和区域选择 */} - {/* 只在下钻页面显示区域选择按钮组 */} - {!isHomePage && } + {/* 只在下钻页面显示区域选择按钮组,传递地图实例 */} + {!isHomePage && } {/* 只在下钻页面显示导航栏 */} {!isHomePage && } -- GitLab