百度地图绘制电子围栏(包括移动端绘制操作)以及检测坐标是否在电子围栏内

news2024/12/25 3:47:57

由于本人在PC端仅使用了多边形绘制,但矩形跟多边形用法基本一样,圆形并未使用,如不符合读者需求也可以参考一下。

绘制后得到的数据可能不同,但绘制方法仅仅是传递的参数不同。

关于给坐标数组在地图绘制图形的效果在移动端部分包含。

一、绘制需要引入的文件

1.首先肯定包括咱们最需要的百度地图的文件

<script type="text/javascript" src="https://api.map.baidu.com/getscript?v=3.0&ak=申请的ak"></script> 
<!-- 注意切换为自己申请的ak -->

2.随后需要的则是我们绘制所需要的文件,一个css文件,一个js文件

<!-- JS文件 -->
<script type="text/javascript" src="http://api.map.baidu.com/library/DrawingManager/1.4/src/DrawingManager_min.js"></script>
<!-- CSS文件 -->
<link rel="stylesheet" href="http://api.map.baidu.com/library/DrawingManager/1.4/src/DrawingManager_min.css">

3.最后则是我们判断坐标是否在电子围栏内的js文件

<script type="text/javascript" src="http://api.map.baidu.com/library/GeoUtils/1.2/src/GeoUtils_min.js"></script>

做完这些准备后我们就可以在代码中进行使用了

二、PC端进行绘制

其实绘制也很简单,我使用的操作也更简单一些,这里主要使用的则是BMapLibrary - BMapLib.DrawingManager (baidu.com)

这个方法,同时这个方法可以控制显示一个工具条(说实话,感觉有点老...)

    onMounted(() => {
        init()
    })
    function init() {
        // 这里的 map 和 drawingManager 都是全局变量,在这里初始化,在其他地方都需要使用
        map = new BMap.Map("allmap", { enableMapClick: false });
        map.centerAndZoom(new BMap.Point(103.388611, 35.563611), 5);
        map.setCurrentCity("北京");
        map.enableScrollWheelZoom();
        // 初始化部分,根据自己需求适当修改

        drawingManager = new BMapLib.DrawingManager(map, {
            isOpen: false, // 是否开启绘制模式,肯定是我们交互之后才需要开启了,所以要先关闭
            enableDrawingTool: false, // 是否显示工具栏,默认开启,但是太丑了就不开了
        });
        drawingManager.addEventListener('overlaycomplete', overlaycomplete); // 绘制完成事件
    }

    // 绘制完成事件的处理函数(获取绘制的数据)
    function overlaycomplete(e) {
        // e.overlay.Po 就是绘制的数据,多边形和矩形的数据都是数组包含各个角的坐标,圆形则不太了解了
        // 这里可以根据自己的业务需求进行处理
        console.log(e.overlay.Po);
    };

    // 绘制图形的函数
    function setDrawingMode(mode) {
        // mode:代表我们需要绘制的图形
        // BMAP_DRAWING_POLYGON:多边形
        // BMAP_DRAWING_RECTANGLE:矩形
        // BMAP_DRAWING_CIRCLE:圆形
        // 这里可以根据自己的业务需求进行传参
        drawingManager.setDrawingMode(mode);
        // 开启绘制模式
        drawingManager.open();
    }

这里我们自定义按钮给 setDrawingMode 函数传参即可,从 overlayconplete 函数中获取绘制结果。

注意:因为多边形双击即可完成绘制,但其他图形不可以,我们可以添加一个完成绘制的按钮,函数中使用 drawingManager.close() 即可

三、移动端进行绘制

这里先讲一下移动端的思路,如果不适合或者感觉有更好的想法可以在评论区发表建议哈。

感觉移动端整体还是不适合做这个效果的,也有可能是我太菜了。

整体思路是:自定义添加按钮 => 点击地图 => 获取点击坐标 => 根据坐标生成一个长宽为1米的矩形 => 绘制到地图上,同时有一个弹框,可以修改宽高,修改后,确定按钮则会将数据保存,取消按钮则取消绘制。

在上述的第二个步骤会禁用地图的拖拽和双指操作缩放,因为会影响地图的点击事件,点击时移动一点都不会触发点击事件,触发拖拽事件,很麻烦。

    // 这里不展示弹框代码了,直接展示变量,默认为1米
    const formInline = ref({
        wide: 1,
        long: 1
    });
    // 自定义的添加围栏函数
    function addFence() {
        map.disableDragging(); // 禁止拖动地图
        map.disablePinchToZoom(); // 禁用双指操作缩放
        map.addEventListener('click', mapClick);
    };
    // 地图的点击事件
    const pointRectangle = ref();
    function mapClick(e) {
        // pointRectangle全局变量,后续封装的函数会用到
        pointRectangle.value = { lng: e.point.lng, lat: e.point.lat };
        let point = new BMap.Point(pointRectangle.value.lng, pointRectangle.value.lat);
        changeRect();
    };

    const EARTH_RADIUS = 6378137;
    function metersToLatitudeDegrees(meters) {
        return meters / (EARTH_RADIUS * Math.PI / 180);
    };

    function metersToLongitudeDegrees(meters, latitude) {
        const metersPerDegree = EARTH_RADIUS * Math.cos(latitude * Math.PI / 180) * Math.PI / 180;
        return meters / metersPerDegree;
    };
    // 根据点击的那个坐标点以及长宽,计算围栏的四个点坐标,并画出围栏
    const rectangularBox = ref(undefined);
    function changeRect() {
        let centerLat = pointRectangle.value.lat;
        let centerLng = pointRectangle.value.lng;
        let latOffset = metersToLatitudeDegrees(formInline.value.long * 500);
        let lngOffset = metersToLongitudeDegrees(formInline.value.wide * 500, centerLat);
        const pointArr = [
            new BMap.Point(centerLng + lngOffset, centerLat + latOffset),
            new BMap.Point(centerLng - lngOffset, centerLat + latOffset),
            new BMap.Point(centerLng - lngOffset, centerLat - latOffset),
            new BMap.Point(centerLng + lngOffset, centerLat - latOffset),
        ];
        // rectangularBox 全局变量,我们取消按钮的移出则是传入这个数据
        rectangularBox.value = new BMap.Polygon(pointArr);
        map.addOverlay(rectangularBox.value);
        // 根据绘制的图形适当修改地图的可视区域
        map.setViewport(pointArr);
    };

    // 自定义确定按钮的点击事件
    function onSub() {
        map.removeEventListener('click', mapClick);
        formInline.value = {
            wide: 1,
            long: 1
        };
        // 溢出监听、重置数据
        map.enableDragging(); // 启用拖动地图
        map.enablePinchToZoom(); // 启用双指操作缩放
        // 可以根据自己需求添加更多的功能
    };

    // 自定义取消按钮的点击事件
    function onCancel() {
        map.removeEventListener('click', mapClick);
        formInline.value = {
            wide: 1,
            long: 1
        };
        map.removeOverlay(rectangularBox.value);
        rectangularBox.value = undefined;
        map.centerAndZoom(new BMap.Point(103.388611, 35.563611), 5); //初始显示中国。
    };

四、删除电子围栏

删除的话就展示两句代码了,很好理解

    const pointArr = [
        new BMap.Point(centerLng + lngOffset, centerLat + latOffset),
        new BMap.Point(centerLng - lngOffset, centerLat + latOffset),
        new BMap.Point(centerLng - lngOffset, centerLat - latOffset),
        new BMap.Point(centerLng + lngOffset, centerLat - latOffset),
    ];
    // new BMap.Polygon的参数则必须是通过new BMap.Point创建的点数组
    rectangularBox.value = new BMap.Polygon(pointArr);
    // removeOverlay的参数必须是通过new BMap.Polygon创建的对象
    map.removeOverlay(rectangularBox.value);

五、检测点坐标是否在电子围栏内

这里使用的是咱们引用的 GeoUtils

    // point 是一个通过new BMap.Point(x, y)创建的点对象
    // pointList 是一个通过new BMap.Polygon(pointList)创建的多边形对象,和删除所需要的数据同理
    // 返回的数据未布尔值, true 表示在多边形内 false 表示在多边形外
    let flag = BMapLib.GeoUtils.isPointInPolygon(point, pointList);

那么以上便是本次的分享了,如果有什么问题或建议,欢迎评论交流

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

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

相关文章

【读书笔记-《30天自制操作系统》-14】Day15

本篇内容开始讲解多任务。本篇内容结构很简单&#xff0c;先讲解任务切换的原理&#xff0c;再讲解任务切换的代码实践。但是涉及到的知识不少&#xff0c;理解上也有些难度。 1. 任务切换与多任务原理 1.1 多任务与任务切换 所谓多任务&#xff0c;指的是操作系统同时运行多…

ambari-hdp启动yarn报错Corruption: checksum mismatch

ambari-hdp启动yarn报错Corruption: checksum mismatch 页面报错 Traceback (most recent call last):File "/var/lib/ambari-agent/cache/stacks/HDP/3.0/services/YARN/package/scripts/nodemanager.py", line 102, in <module>Nodemanager().execute()Fil…

万字文档带你走进Python的世界

目录 Python基本使用语法 老生常谈 Python中的注释 Python变量 定义变量 变量类型 Python变量的特点 Python中的输入与输出 Python中的运算符 算术运算符 /和// **运算符 关系运算符 逻辑运算符 赋值运算符 Python运算符优先级 Python分支语句 if语句和if-else语句 if-else if-…

Java | Leetcode Java题解之第386题字典序排数

题目&#xff1a; 题解&#xff1a; class Solution {public List<Integer> lexicalOrder(int n) {List<Integer> ret new ArrayList<Integer>();int number 1;for (int i 0; i < n; i) {ret.add(number);if (number * 10 < n) {number * 10;} els…

Datawhale X 李宏毅苹果书 AI夏令营

文章目录 我认为苹果书是最好的深度学习原理教材 第三章开篇讲的就是为什么深度学习模型会优化失败&#xff0c;这个问题其它在我们训练深度学习模型的过程中是非常常见的一种现象&#xff1a;明明使用了更加深层的结构&#xff0c;但它的表现与之前一样&#xff0c;有时甚至不…

企业IT服务管理(ITSM)的实践与探索

随着信息技术的飞速发展&#xff0c;企业对IT服务管理&#xff08;ITSM&#xff09;的需求也日益增长。在这个背景下&#xff0c;某大型集团&#xff08;以下简称“该机构”&#xff09;逐步构建了完善的IT服务管理体系&#xff0c;其发展历程和实践经验对于广大运维团队而言&a…

OceanBase V4.2解析:如何用迭代器 Generator快速生成任意数据

前言 OceanBase 4.2 版本新增了迭代器 generator 函数。尽管这一功能在数据库领域中已属于通用能力&#xff0c;postgresql 也提供了类似的函数&#xff0c;然而&#xff0c;与MySQL和Oracle数据库在默认情况下是需要用户额外编写函数来实现的。OceanBase 4.2 的这一更新也是满…

鸿蒙(API 12 Beta6版)图形【AR物体摆放】 AR引擎服务

概要 本章节通过AR Engine识别设备周围的平面&#xff0c;并允许用户在平面上放置虚拟物体&#xff0c;实现虚拟和现实的融合。AR物体摆放可用于虚拟家具、数字展厅等应用&#xff0c;给用户提供虚实结合的新体验。通过本示例&#xff0c;您可以学习并掌握如何使用AR Engine开…

刷题记录(2)

1. HWOD机试 - 模拟消息队列(100) package com.yue.test;import org.junit.Test;import java.util.ArrayList; import java.util.Arrays; import java.util.LinkedList; import java.util.List;/*** Author: 夜雨* Date: 2021-12-08-10:31* Description:* Version 1.0*/ public…

C#编译成32和64位的区别

C#编译成32和64位的区别 背景 C#32位客户端项目在把代码提交到客户端之后&#xff0c;jinkens直接崩掉了。原因是内存占用100%运维同学建议改成64位&#xff0c;理由是电脑内存大&#xff0c;客观条件IT不给扩。那么在同一台电脑上&#xff0c;32位和64位在编译过程中有什么区…

【DEV工具-IDEA】idea的光标变成黑块了?

项目场景&#xff1a; 解决&#xff1a;windows&#xff1a;按一下insert键。

Python获取次幂数据公众号榜单数据

公众号排行榜,wx公众号排行榜,原创排行榜,赞赏排行榜,评论排行榜 教程仅供参考,请勿滥用,由此带来的法律责任,需由自己承担。 一、运行效果 二、程序代码 #!/usr/bin/python # -*- coding: UTF-8 -*- """ @author: Roc-xb """import request…

Java学习第六天

Java进阶知识面向对象 static&#xff1a;是静态的意思&#xff0c;可以修饰成员变量&#xff0c;表示该成员变量在内存中只存储一份&#xff0c;可以被共享访问。 静态成员变量&#xff08;有static修饰&#xff0c;属于类&#xff0c;内存中加载一次&#xff09;&#xff1a…

三元里等你!融合三个经典模型!Transformer-LSTM-SVM多变量时间序列预测(Matlab)

三元里等你&#xff01;融合三个经典模型&#xff01;Transformer-LSTM-SVM多变量时间序列预测&#xff08;Matlab&#xff09; 目录 三元里等你&#xff01;融合三个经典模型&#xff01;Transformer-LSTM-SVM多变量时间序列预测&#xff08;Matlab&#xff09;效果一览基本介…

I2C总线的标准收发代码

结合I2C总线协议的知识&#xff0c;我们可以知道I2C写数据由一下10个步骤组成。 第一步&#xff0c;发送一个起始信号。 第二步&#xff0c;发送7bit从机地址&#xff0c;即OZ9350的地址。此处需要注意&#xff0c;发送数据时&#xff0c;无法发送7bit数据&#xff0c;此处发…

求和放大器(单位/非单位增益加法器+比例加法器)+运算放大器实现积分器和微分器

2024-9-2&#xff0c;星期一&#xff0c;22:00&#xff0c;天气&#xff1a;晴转雨&#xff0c;心情&#xff1a;晴。新的一周开始了&#xff0c;新的一个月又开始啦&#xff0c;希望大家开开心心&#xff0c;以崭新的面貌迎接中秋和十一假期&#xff01;废话不多说&#xff0c…

LinkAI工作流支持广场访问和api调用啦

什么是工作流 LinkAI工作流&#xff08;WorkFlow&#xff09;是一种灵活的智能体搭建方式。可以自由选择「大模型、应用、知识库、插件、意图识别、转人工、渠道消息发送」等多种原子能力&#xff0c;通过可视化拖拉拽的方式进行组合编排&#xff0c;零代码搭出一个业务流程。…

PPT制作加速器:3款工具插件的演示文稿制作更高效

IvyhTools英豪插件 IvyhTools是一款功能强大的PPT插件&#xff0c;主要用于辅助用户进行各种PPT编辑和处理操作。该插件具备以下主要功能&#xff1a; 字体编辑&#xff1a;用户可以对PPT中的字体进行编辑和调整。 动图录制&#xff1a;支持录制动态图像&#xff0c;方便用户在…

深度学习(四)-卷积神经网络

神经网络局限 不考虑数据形状 未考虑数据的“形状”&#xff0c;会破坏数据空间结构。例如&#xff0c;输入数据是图像时&#xff0c;图像通常是高长通道方向上的3维形状。但是&#xff0c;向全连接层输入时&#xff0c;需要将3维数据拉平为1维数据 参数庞大 全连接网络参数…

中小企业怎么选择MES:专用MES、集成MES和可配置MES

专用MES、集成MES和可配置MES是MES&#xff08;制造执行系统&#xff09;在不同发展阶段和应用场景下的三种主要形式。它们各自具有不同的特点和应用优势&#xff0c;下面将分别进行详细介绍。 专用MES 定义与特点&#xff1a; 专用MES是针对特定行业或特定生产流程而设计的…