openlayers动态添加自定义div图层 具有筛选功能 和浮窗

2023-05-16

openlayers动态加自定义div图层 并且具有筛选功能 和浮窗效果,代码拿过去就可以用

联系方式:45664741@qq.com
个人博客: flot.top

效果图:
在这里插入图片描述

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
    <link
      crossorigin="anonymous"
      integrity="sha384-iYkPgcf1wF1oXt3CosKxNMQJgQUC60vjUJM+HANnq9ULJXg8DLX/ywPXMZHto90Q"
      href="https://lib.baomitu.com/ol3/4.6.5/ol.css"
      rel="stylesheet"
    />
    <script
      crossorigin="anonymous"
      integrity="sha384-207r7vJ98mMvHGcgRttnhkJLl6RU+DuD+ttp1G5k4ShCtw03LfZIIDNNWDGU8m2F"
      src="https://lib.baomitu.com/ol3/4.6.5/ol.js"
    ></script>
    <script
      crossorigin="anonymous"
      integrity="sha384-6ePHh72Rl3hKio4HiJ841psfsRJveeS+aLoaEf3BWfS+gTF0XdAqku2ka8VddikM"
      src="https://lib.baomitu.com/jquery/1.11.3/jquery.min.js"
    ></script>
    <style>
      #map {
        width: 1200px;
        height: 700px;
        position: absolute;
      }
      .my-map-overlay {
        width: 100px;
        height: 100px;
        /* border: 1px solid; */
      }
      .my-map-overlay-tag {
        width: 100px;
        height: 100px;
        background: url("")
          no-repeat;
        animation: identifier steps(4) 1s infinite;
      }

      #map-overlay-info {
        background: #fff;
        position: absolute;
        border-radius: 4px;
        display: none;
        padding: 10px;
        z-index: 1;
      }
      #map-overlay-info .info-shape {
        position: absolute;
        top: 0.5em;
        border: 1px solid #fff;
        height: 0;
        width: 0;
        border-left: 4px solid transparent;
        border-right: 4px solid #fff;
        border-top: 3px solid transparent;
        border-bottom: 3px solid #fff;
        left: -8px;
      }

      @keyframes identifier {
        0% {
          background-position: 0px 0;
        }
        100% {
          background-position: -400px 0;
        }
      }
    </style>
  </head>
  <body>
    <input type="text" id="myInput" oninput="handleInput()" />根据名称筛选
    <div id="map"></div>
    <script>
      var option = {
        map: null,
        overlayItem: {},
        list: [],
        // 初始化地图
        initMap() {
          const osm = new ol.layer.Tile({
            source: new ol.source.OSM(),
          });
          // 切片方式加载地图
          const tiled = new ol.layer.Tile({
            // visible: false,
            source: new ol.source.TileWMS({
              url: "https://ahocevar.com/geoserver/wms",
              params: { LAYERS: "topp:states", TILED: true },
              serverType: "geoserver",
              // Countries have transparency, so do not fade tiles:
              transition: 0,
            }),
          });

          const map = new ol.Map({
            target: "map",
            layers: [osm, tiled],
            view: new ol.View({
              center: [-10997148, 4569099],
              zoom: 4,
            }),
          });
          this.map = map;
        },
        // 新增自定义图层
        mapAddOverlay() {
          // 创建div的函数
          function createDiv({ id, status, statusType }, index) {
            // 创建一个div元素
            var elementDiv = document.createElement("div");
            elementDiv.className = "my-map-overlay";
            elementDiv.innerHTML = `
                  <div class='my-map-overlay-tag ${formatInfoClass(status)}'>
                    <span class='my-map-overlay-status'>${
                      status === "故障" ? statusType : status
                    }${id}</span>
                  </div>
                  `;
            // 设置div元素的cid属性
            elementDiv.setAttribute("myid", id);
            return elementDiv;
          }
          // 根据状态 加载对应的类名
          function formatInfoClass(status) {
            var className = "";
            switch (status) {
              case "工作中":
                className = "green";
                break;
              case "未使用":
                className = "gray";
                break;
              case "故障":
                className = "red";
                break;
            }
            return className;
          }
          // 循环创建循环覆盖层
          this.list.forEach((item, index) => {
            // 新建一个覆盖层
            var newMarker = new ol.Overlay({
              // 设置位置为当前节点的坐标
              position: [item.lon, item.lat],
              // 设置覆盖层与位置之间的匹配方式
              positioning: "bottom-left",
              // 偏移量  默认值是 [0, 0] 正值分别向右和向下偏移
              // offset: [-50, 20],
              // 覆盖层元素
              element: createDiv(item, index),
              // 事件传播到地图视点的时候是否应该停止
              stopEvent: false,
            });
            this.map.addOverlay(newMarker);
          });

          // 自定义div列表
          const domlist = document.getElementsByClassName("my-map-overlay");
          this.mapOverlayOver(domlist);
        },
        /** mapRemoveLayer
         * @function  删除自定义div图层
         */
        mapRemoveOverlays() {
          this.map.getOverlays().clear();
        },
        /**
         * 监听地图自定义图层的单击/移入/移出事件
         * @function mapOverlayOver
         * @param {Array} domlist  元素列表
         */
        mapOverlayOver(domlist) {
          Array.from(domlist).forEach((item) => {
            // 鼠标点击事件
            item.addEventListener("click", (e) => {
              const ev = e || window.event;
              const target = ev.path.filter(
                ({ className }) => className === "my-map-overlay"
              )[0];
              // // 获取在div上设置的myid
              const myid = target.getAttribute("myid");
              this.setMapZoomCenter(myid);
            });

            // 鼠标移入事件 创建浮窗
            item.addEventListener("mouseenter", (e) => {
              const ev = e || window.event;
              const target = ev.target || ev.srcElement;

              // 获取在div上设置的myid
              const myid = target.getAttribute("myid");
              // 在设备列表筛选符合条件的对象赋值
              const overlayItem = this.list.filter(({ id }) => id === myid);
              this.overlayItem = overlayItem[0];

              // getBoundingClientRect用于获取元素相对于视窗的位置集合
              const { top, left } = target.getBoundingClientRect();
              this.addTootipDom(top + 30, left + 100);
            });

            // 鼠标移出事件 删除浮窗
            item.addEventListener("mouseleave", (e) => {
              this.removeTootipDom();
              // mapOverlayInfo.setAttribute("style", "display: none");
            });
          });
        },
        /**
         * 设置地图的中心点和放大倍数
         * @function setMapZoomCenter
         * @param {String} ItemId  设备id
         */
        setMapZoomCenter(ItemId) {
          const overlayItem = this.list.filter(({ id }) => ItemId === id);
          this.map.getView().animate({
            center: [overlayItem[0].lon, overlayItem[0].lat],
            duration: 800,
            zoom: 6,
          });
        },
        addTootipDom(top, left) {
          // const mapOverlayInfo = document.getElementById("map-overlay-info");
          // if (mapOverlayInfo) {
          //   mapOverlayInfo.parentNode.removeChild(mapOverlayInfo);
          // }
          const parentDom = document.getElementById("map");
          const childDom = document.createElement("div");
          childDom.setAttribute("id", "map-overlay-info");
          childDom.innerHTML = `
      <div class="info-shape"></div>
        <p>名称:${this.overlayItem.stationName}</p>
        <p>状态:${this.overlayItem.status}</p>
        <p>故障名称:${this.overlayItem.statusType}</p>
      `;
          childDom.setAttribute(
            "style",
            `top: ${top}px; left: ${left}px; display: block`
          );
          parentDom.appendChild(childDom);
        },
         // 删除自 浮窗
        removeTootipDom() {
          const mapOverlayInfo = document.getElementById("map-overlay-info");
          mapOverlayInfo.parentNode.removeChild(mapOverlayInfo);
        },
      };
      // 添加浮窗
      option.initMap();
      option.mapAddOverlay();
      handleInput();
      // 筛选值发生变化
      function handleInput() {
        const val = document.getElementById("myInput").value;
        const list = [
          {
            Type: "客厅",
            lat: 5925601.900570343,
            lon: -13109428.402459577,
            stationName: "598客厅1",
            id: "kt-1",
            status: "工作中",
          },
          {
            Type: "客厅",
            lat: 6024213.9906833647,
            lon: -11040972.000962155,
            stationName: "598客厅2",
            id: "kt-2",
            status: "工作中",
          },
          {
            Type: "客厅",
            lat: 5058323.196052412,
            lon: -13293746.428576976,
            stationName: "598客厅3",
            id: "kt-3",
            status: "工作中",
          },
          {
            Type: "客厅",
            lat: 4893844.222583916,
            lon: -13195443.525842169,
            stationName: "598客厅4",
            id: "kt-4",
            status: "工作中",
          },
          {
            Type: "客厅",
            lat: 5008981.7571222,
            lon: -10393810.018663468,
            stationName: "598客厅5",
            id: "kt-5",
            status: "未使用",
          },
          {
            Type: "客厅",
            lat: 3659552.246270453,
            lon: -10819789.300954165,
            stationName: "598客厅6",
            id: "kt-6",
            status: "未使用",
          },
          {
            Type: "客厅",
            lat: 4186335.64109729,
            lon: -12949686.157685628,
            stationName: "598客厅7",
            id: "kt-7",
            status: "故障",
            statusType: "温度异常",
          },
        ];
        option.list = list.filter(
          ({ stationName }) => stationName.indexOf(val) !== -1
        );
        option.mapRemoveOverlays() // 清除自定义图层,再重新加载自定义div图层
        option.mapAddOverlay();
      }
      // GIS坐标系  EPSG:4326转EPSG:3857
      function mercatorTolonlat(mercator) {
        var lonlat = {
          x: 0,
          y: 0,
        };
        var x = (mercator.x / 20037508.34) * 180;
        var y = (mercator.y / 20037508.34) * 180;
        y =
          (180 / Math.PI) *
          (2 * Math.atan(Math.exp((y * Math.PI) / 180)) - Math.PI / 2);
        lonlat.x = x;
        lonlat.y = y;
        return lonlat;
      }
      // 经纬度(EPSG:4326)转换EPSG:3857
      function lonLat2Mercator(lonlat) {
        var mercator = {
          x: 0,
          y: 0,
        };
        var earthRad = 6378137.0;
        mercator.x = ((lonlat.lng * Math.PI) / 180) * earthRad;
        var a = (lonlat.lat * Math.PI) / 180;
        mercator.y =
          (earthRad / 2) * Math.log((1.0 + Math.sin(a)) / (1.0 - Math.sin(a)));
        return mercator;
      }
	console.log(mercatorTolonlat({x:13525596.430109723,y:3663568.4585909187}));
     console.log(lonLat2Mercator({lng: 121.5025, lat: 31.247015}))
    </script>
  </body>
</html>

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

openlayers动态添加自定义div图层 具有筛选功能 和浮窗 的相关文章

  • Haffman编码(算法导论)

    上次算法导论课讲到了Haffman树 xff0c 笔者惊叹于Haffman编码的压缩效果 xff0c 故想自己亲自动手尝试写一个极简的Haffman压缩程序 首先 xff0c 我们来了解一下什么是Haffman编码 Haffman编码 赫夫
  • C++ Primer- 变量声明和定义的关系

    为了允许把程序拆分成多个逻辑部分来编写 xff0c C 43 43 语言支持分离式编译 xff08 separatecompilation xff09 机制 xff0c 该机制允许将程序分割为若干个文件 xff0c 每个文件可被独立编译 如
  • Compiler Lab2- 自制极简编译器

    笔者实现的这个超级迷你版编译器 xff08 词法分析 语法分析 生成中间代码 xff08 cpp xff09 xff09 仅支持四则运算 xff0c 功能真的是非常非常简单 xff0c 不过其中的流程 xff08 词法分析 gt 语法分析
  • 贪心算法实现最佳任务调度实验

    题目描述 一个单位时间任务是恰好需要一个单位时间完成的任务 给定一个单位时间任务的有限集S 关于S 的一个时间表用于描述S 中单位时间任务的执行次序 时间表中第1 个任务从时间0 开始执行直至时间1 结束 xff0c 第2 个任务从时间1
  • 在CentOS安装CMake

    你或许听过好几种 Make 工具 xff0c 例如 GNU Make xff0c QT 的 qmake xff0c 微软的 MS nmake xff0c BSD Make xff08 pmake xff09 xff0c Makepp xff
  • CSAPP Lab3- bufbomb

    实验目标 xff08 1 xff09 掌握函数调用时的栈帧结构 xff08 2 xff09 利用输入缓冲区的溢出漏洞 xff0c 将攻击代码嵌入当前程序的栈帧中 xff0c 使得程序执行我们所期望的过程 实验代码 xff08 1 xff09
  • 金蝶部署SpringBoot+vue项目总结

    金蝶部署SpringBoot 43 vue项目文章目录 文章目录 金蝶部署SpringBoot 43 vue项目文章目录前言一 金蝶部署中两个重要概念1 域2 应用基础上下文 二 部署准备1 前后台打包2 部署过程3 部署成功后的注意事项
  • Gazebo-Realsense 编译问题解决

    64 Gazebo仿真 Realsense 编译问题解决 参考 这篇文章配置 首先git clone代码到你的工作空间 git clone https github com nilseuropa realsense ros gazebo g
  • docker快捷添加用户组,远离sudo困扰

    使用docker命令时 xff0c 总是需要root权限 xff0c 同时还要验证密码 xff0c 过于麻烦便将root添加进docker分组 xff0c 远离sudo困扰 span class token comment 创建docker
  • 全注解下的SpringIoc 续4-条件装配bean

    Spring Boot默认启动时会加载bean xff0c 如果加载失败 xff0c 则应用就会启动失败 但是部分场景下 xff0c 我们希望某个bean只有满足一定的条件下 xff0c 才允许Spring Boot加载 xff0c 所以
  • 卡尔曼滤波结果和图优化的融合

    仅使用卡尔曼滤波结果作为初始值辅助NDT算法的进行 xff0c 并没有进行实际融合 论文地址 算法流程 扩展的卡尔曼滤波融合GPS 惯性测量单元IMU 编码器里程计得到融合后的定位信息 基于3D NDT配准得到激光里程计 建图优化 融合单帧
  • Kalibr工具进行相机IMU标定

    github地址 xff1a https github com ethz asl kalibrWiki文档 xff1a https github com ethz asl kalibr wiki camera imu calibration
  • 保研复试-计算机网络复习

    计算机网络是我专业课里成绩较高的课 xff0c 老师应该会问到 xff0c 面试的时候计网也是公司里爱问的科目 整理一下看到的网上的一些保研 考研复试题 xff0c 便于复习 一 计算机网络的体系结构 1 计算机网络的分类 按分布范围 xf
  • Hbase的API

    Hbase API 文章目录 Hbase API准备工作Namespace DDL1 Junit执行代码模板2 创建namespace的API3 列出所有的namespace4 列出所有namespace中的表名5 修改namespace的
  • 【Python】 matplotlib 以pdf形式保存图片

    Python matplotlib 以pdf形式保存图片 span class token keyword import span matplotlib span class token punctuation span pyplot as
  • 51单片机 | 点亮第一个LED | LED 闪烁实验 | LED流水灯实验

    文章目录 一 51单片机GPIO介绍1 GPIO概念2 GPIO 结构框图与工作原理2 1 P0端口2 2 P1端口2 3 P2端口2 4 P3端口 2 5 要点 二 LED简介三 硬件设计四 软件设计1 点亮第一个LED2 LED 闪烁实
  • matlab二维矩阵可视化几种方法

    目录 一 pcolor 二 imagesc 三 spy 四 文末彩蛋 一 pcolor 以一个100x100块对角矩阵B为例 1 xff1a 原生pcolor 可以在矩阵维度不大时进行可视化 xff0c 带小方格 矩阵比较大时画出整体为黑色
  • 【ROS入门】双系统安装和ros安装踩坑

    记录一下安装Ubuntu双系统和和ros过程中踩的坑 xff0c 防止下次再犯错 一 双系统安装与删除 双系统的安装比一开始想的说实话要简单得多 xff0c 随便用u盘制作一个系统安装盘 xff0c 或者在u盘里面放需要的Ubuntu镜像
  • 【ROS入门】TF与URDF

    一 什么是TF TF全程就是transform xff0c 就是一个坐标系的转换 在ROS中坐标的转换是一个很重要的内容 xff0c 主要还是因为机器的不灵活性 xff0c 如果是人 xff0c 完全可以灵活地控制手臂去抓取一个物体 xff
  • 树莓派3B+——系统安装及显示

    树莓派3B 43 系统安装及显示 目录 树莓派3B 43 系统安装及显示系统安装显示显示器显示3 5寸显示屏显示pc显示 树莓派3B 43 系统安装及显示 初次接触树莓派 xff0c 花了差不多一天的时间把系统的安装和显示全部搞定 xff0

随机推荐

  • VSLAM 相关知识点总结

    VSLAM 相关知识点 这篇文章是对VSLAM涉及的知识点进行系统性的总结和更新 xff0c 一些内容来源至VSLAM经典教材 xff0c 博客 xff0c 和开源项目 引用材料如下表 SLAM十四讲高博古月老师的技术博客崔神的github
  • Windows下CLion中文乱码最有效的解决方式

    作者开发环境 Windows 10 Clion 18 3 MinGW W64 很多人都遇到了Clion的中文乱码问题 xff0c 然后在CSDN上面寻找解决办法 比如这篇很有代表性https blog csdn net Cbk XLL ar
  • Neutron复盘及学习笔记

    前言 对于openstack neutron xff0c 曾花费很多的时间去看它的源码 xff0c 结果啥都没有看出来 openstack代码风格是 xff0c 为了实现plugin的可插拔 xff0c 运用了很多设计模式 xff0c 设计
  • 匿名飞控底层PID代码解析

    大半年就快过去了 大二上学期转瞬即逝 xff0c 离上一次博客也有一段时间了 xff0c 没错 xff0c 庆幸的是我又回来了 xff0c 其实本来第二篇博客是想着写一下两轮平衡车来着 xff0c 代码早就写好了 xff0c 但是硬件不给力
  • IAR、KEIL常见编译报错警告的解决方法(持续更新中)

    Warning Pe550 variable 变量 was set but never used 原因 xff1a 变量定义赋值了但从未使用 解决 xff1a 变量定义删了就可以 xff0c 编译器也不会真的给这个变量分配空间 Warnin
  • VS2019 未能正确加载”balabal.........Package“

    修复VS2019不能自动补全Unity脚本代码时 xff0c 更新了下面这个东西 导致一直蹦下面这些警告 嘣 嘣 嘣 嘣 嘣 网上有很多博客说重置用户数据 禁用Live Share 分析Log信息等等 xff0c 最后看到一个博文说检查更新
  • 51单片机 | 蜂鸣器实验

    文章目录 一 蜂鸣器介绍二 硬件设计三 软件设计四 实验现象 通过单片机的一个 IO 口控制 板载无源蜂鸣器 xff0c 实现蜂鸣器控制 一 蜂鸣器介绍 蜂鸣器是一种一体化结构的电子讯响器 xff0c 采用直流电压供电 xff0c 广泛应用
  • Vue 中WebSocket的使用

    span class token function data span span class token punctuation span span class token punctuation span span class token
  • npm版本与node版本不对应

    npm与node版本不对应 导致无法npm i 报错提示 WARNING You are likely using a version of node tar or npm that is incompatible with this ve
  • 吴恩达阅读文献步骤

    接下来 xff0c 就集中介绍一下如何研究一篇论文 吴恩达认为 xff0c 要理解一篇论文 xff0c 一次将一篇论文从第一个字读到最后一个字 xff0c 可能并不是最佳方式 正确的打开方式是 xff0c 一篇论文至少要看三遍 第一遍 xf
  • 更新docker

    以下是做为前端开发如何更新docker xff0c 萌新操作 1 xff0c 使用工具Finalshell xff0c 连接到ssh 2 xff0c 连接到服务器 xff0c 选择ssh xff0c 并配置服务名称 主机 服务地址 xff0
  • 计算机视觉理论笔记 (11) - 运动估计 (Motion Estimation)

    运动估计 光流法 Optical Flow 限制Horn xff06 Schunck 全局方法 global method 光流法约束等式 Constrain Equation 小幅度动作问题全局方法和局部方法的比较全局方法局部方法 基于块
  • 嵌入式,面试题目网页搜集

    指针笔试题 嵌入式面试
  • Java第一天笔记02——eclipse的常用快捷键

    1 main方法 main 43 Alt 43 2 System out println syso 43 Alt 43 3 批量修改 Alt 43 Shift 43 R 4 for for 43 Alt 43 5 代码格式化 Ctrl 43
  • FREERTOS源码解析——heap4.c

    目录 内存管理heap4无法解决的内存碎片 xff1a HEAP4简析分配内存在哪 xff0c 大小多少如何管理 重要源码解析 内存管理 freertos目前提供了以下5种内存管理 xff0c 特点如下 heap1 xff1a 最简单的内存
  • vscode + clangd 开发 c\c++

    目录标题 背景LSPClangdquick fixIWYUclang tidy 环境配置插件配置语言服务器下载自定义配置生成compile command jsonmakefile生成compile command json 强烈推荐cla
  • echarts世界地图(含中国省份地图),中国省份下钻到县,点击按钮返回上一级

    效果图 中国地图高亮显示 并且地图中国地图在可视范围的中心 中国地图下钻 省份下钻 小站 www flot top 在线预览地址 https guangshizhao github io world map 需要源码的同学可以自行去本人的仓
  • 51单片机 | LED点阵实验 | 点亮一个点 | 显示数字 | 显示图像

    文章目录 一 LED点阵介绍二 硬件设计三 软件设计1 LED点阵 xff08 点亮一个点 xff09 2 LED点阵 xff08 显示数字 xff09 3 LED点阵 xff08 显示图像 xff09 四 实验现象1 LED 点阵 xff
  • 教你通过bigemap和geojson获取echarts精确到乡镇、街道的地图json数据

    2022 10 09更新 xff01 xff01 xff01 bigemap导出kml文件的功能不开放了 后续有什么好的办法获取乡镇 街道数据再做记录 大家有什么好的方法也可以留言 写在前面 本文重点介绍关于怎么获取地图数据的部分 我们可以
  • openlayers动态添加自定义div图层 具有筛选功能 和浮窗

    openlayers动态加自定义div图层 并且具有筛选功能 和浮窗效果 代码拿过去就可以用 联系方式 45664741 64 qq com 个人博客 flot top 效果图 span class token doctype span c