openLayers 4 canvas图例绘制,canvas循环添加图片,解决图片闪烁问题

一、问题来源:

      接触Openlayers 一段时间了,最近做了一个农业产业系统,项目中涉及到产业图例,最后考虑用canvas来绘制图例图像。当中带图片的图例移动时,图片会实现闪烁留白情况。闪烁是因为绘制图片本身的复杂性,导致canvas绘制频率和浏览器绘制频率不同步,出现图片出不来或者延迟出现,这过程中间就出现了空白显示为canvas底图颜色白色的情况。这里说的闪烁是,在单击地图移动图例时,文字前面的图片并没有出来。但是单击地图准备移动图例时别松开鼠标图片能出来,这个有点奇怪....

    有些解决方案是用一个叫双缓冲的技术实现,这种办法确实能行,这里就不写了,用我们的办法实现。

    双缓冲实现原理:创建一个临时canvas,先把下一帧动画绘制到临时canvas上。在每次真正绘制的时候,擦除正式canvas后,马上drawImage把临时canvas的内容copy过去,而这个copy过程是非常非常高效的,所以基本可以杜绝闪烁。  

 

二 、实现步骤

1、new一个layer层,new一个Feature要素。

2、绘制一个canvas图像。

3、new一个样式,设置样式的Icon图片为上一步绘制好的canvas,并设置Icon宽高为canvas宽高。

4、设置要素Feature的样式为上一步new的样式,将要素Feature添加到layer层上。

5、最后将layer层添加到地图中。

三、加载ol地图,加载天地图矢量图为底图

 1            
      <script type="text/javascript">
var projection = ol.proj.get('EPSG:4326'); 2 var projectionExtent = projection.getExtent(); 3 var size = ol.extent.getWidth(projectionExtent) / 256; 4 var resolutions = new Array(18), 5 matrixIds = new Array(18), 6 shapeArr = []; 7 for(var z = 1; z <= 18; ++z) { 8 resolutions[z] = size / Math.pow(2, z); 9 matrixIds[z] = z; 10 } 11 //天地图矢量图 12 var tdt_road_layer = new ol.layer.Tile({ 13 name: '矢量底图', 14 source: new ol.source.WMTS({ 15 url: 'http://t1.tianditu.com/vec_c/wmts', 16 layer: 'vec', 17 matrixSet: 'c', 18 format: 'tiles', 19 projection: projection, 20 tileGrid: new ol.tilegrid.WMTS({ 21 origin: ol.extent.getTopLeft(projectionExtent), 22 resolutions: resolutions, 23 matrixIds: matrixIds 24 }), 25 style: 'default' 26 }), 27 visible: true, 28 zIndex: 1 29 }); 30 var view = new ol.View({ 31 center: [118.77, 32.05], 32 zoom: 9.5, 33 projection: "EPSG:4326", 34 maxZoom: 18 35 }); 36 var map = new ol.Map({ 37 // 设置地图控件,默认的三个控件都不显示 38 controls: ol.control.defaults({ 39 attribution: true, 40 rotate: true, 41 zoom: true 42 }).extend([ 43 new ol.control.FullScreen(), 44 new ol.control.ScaleLine() 46 ]), 47 layers: [ 48 tdt_road_layer 49 ], 50 target: 'map', 51 loadTilesWhileAnimating: true, 52 view: view 53 });
      </script>

 

四、实现代码,实现文字前面带图片和带色块图例,有背景色的代码为方法调用,调用即可

  1      <script type="text/javascript">
          window.onload = function() { 2 getMapPoint(); 3 drawMapTuliMethod(); 4 } 5 /* 6 * 图例数据 7 */ 8 var dataObj = [{ 9 tname: '国家级文物保护建筑', 10 color: '#365e96', 11 }, { 12 tname: '省级文物保护建筑', 13 color: '#d1702f', 14 }, { 15 tname: '市级级文物保护建筑', 16 color: '#4fa1dc', 17 }, { 18 tname: '区县级文物保护建筑', 19 color: '#368829', 20 }] 21 /* 22 * 图例经纬度坐标,地图绑定了单击事件 23 * 单击返回经纬度并重新绘制canvas 24 */ 25 var removeData = { 26 tx: 118.82368355230953, 27 ty: 32.2359887979324 28 } 29 var canvas = document.createElement('canvas'); 30 31 //绘制图例 32 function drawMapTuliMethod() { 33 var layers = new ol.layer.Vector({ 34 type: 'tuli', 35 source: new ol.source.Vector(), 36 zIndex: 9 37 }) 38 var shape = new ol.Feature({ 39 geometry: new ol.geom.Point([removeData.tx, removeData.ty]) 40 }); 41 42 var ctx = canvas.getContext("2d"); 43 var yheight = 30; 44 yheight += dataObj.length * 27; //计算canvas高度 45 canvas.width = 180; 46 canvas.height = yheight; 47 48 /*设置图例样式*/ 49 ctx.fillStyle = "#fff"; 50 ctx.fillRect(0, 0, 200, yheight); //绘制底图 51 ctx.font = "16px Arial"; 52 ctx.fillStyle = "#000"; 53 ctx.fillText('图例', canvas.width / 2.5, 25); 54 for(var i = 0; i < dataObj.length; i++) { 55 //实现文字前面带色块 56 //ctx.fillStyle = dataObj[i].color; //块颜色 57 //ctx.fillRect(10, 60 + (i - 1) * 25, 15, 15); //颜色块:x,y,w,h 58 59 ctx.font = "12px Arial"; 60 ctx.fillStyle = "#555"; 61 ctx.fillText(dataObj[i].tname, 30, 72 + (i - 1) * 25); //文字 62 63 //添加图片方法一,实现文字前面带图片,移动图例不会出现闪烁 64 drawImg_first('xiushan.png', i); 65 66 //添加图片方法二,移动图例会出现闪烁 67 //drawImg_Second(ctx, 'xiushan.png', i); 68 } 69 //将canvas添加到样式中 70 var style = new ol.style.Style({ 71 image: new ol.style.Icon({ 72 img: canvas, 73 imgSize: [canvas.width, canvas.height], 74 }) 75 }); 76 shape.setStyle(style); 77 layers.getSource().addFeature(shape); 78 map.addLayer(layers); 79 } 80 81 /* 82 * 将绘制完成的图片添加到canvas上 83 * @imgObj:图片对象 84 * @p:循环序号,确定图片坐标 85 */ 86 function drawTuliImage(imgObj, p) { 87 var ctxImge = canvas.getContext("2d"); 88 ctxImge.drawImage(imgObj, 5, 30 + (p * 25), 24, 26); 89 } 90 91 /* 92 * 绘制图例上的图片,方法一 93 * 此方法能解决重绘canvas时图片闪烁留白的问题 94 * @imgs:图片名称 95 * @p:序号 96 * @complete:HTMLImageElement对象的一个属性,可以判断图片加载完成 97 */ 98 function drawImg_first(imgs, p) { 99 var imgObj = new Image(); 100 imgObj.src = 'img/' + imgs; 101 //如果图片加载完成 102 if(imgObj.complete) { 103 drawTuliImage(imgObj, p); 104 } else { 105 //onload:重绘,重新加载 106 imgObj.onload = function() { 107 drawTuliImage(imgObj, p); 108 }; 109 //加载失败 110 imgObj.onerror = function() { 111 console.log('canvas图片加载失败,请重试!') 112 }; 113 } 114 } 115 116 /* 117 * 添加数据前面的图片,方法二 118 * 此方法绘制图片会出现闪烁留白情况, 119 * @ctx:绘图环境 120 * @imgs:图片名称 121 * @p:循环序号 122 */ 123 function drawImg_Second(ctx, imgs, p) { 124 var imgObj = new Image(); 125 imgObj.src = 'img/' + imgs; 126 imgObj.onload = function() { 127 ctx.drawImage(imgObj, 5, 30 + (p * 25), 25, 27); 128 } 129 } 130 131 /* 132 * 添加图例之前删除原来 133 * 引用类型。length会变化,for循环倒着删除 134 * @deType:要删除的覆盖物名称 135 */ 136 function addNewsChartsDelectOring(deType) { 137 var layersArr = map.getLayers().getArray(); //获取所有覆盖物 138 //移除全部 139 if(deType == 'all') { 140 for(var i = layersArr.length - 1; i >= 0; i--) { 141 var ltype = layersArr[i].get('type'); 142 if(ltype == 'tuli') map.removeLayer(layersArr[i]); 143 } 144 return; 145 } 146 //移除具体 147 else { 148 for(var i = layersArr.length - 1; i >= 0; i--) { 149 var ltype = layersArr[i].get('type'); 150 if(ltype == deType) map.removeLayer(layersArr[i]); 151 } 152 return; 153 } 154 } 155 156 //地图单击事件 157 function getMapPoint() { 158 map.on('click', function(evt) { 159 var point = evt.coordinate; //鼠标单击点坐标 160 removeData.tx = point[0]; 161 removeData.ty = point[1]; 162 addNewsChartsDelectOring('all'); 163 drawMapTuliMethod(); 164 }); 165 }
</script>

 

①文字前面带图片的图例,移动图例时canvas绘制图片频率和浏览器绘制频率不一导致图片不出来的效果图:

 

②文字前面带图片的效果图:

 

③文字前面带色块的效果图:

 

 

 

技术群 : 192713488

转载于:https://www.cnblogs.com/webonline/p/8337236.html

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

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

相关文章

Java DB嵌入式模式

Java DB是基于Java编程语言和SQL的关系数据库管理系统。 这是Apache软件基金会的开源Derby项目的Oracle版本。 Java SE 7 SDK中包含Java DB。 Java DB有两个部署选项&#xff1a; Embedded和Network Server 。 这篇文章是关于嵌入式部署或模式的。 1.嵌入式 在嵌入式模式下&…

清除浮动方法解析

清除浮动方法解析 清除浮动带来的额外影响 如果对于浮动不熟悉的同学&#xff0c;可以看看介绍float的文章。传送门&#xff1a;CSS float 我们知道&#xff0c;在一个父元素内如果遇到某个浮动元素&#xff0c;此时父元素的高度会发生塌陷。针对父元素高度塌陷的问题&#xff…

进程间的通信----管道

前提&#xff1a;本文是基于Linux系统下的学习 用户态的进程是如何组织的呢&#xff1f;所有的用户态进构成了一棵树。进程树。 进程树的树根是init.也就是1号进程。是用户态进程的祖宗进程。如何查看进程树&#xff1f;pstree 进程之间的关系 父子进程和兄弟进程查看进程的信息…

web项目启动时,自动执行代码的几种方式

在项目开发过程中&#xff0c;往往需要一些功能随着项目启动而优先启动&#xff0c;下面我总结几种方式&#xff08;非spring boot&#xff09; spring boot的参考 spring boot 学习之路9 (项目启动后就执行特定方法) 方式一&#xff1a; ServletContextListener监听器&#…

设计模式:状态

本文将介绍状态设计模式 。 它是行为设计模式之一 。 您无需了解许多理论即可了解模式的主要概念。 该文章将分为几个部分&#xff0c;在其中我将提供有关需要应用该模式的情况&#xff0c;它所具有的利弊以及用法示例的信息。 有时&#xff0c;当对象的内部状态更改时&#…

[UWP]了解模板化控件(9):UI指南

[UWP]了解模板化控件(9)&#xff1a;UI指南 原文:[UWP]了解模板化控件(9)&#xff1a;UI指南1. 使用TemplateSettings统一外观 TemplateSettings提供一组只读属性&#xff0c;用于在新建ControlTemplate时使用这些约定的属性。 譬如&#xff0c;修改HeaderedContentControl的Co…

Java的反射API

如果您曾经问​​过自己以下问题&#xff1a; –“如何在字符串中仅包含其名称的方法调用&#xff1f;” –“如何动态列出类中的所有属性&#xff1f;” –“如何编写一种将任何给定对象的状态重置为默认值的方法&#xff1f;” 然后您可能已经听说过Java的Reflection API…

linux服务器基本常识,服务器搭建-Linux基础知识

服务器搭建还是需要一些Linux知识的&#xff0c;这节就聊点基础的。文件权限操作查看权限Linux中每个文件对每个用户来说都有对应的权限&#xff0c;在任一路径中输入ll就可以查看这些信息&#xff1a;rootip-*** /usr/local # lltotal 32Kdrwxr-xr-x 2 root root 4.0K Jan 14 …

CSS实现单行与多行文字省略(truncation)

在上一篇文章小div布局之卡片堆叠&#xff08;card-stacking&#xff09;中有多行文字溢出省略的效果&#xff0c;这篇文章就对这种效果&#xff08;包括单行文字溢出省略&#xff09;的实现做个简单的记录&#xff0c;以防自己忘记。具体来说&#xff0c;就是要实现这种文字排…

logback配置详解和使用

最近知道一种打印日志的新方法&#xff0c;在此做一下学习总结。 转自&#xff1a;行走在云端的愚公 https://www.cnblogs.com/warking/p/5710303.html 一、logback的介绍 Logback是由log4j创始人设计的另一个开源日志组件,官方网站&#xff1a; http://logback.qos.ch。它当前…

RAID技术超详细讲解

RAID 技术是一种多磁盘技术&#xff0c;面对数据的各方面有着两面性的影响&#xff0c;整体来说优点大于缺点的&#xff0c;下面我将详细介绍一下 RAID &#xff0c;简称磁盘阵列技术。 一、RAID 概述 1988 年美国加州大学伯克利分校的 D. A. Patterson 教授等首次在论文 “A C…

Java安全教程–创建SSL连接和证书的分步指南

在有关应用JEE安全性的系列文章中&#xff0c;我们为您提供了另一个有关如何在Java EE应用程序中创建SSL连接和创建证书的详细教程。 如我们之前的文章中所述&#xff0c; 安全套接字层&#xff08;SSL&#xff09;/传输层安全性&#xff08;TLS&#xff09;将启用客户端和Web服…

[转]CSS hack大全详解

转自&#xff1a;CSS hack大全&详解 1、什么是CSS hack? CSS hack是通过在CSS样式中加入一些特殊的符号&#xff0c;让不同的浏览器识别不同的符号&#xff08;什么样的浏览器识别什么样的符号是有标准的&#xff0c;CSS hack就是让你记住这个标准&#xff09;&#xff0c…

产品经理应该具备的技能(2)如何做一个好的数据产品经理?

一、如何做一个好的数据产品经理&#xff1f; PD&#xff08;指产品经理&#xff0c;下同&#xff09;本身就是在做牛做马&#xff0c;关系圈异常复杂。数据PD也不例外。而且打交道的人更多。以下是我用PPT绘制的数据产品经理关系圈。如果你也做过 数据产品的产品经理&#xff…

Java EE CDI程序化依赖关系消歧示例–注入点检查

在本教程中&#xff0c;我们将看到在注入Java EE CDI bean时如何避免程序依赖消除歧义。 我们已经在Jave EE依赖关系消除歧义示例中展示了如何避免CDI Bean中的依赖关系歧义消除。 在这里&#xff0c;我们将向您展示如何以动态方式避免依赖消除歧义。 我们将通过检查注入另一个…

机器学习算法整理(四)集成算法—随机森林模型

随机&#xff1a;数据采样随机&#xff0c;特征选择随机 &#xff08;数据采样&#xff0c;有放回&#xff09; 转载于:https://www.cnblogs.com/douzujun/p/8386930.html

linux人脸识别视频推流,RTMP推流协议视频智能分析/人脸识别/直播点播平台EasyDSS接口调用注意事项介绍...

TSINGSEE青犀视频目前推出了前端支持不同协议设备接入的视频智能分析平台&#xff0c;包括RTSP协议的EasyNVR、GB28181协议的EasyGBS&#xff0c;RTMP推流协议的EasyDSS&#xff0c;还有能够进行人脸识别、车牌识别的EasyCVR&#xff0c;这些平台均提供了视频转码分发的能力&am…

在基于图论的Java程序中基于DSL的输入图数据的方法

我们大多数人已经编写了一些程序来处理图论算法&#xff0c;例如查找两个顶点之间的最短路径&#xff0c;查找给定图的最小生成树等等。 在这些算法的每一种中&#xff0c;表示图形的编程方式是使用邻接矩阵或邻接表 。 两者都不是定义图形输入的非常直观的方法。 例如&#xf…

Remmarguts' Date(POJ2449+最短路+A*算法)

题目链接&#xff1a;http://poj.org/problem?id2449 题目&#xff1a; 题意&#xff1a;求有向图两点间的k短路。 思路&#xff1a;最短路A*算法 代码实现如下&#xff1a; 1 #include <set>2 #include <map>3 #include <queue>4 #include <stack>5 …

usb3.0 linux无法识别,USB3.0接口不能识别U盘的解决方法

USB接口可以说是电脑的标配&#xff0c;现在基本上所有电脑都会搭载USB接口。而USB标准从1.0发展到现在的3.0&#xff0c;甚至更新的也已出来。不过&#xff0c;如果USB3.0无法识别U盘&#xff0c;那该怎么办呢?USB3.0是一种技术也是一种规范&#xff0c;现在很多笔记本都是默…