高德地图在Vue3中的使用方法

news2025/5/15 6:49:21

1.地图初始化

        容器创建:通过 <div> 标签定义地图挂载点。

<div id="container" style="height: 300px; width: 100%; margin-top: 10px;"></div>

        密钥配置:绑定高德地图安全密钥,确保 API 合法调用。

        参数配置:定义地图缩放比例、初始中心点经纬度(数组格式 [lng, lat])。

        标记点初始化:创建 Marker 实例用于后续定位

// 初始化地图(完全异步化)
const initMap = async () => {
  // 确保DOM已渲染
  await nextTick();
  // 检查容器是否存在
  const container = document.getElementById('container');
  if (!container) {
    console.error('地图容器不存在,可能对话框尚未完全渲染');
    return false;
  }
  if (mapLoaded.value && map.value) return true; // 如果已经加载过且地图已初始化,直接返回
  try {
    // 设置安全配置
    (window as any)._AMapSecurityConfig = {
      securityJsCode: '76f83426ac3692c440bc1efd2b15fff6'
    };
    // 动态加载高德地图JS
    await new Promise((resolve, reject) => {
      if ((window as any).AMap) return resolve(true); // 如果已经加载过,直接返回
      const script = document.createElement('script'); // 创建script标签
      script.src = 'https://webapi.amap.com/maps?v=2.0&key=f10b74bad0d89466b9e174aaefb9a3f7&plugin=AMap.Geocoder';
      script.onload = resolve; // 加载成功后调用 resolve 函数
      script.onerror = reject; // 加载失败后调用 reject 函数
      document.head.appendChild(script);
    });
    // 再次检查容器是否存在 (id 是唯一的)
    if (!document.getElementById('container')) {
      console.error('地图容器在脚本加载后不存在');
      return false;
    }
    // 初始化地图
    map.value = new (window as any).AMap.Map('container', {
      zoom: 13, // 初始缩放级别为 13
      center: [116.397428, 39.90923], // 初始中心点为 [116.397428, 39.90923],即天安门广场
    });
    // 初始化标记
    marker.value = new (window as any).AMap.Marker({
      map: map.value
    });
    mapLoaded.value = true;
    return true; // 返回 true 表示成功加载
  } catch (error) {
    console.error('地图初始化失败:', error);
    proxy?.$modal.msgError('地图加载失败');
    return false; // 返回 false 表示加载失败
  }
};

2.地址选择与解析

级联选择器:选择省、市、县三级行政区(普通省份为三级,直辖市可能缺少县级)。

详细地址输入:用户手动补充街道、楼栋等详细信息。

地址合并:将级联选择结果与输入框内容拼接成完整地址字符串。

地理编码:调用高德地图 API 将地址转为经纬度。

更新标记点:将解析后的经纬度更新到地图标记。

防抖函数:控制高频触发的事件或函数的执行频率,确保在事件连续快速触发时,只在最后一次触发后的一段时间内执行一次目标操作。

// 防抖函数
const debounce = (fn: Function, delay: number) => {
  let timer: ReturnType<typeof setTimeout>; // 用于存储定时器的返回值
  return function (this: any, ...args: any[]) {
    clearTimeout(timer); // 清除上一次的定时器
    timer = setTimeout(() => fn.apply(this, args), delay); // 设置新的定时器
  };
};
// 处理地址选择变化(核心逻辑)
const handleAddressChange = debounce(async () => {
  try {
    await initMap();
    const selectedLabels = form.value.addressArray?.map(code => codeToText[code]) || []; // 获取选中的省市区
    const detailAddress = form.value.addressDetail?.trim() || ''; // 获取详细地址
    const fullAddress = [...selectedLabels, detailAddress].join(''); // 拼接完整地址
    // 这里赋值给address
    form.value.address = fullAddress;
    if (!fullAddress) return; // 如果地址为空,不进行解析
    const geocoder = new (window as any).AMap.Geocoder({
      city: selectedLabels[0] || '' // 根据选中的省设置地图上的城市
    });
    interface GeocoderResult {
      status: string; // 状态码
      result: any; // 解析结果
    }
    const { status, result } = await new Promise<GeocoderResult>((resolve, reject) => {
      geocoder.getLocation(fullAddress, (status, result) => {
        if (status === 'complete') resolve({ status, result }); // 成功解析
        else reject(new Error(result?.info || '地址解析失败')); // 失败
      });
    });
    if (status === 'complete' && result.geocodes?.length) {
      // geocodes 是高德地图API返回的地理编码结果数组,它包含了地址解析后的地理信息。
      /** geocodes:[{
       *      location : 包含经纬度坐标(lng, lat)
       *      formattedAddress : 格式化后的完整地址
       *      addressComponent : 地址组成信息(省、市、区等)
       *      level : 地址匹配的精确度级别
       * }]
       */
      const lnglat = result.geocodes[0].location;
      map.value?.setCenter(lnglat); // 将地图中心移动到解析后的位置
      marker.value?.setPosition(lnglat); // 将标记移动到解析后的位置
      // 更新表单坐标
      form.value.longitude = lnglat.lng;
      form.value.latitude = lnglat.lat;
    }
  } catch (error) {
    console.error('地址解析错误:', error);
    proxy?.$modal.msgError(error.message);
  }
}, 500);

3. 直辖市特殊处理以及回显

问题背景:直辖市(如北京、上海)的级联数据层级缺失(如 北京市 -> 北京市)。
直接拼接会导致地址重复(如 北京市北京市朝阳区xx路)。

修复逻辑:通过判断省、市名称是否相同,动态修正地址字符串

/** 修改按钮操作 */
const handleUpdate = async (row?: TenantVO) => {
  reset();
  await getTenantPackage();
  const _id = row?.id || ids.value[0];
  const res = await getTenant(_id);
  Object.assign(form.value, res.data);
  // 确保对话框完全打开
  dialog.visible = true;
  dialog.title = '修改租户';
  try {
    // 等待地图初始化完成
    await initMap();
    // 判读经纬度是否有值
    if (form.value.longitude && form.value.latitude) {
      map.value?.setCenter([form.value.longitude, form.value.latitude]); // 将地图中心移动到解析后的位置
      marker.value?.setPosition([form.value.longitude, form.value.latitude]); //  将标记移动到解析后的位置
      // 执行逆地理编码
      // 传入经纬度数组 [form.value.longitude, form.value.latitude] ,获取地址信息
      const geocoder = new (window as any).AMap.Geocoder();
      const { status, result } = await new Promise((resolve, reject) => {
        geocoder.getAddress([form.value.longitude, form.value.latitude], (status, result) => {
          if (status === 'complete') resolve({ status, result });
          else reject(new Error(result?.info || '地址解析失败'));
        });
      });
      if (status === 'complete' && result.regeocode) {
        const address = result.regeocode.formattedAddress;
        const addressComponent = result.regeocode.addressComponent;
        const province = addressComponent.province;
        const city = addressComponent.city || province; // 如果没有city,使用province代替 因为直辖市没有city字段
        /**
         * addressComponent = {
         *    province: "北京市",
         *    city: "", // 直辖市可能没有city字段
         *    district: "朝阳区"
         *   }
         */
        const district = addressComponent.district;
        // 更新表单中的地址信息
        form.value.address = address;
        // 将省市区转换为级联选择器需要的code格式
        const findCode = (name, data) => {
          // name 为省市区名称,data 为 regionData 数据包含了中国的省、市、区三级行政区划信息
          for (const item of data) {
            if (item.label === name) return item.value;
            if (item.children && item.children.length > 0) {
              const childCode = findCode(name, item.children); // 递归查找子级
              if (childCode) return childCode; // 如果找到了,返回code
            }
          }
          return null;
        };
        const provinceCode = findCode(province, regionData); // 查找省的code
        let cityCode = findCode(city, regionData); // 查找市的code
        const districtCode = findCode(district, regionData); // 查找区的code
        // 设置级联选择器值
        if (provinceCode && cityCode && districtCode) {
          cityCode = cityCode.length == 2 ? cityCode + '01' : cityCode; // 给直辖市的市code添加01
          form.value.addressArray = [provinceCode, cityCode, districtCode];
        }
        // 设置详细地址
        // 需要三元运算符,因为直辖市会重复显示,所以进行判断 直辖市的 province 和 city 是一样的
        province == city ? form.value.addressDetail = address.replace(`${province}${district}`, '') : form.value.addressDetail = address.replace(`${province}${city}${district}`, '');
        // console.log(province, city, district);
      }
    } else {
      // 默认显示天安门广场
      const tiananmenPosition = [116.397428, 39.90923];
      map.value?.setCenter(tiananmenPosition);
      marker.value?.setPosition(tiananmenPosition);
    }
  } catch (error) {
    console.error('地图操作错误:', error);
    if (!form.value.longitude || !form.value.latitude) {
      // 如果是没有经纬度的情况,不显示错误提示
      return;
    }
    proxy?.$modal.msgError('地图加载失败,请检查网络连接或稍后再试');
  }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2375886.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Llama:开源的急先锋

Llama:开源的急先锋 Llama1&#xff1a;开放、高效的基础语言模型 Llama1使用了完全开源的数据&#xff0c;性能媲美GPT-3&#xff0c;可以在社区研究开源使用&#xff0c;只是不能商用。 Llama1提出的Scaling Law 业内普遍认为如果要达到同一个性能指标&#xff0c;训练更…

“redis 目标计算机积极拒绝,无法连接” 解决方法,每次开机启动redis

如果遇到以上问题 先打开“服务” 找到App Readiness 右击-启动 以管理员身份运行cmd&#xff0c;跳转到 安装redis的目录 运行&#xff1a;redis-server.exe redis.windows.conf 以管理员身份打开另一cmd窗口&#xff0c;跳转到安装redis的目录 运行&#xff1a;redis-…

LeetCode 热题 100 35.搜索插入位置

目录 题目&#xff1a; 题目描述&#xff1a; 题目链接&#xff1a; 思路&#xff1a; 核心思路&#xff1a; 思路详解&#xff1a; 代码&#xff1a; Java代码&#xff1a; 题目&#xff1a; 题目描述&#xff1a; 题目链接&#xff1a; 35. 搜索插入位置 - 力扣&…

从 “学会学习” 到高效适应:元学习技术深度解析与应用实践

一、引言&#xff1a;当机器开始 “学会学习”—— 元学习的革命性价值 在传统机器学习依赖海量数据训练单一任务模型的时代&#xff0c;元学习&#xff08;Meta Learning&#xff09;正掀起一场范式革命。 这项旨在让模型 “学会学习” 的技术&#xff0c;通过模仿人类基于经验…

AI开发者的算力革命:GpuGeek平台全景实战指南(大模型训练/推理/微调全解析)

目录 背景一、AI工业化时代的算力困局与破局之道1.1 中小企业AI落地的三大障碍1.2 GpuGeek的破局创新1.3 核心价值 二、GpuGeek技术全景剖析2.1 核心架构设计 三、核心优势详解‌3.1 优势1&#xff1a;工业级显卡舰队‌‌‌3.2 优势2&#xff1a;开箱即用生态‌3.2.1 预置镜像库…

AWS SNS:解锁高并发消息通知与系统集成的云端利器

导语 在分布式系统架构中&#xff0c;如何实现高效、可靠的消息通知与跨服务通信&#xff1f;AWS Simple Notification Service&#xff08;SNS&#xff09;作为全托管的发布/订阅&#xff08;Pub/Sub&#xff09;服务&#xff0c;正在成为企业构建弹性系统的核心组件。本文深度…

【PmHub后端篇】PmHub集成 Sentinel+OpenFeign实现网关流量控制与服务降级

在微服务架构中&#xff0c;保障服务的稳定性和高可用性至关重要。本文将详细介绍在 PmHub 中如何利用 Sentinel Gateway 进行网关限流&#xff0c;以及集成 Sentinel OpenFeign 实现自定义的 fallback 服务降级。 1 熔断降级的必要性 在微服务架构中&#xff0c;服务间的调…

2025最新出版 Microsoft Project由入门到精通(八)

目录 查找关键路径方法 方法1:格式->关键任务 方法2:插入关键属性列 方法3&#xff1a;插入“可宽延的总时间”进行查看&#xff0c;>0不是关键路径&#xff0c;剩余的全是关键路径 方法4:设置关键路径的工作表的文本样式​编辑 方法5&#xff1a;突出显示/筛选器…

3.0/Q2,Charls最新文章解读

文章题目&#xff1a;Development of a visualized risk prediction system for sarcopenia in older adults using machine learning: a cohort study based on CHARLS DOI&#xff1a;10.3389/fpubh.2025.1544894 中文标题&#xff1a;使用机器学习开发老年人肌肉减少症的可视…

使用matlab进行数据拟合

目录 一、工作区建立数据 二、曲线拟合器(在"APP"中) 三、曲线拟合函数及参数 四、 在matlab中编写代码 一、工作区建立数据 首先&#xff0c;将数据在matlab工作区中生成。如图1所示&#xff1a; 图 1 二、曲线拟合器(在"APP"中) 然后&#xff0c;…

分布式1(cap base理论 锁 事务 幂等性 rpc)

目录 分布式系统介绍 一、定义与概念 二、分布式系统的特点 三、分布式系统面临的挑战 四、分布式系统的常见应用场景 CAP 定理 BASE 理论 BASE理论是如何保证最终一致性的 分布式锁的常见使用场景有哪些&#xff1f; 1. 防止多节点重复操作 2. 资源互斥访问 3. 分…

Myshell与清华联合开源TTS模型OpenVoiceV2,多语言支持,风格控制进一步增强~

项目背景 开发团队与发布 OpenVoice2 由 MyShell AI&#xff08;加拿大 AI 初创公司&#xff09;与 MIT 和清华大学的研究人员合作开发&#xff0c;技术报告于 2023 年 12 月发布 &#xff0c;V2 版本于 2024 年 4 月发布 。 项目目标是提供一个高效、灵活的语音克隆工具&…

YOLO11解决方案之热力图探索

概述 Ultralytics提供了一系列的解决方案,利用YOLO11解决现实世界的问题,包括物体计数、模糊处理、热力图、安防系统、速度估计、物体追踪等多个方面的应用。 使用YOLO11生成的热力图把复杂的数据转换成生动的彩色编码矩阵。这种可视化工具采用色谱来表示不同的数据值,暖色…

如何在终端/命令行中把PDF的每一页转换成图片(PNG)

今天被对象安排了一个任务&#xff1a; 之前自己其实也有这个需要&#xff0c;但是吧&#xff0c;我懒&#xff1a;量少拖拽&#xff0c;量大就放弃。但这次躲不过去了&#xff0c;所以研究了一下有什么工具可以做到这个需求。 本文记录我这次发现的使用 XpdfReader 的方法。…

计算机系统结构——Cache性能分析

一、实验目的 加深对Cache的基本概念、基本组织结构以及基本工作原理的理解。掌握Cache容量、相联度、块大小对Cache性能的影响。掌握降低Cache不命中率的各种方法以及这些方法对提高Cache性能的好处。理解LRU与随机法的基本思想以及它们对Cache性能的影响。 二、实验平台 实…

GESP2023年12月认证C++八级( 第三部分编程题(2)大量的工作沟通)

参考程序&#xff1a; #include <cstdio> #include <cstdlib> #include <cstring> #include <algorithm> #include <string> #include <map> #include <iostream> #include <cmath> #include <vector> #include <qu…

015枚举之滑动窗口——算法备赛

滑动窗口 最大子数组和 题目描述 给你一个整数数组 nums &#xff0c;请你找出一个具有最大和的连续子数组&#xff08;子数组最少包含一个元素&#xff09;&#xff0c;返回其最大和。 原题链接 思路分析 见代码注解 代码 int maxSubArray(vector<int>& num…

新型深度神经网络架构:ENet模型

语义分割技术能够为图像中的每个像素分配一个类别标签&#xff0c;这对于理解图像内容和在复杂场景中找到目标对象至关重要。在自动驾驶和增强现实等应用中&#xff0c;实时性是一个硬性要求&#xff0c;因此设计能够快速运行的卷积神经网络非常关键。 尽管深度卷积神经网络&am…

【免杀】C2免杀技术(三)shellcode加密

前言 shellcode加密是shellcode混淆的一种手段。shellcode混淆手段有多种&#xff1a;加密&#xff08;编码&#xff09;、偏移量混淆、UUID混淆、IPv4混淆、MAC混淆等。 随着杀毒软件的不断进化&#xff0c;其检测方式早已超越传统的静态特征分析。现代杀软往往会在受控的虚…