如何在vue中使用百度地图添加自定义覆盖物(水波纹)


Posted in Javascript onNovember 03, 2020

简介

一如既往,我来给大家分享一个项目中遇到的比较有意思的需求并介绍一下相应的实现过程。

话不多说,直接上图:

如何在vue中使用百度地图添加自定义覆盖物(水波纹)

具体的应用场景简而言之就是需要我们在地图上添加如图中所示的自定义覆盖物。实现的过程作者分为以下两点给大家介绍介绍。

  • 水波?的实现
  • 自定义覆盖物的实现

水波?的实现

这个需求的实现肯定是离不开我们自己写自定义覆盖物的,那么首先我们来讨论一下水波纹动画的实现。

首先我们可以看到图中的覆盖物是由一个红心和水波?组成,其中红心是固定不动的,那么我们可以直接这么写:

<div class="radar"></div>
.radar {
  width: 40px;
  height: 40px;
  border-radius: 50%;
  background-color: red;
 }

如何在vue中使用百度地图添加自定义覆盖物(水波纹)

这样子我们首先就实现了红心部分的样式。那么水波?又是怎么实现的呢?

我们可以从图中观察得出先后总共有三个水波?从里到外逐渐的往外扩散。我们单独从一个水波?来看的话,其实往外扩散的原理是通过动画让水波?的宽高逐渐递增到一定程度即可,具体扩散多大呢读者可以根据自己的需求设定水波?的最后宽高。

水波?的基本结构和样式实现如下:

<div class="radar">
 <div class="ripple"></div>
 <div class="ripple"></div>
 <div class="ripple"></div>
 </div>
.radar {
 width: 40px;
 height: 40px;
 border-radius: 50%;
 background-color: red;
 position: relative;
 .ripple {
  width: 40px;
  height: 40px;
  border-radius: 50%;
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%,-50%);
  border: 1px solid red;
  animation: ripple 2s linear infinite;
 }
 }

之所以将水波?的dom节点嵌套在radar节点里面主要是做一个“子绝父相”定位以达到水波?居中对齐的效果,这个应该很容易理解。至于水波?的初始宽高呢,为了方便动画延迟时间的计算,我就设置为跟红心宽高相等即可,如果读者有兴趣可以尝试一下宽高设置在0~40px任意值的看看效果(作者盲猜动画可能会有点瑕疵,可能也不会------>“手动狗头”))。

至于动画的实现呢,这个就很简单了,我们只需要同时将水波?的宽高都逐渐变大即可,但是有个点要注意一下就是水波?最后是会消失的,这里我们利用opacity:0配合动画达到逐渐消失的效果。

@keyframes ripple {
 to {
  width: 150px;
  height: 150px;
  opacity: 0;
 }
 }

代码到了这里,我们水波?已经是实现了,但是不是少了点什么?说好的三个水波?呢?

这个问题也很好解决,我们只需要给水波?设置对应的动画延迟即可。

.radar :nth-child(1) {
 animation-delay: 0.666s;
 }
 .radar :nth-child(2) {
 animation-delay: 1.322s;
 }

这样子我们就把水波?给实现出来啦。

如何在vue中使用百度地图添加自定义覆盖物(水波纹)

自定义覆盖物的实现

本次作者采用的是百度地图实现的需求,那么如何通过百度地图来实现自定义覆盖物呢?

其实百度地图开发文档中也很直白的向我们展示自定义覆盖物的开箱方法。这里作者用es6的class稍微改写了一下写法,道理都一样的,官方案例可以参考百度地图demo

自定义覆盖物实现分以下三步:

  • 定义构造函数并继承Overlay
  • 初始化自定义覆盖物
  • 绘制覆盖物

定义构造函数并继承Overlay

按照官网的说法是:“首先您需要定义自定义覆盖物的构造函数,通过构造函数参数可以传递一些自由的变量。设置自定义覆盖物对象的prototype属性为Overlay的实例,以便继承覆盖物基类”。

运用es6 class我们可以很快的实现:

其中point传递的是坐标位置,用于后续计算覆盖物在地图上的位置。

class RadarOverlay extends BMap.Overlay	{
 constructor(point) {
  super();
  this.point = point;
 }
}

初始化自定义覆盖物

官方的说法是:“实现initialize方法,当调用map.addOverlay方法时,API会调用此方法。当调用map.addOverlay方法添加自定义覆盖物时,API会调用该对象的initialize方法用来初始化覆盖物,在初始化过程中需要创建覆盖物所需要的DOM元素,并添加到地图相应的容器中。这里我们选择添加在容器markerPane上。”

其实意思大概就是说,因为百度地图在添加overLay时会调用自定义覆盖物构造函数中的initialize方法,这个initialize方法是用来初始化覆盖物的,他会在初始化过程中创建所需要的DOM,因此我们要在我们自己的构造函数(这里我们用的是类)中实现一个initialize。

百度地图官网的方法:

如何在vue中使用百度地图添加自定义覆盖物(水波纹)

那么我们就按照官网的要求,在类中定义一个initialize方法。其中template存的是我们初始化中所需要的DOM元素,这里的dom节点会比上文中实现水波纹的环节多了一个className为rader-box的父节点,因为在官网中我们也可以看到官方实例在创建DOM元素时给DOM节点添加了position:absolute的样式,但由于我们的radar节点的position为relative,所以我们需要在最外层多嵌套一层dom节点以确保整个dom相对于地图定位。当然啦该父节点的position也就为绝对定位

initialize(map) {
  this._map = map;
  const template = `<div class="radar-box">
   <div class="radar">
    <div class="ripple"></div>
    <div class="ripple"></div>
    <div class="ripple"></div>
   </div>
  </div>`;
  // 创建文档碎片
  const divFragment = document.createRange().createContextualFragment(template);
  const div = divFragment.querySelectorAll('.radar-box')[0];
  // 将div添加到覆盖物容器中
  map.getPanes().markerPane.appendChild(div);
  this._div = div;
  return div;
 }

这样子我们就成功的在类中定义了一个initialize方法,至于为什么要将div添加到覆盖物容器中官网的也有如下说明:

大概就是说百度地图里面已经给我们提供了若干的覆盖物展示方法,我们自定义的覆盖物其实是保存在我们选择的其中某个覆盖物容器之下,有兴趣的小伙伴可以去了解一下各种覆盖物容器。。。

绘制覆盖物

这里作者继续抛砖引玉阿,官方的说法是:“到目前为止,我们仅仅把覆盖物添加到了地图上,但是并没有将它放置在正确的位置上。您需要在draw方法中设置覆盖物的位置,每当地图状态发生变化(比如:位置移动、级别变化)时,API都会调用覆盖物的draw方法,用于重新计算覆盖物的位置。通过map.pointToOverlayPixel方法可以将地理坐标转换到覆盖物的所需要的像素坐标。”

简单来讲就是比如地图缩放比例之后,会重新调用覆盖物中的draw方法,用于重新鸡算覆盖物的位置。

那么问题来了,draw方法哪里来的?

所以官方的意思是我们在自定义覆盖物中的构造函数定义一个draw方法,方法中我们利用百度地图的map.pointToOverlayPixel方法(用来将地理坐标转换为像素坐标)可以将地理坐标转换到覆盖物的所需要的像素坐标。
那么我们只需要在类中加入draw方法即可:

其中this.point就是我们构造函数中的point坐标点,40表示的是我们整个水波纹的初始宽高,读者其实可以将40改为可以传参的形式比如this.size,这样会方便后续代码的维护,这里为了直观的让读者理解就不讲太复杂了。

draw() {
  // 根据地理坐标转换为像素坐标,并设置给容器
  const position = this._map.pointToOverlayPixel(this.point);
  this._div.style.left = `${position.x - 40 / 2}px`;
  this._div.style.top = `${position.y - 40 / 2}px`;
 }

到此,我们整个自定义覆盖物的类就已经实现了,整体代码图下(这里把40改为this.size代码维护起来稍微方便点):

class RadarOverlay extends BMap.Overlay {
 constructor(point, size) {
  super();
  this.point = point;
  this.size = size;
 }

 initialize(map) {
  this._map = map;
  const template = `<div class="radar-box">
    <div class="radar">
     <div class="ripple"></div>
     <div class="ripple"></div>
     <div class="ripple"></div>
    </div>
   </div>`;
  const divFragment = document.createRange().createContextualFragment(template);
  const div = divFragment.querySelectorAll('.radar-box')[0];
  map.getPanes().markerPane.appendChild(div);
  this._div = div;
  return div;
 }

 draw() {
  // 根据地理坐标转换为像素坐标,并设置给容器
  const position = this._map.pointToOverlayPixel(this.point);
  this._div.style.left = `${position.x - this.size / 2}px`;
  this._div.style.top = `${position.y - this.size / 2}px`;
 }
}

自定义覆盖物的使用

那既然我们都已经实现了自定义的覆盖物,第一件事当然就是把它给使用上,具体如下:

initMap() {
   const map = new BMap.Map(this.$refs.map);
   map.centerAndZoom(new BMap.Point(116.404, 39.915), 15);
   // 实例化我们要绘制自定义覆盖物的点坐标
   const point = new BMap.Point(116.404, 39.915);   
	// 实例化自定义覆盖物,将坐标点和水波纹的尺寸传进构造函数中
   const radar = new RadarOverlay(point, 40);  
   // 添加自定义覆盖物
   map.addOverlay(radar);
  }

效果如图:

如何在vue中使用百度地图添加自定义覆盖物(水波纹)

结语

至于构造自定义覆盖物的其他内容,大佬们如果要深入的话可以参考百度地图的文档,里面讲的肯定要比作者更加详细(狗头)。如果文章的内容有问题话的欢迎留言点出,希望能与各路大佬们交流交流,喜欢作者的文章的话也别忘了给我点个赞,这对我来说很重要。

源码地址:https://gitee.com/zhao_jiahao/baidu-map-custom-cover

到此这篇关于如何在vue中使用百度地图添加自定义覆盖物(水波纹)的文章就介绍到这了,更多相关vue用百度地图添加自定义覆盖物内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
JS文本框不能输入空格验证方法
Mar 19 Javascript
引用 js在IE与FF之间的区别详细解析
Nov 20 Javascript
javascript中解析四则运算表达式的算法和示例
Aug 11 Javascript
用jquery修复在iframe下的页面锚点失效问题
Aug 22 Javascript
推荐一个自己用的封装好的javascript插件
Jan 29 Javascript
JS实现新浪微博效果带遮罩层的弹出框代码
Oct 12 Javascript
Easyui在treegrid添加控件的实现方法
Jun 23 Javascript
React从react-router路由上做登陆验证控制的方法
May 10 Javascript
解决使用bootstrap的dropdown部件时报错:error:Bootstrap dropdown require Popper.js问题
Aug 30 Javascript
解决JavaScript中0.1+0.2不等于0.3问题
Oct 23 Javascript
微信小程序request请求封装,验签代码实例
Dec 04 Javascript
JavaScript中使用Spread运算符的八种方法总结
Jun 18 Javascript
JavaScript语法约定和程序调试原理解析
Nov 03 #Javascript
解决vux 中popup 组件Mask 遮罩在最上层的问题
Nov 03 #Javascript
React Native登录之指纹登录篇的示例代码
Nov 03 #Javascript
解决VantUI popup 弹窗不弹出或无蒙层的问题
Nov 03 #Javascript
wepy--用vantUI 实现上弹列表并选择相应的值操作
Nov 03 #Javascript
使用vant的地域控件追加全部选项
Nov 03 #Javascript
vue vant中picker组件的使用
Nov 03 #Javascript
You might like
php学习之 数组声明
2011/06/09 PHP
php验证码实现代码(3种)
2015/09/07 PHP
PHP实现非阻塞模式的方法分析
2018/07/26 PHP
Laravel推荐使用的十个辅助函数
2019/05/10 PHP
表单项的name命名为submit、reset引起的问题
2007/12/22 Javascript
javascript深入理解js闭包
2010/07/03 Javascript
jquery中focus()函数实现当对象获得焦点后自动把光标移到内容最后
2013/09/29 Javascript
js完美实现@提到好友特效(兼容各大浏览器)
2015/03/16 Javascript
java中String类型变量的赋值问题介绍
2016/03/23 Javascript
js实现带农历和八字等信息的日历特效
2016/05/16 Javascript
Knockout结合Bootstrap创建动态UI实现产品列表管理
2016/09/14 Javascript
原生js实现验证码功能
2017/03/16 Javascript
详解AngularJS 路由 resolve用法
2017/04/24 Javascript
使用Browserify来实现CommonJS的浏览器加载方法
2017/05/14 Javascript
解决Jstree 选中父节点时被禁用的子节点也会选中的问题
2017/12/27 Javascript
Vue 源码分析之 Observer实现过程
2018/03/29 Javascript
webpack+react+antd脚手架优化的方法
2018/04/02 Javascript
详解react阻止无效重渲染的多种方式
2018/12/11 Javascript
koa+jwt实现token验证与刷新功能
2019/05/30 Javascript
Electron 调用命令行(cmd)
2019/09/23 Javascript
Python实现基本线性数据结构
2016/08/22 Python
Django Form 实时从数据库中获取数据的操作方法
2019/07/25 Python
Python reversed函数及使用方法解析
2020/03/17 Python
PyInstaller的安装和使用的详细步骤
2020/06/02 Python
Python lambda表达式原理及用法解析
2020/08/18 Python
一款利用纯css3实现的360度翻转按钮的实例教程
2014/11/05 HTML / CSS
应届本科生推荐信范文
2013/12/25 职场文书
高中生期末评语大全
2014/01/28 职场文书
医德医风演讲稿
2014/05/20 职场文书
关于清明节的演讲稿
2014/09/13 职场文书
检讨书1000字
2014/10/11 职场文书
在职证明范本
2015/06/15 职场文书
上级领导检查欢迎词
2015/09/30 职场文书
如何使用JavaScript策略模式校验表单
2021/04/29 Javascript
如何用JavaScript实现一个数组惰性求值库
2021/05/05 Javascript
Mybatis 一级缓存和二级缓存原理区别
2022/09/23 Java/Android