Openlayers3中实现地图的切割
概述:
本文讲述如何在Openlayers3中结合canvas实现对地图的切割。
效果:
全图
切割北京区域
切割河北区域
实现:
<html xmlns="http://www.w3.org/1999/xhtml"> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <title>Ol3 wms</title> <link rel="stylesheet" type="text/css" href="../../../plugin/ol3/css/ol.css"/> <style type="text/css"> body, #map { border: 0px; margin: 0px; padding: 0px; width: 100%; height: 100%; font-size: 13px; } </style> <script type="text/javascript" src="../../../plugin/ol3/build/ol-debug.js"></script> <script type="text/javascript" src="../../../plugin/jquery/jquery-1.8.3.js"></script> <script type="text/javascript"> function init(){ var bounds = [73.4510046356223, 18.1632471876417, 134.976797646506, 53.5319431522236]; var vec_c = getTdtLayer("vec_c"); var cva_c = getTdtLayer("cva_c"); var map = new ol.Map({ controls: ol.control.defaults({ attribution: false }), target: 'map', layers: [vec_c, cva_c], view: new ol.View({ projection: 'EPSG:4326' }) }); map.getView().fit(bounds, map.getSize()); function getTdtLayer(lyr){ var url = "http://t0.tianditu.com/DataServer?T="+lyr+"&X={x}&Y={y}&L={z}"; var projection = ol.proj.get("EPSG:4326"); var projectionExtent = [ -180, -90, 180, 90 ]; var maxResolution = (ol.extent.getWidth(projectionExtent) / (256 * 2)); var resolutions = new Array(16); var z; for (z = 0; z < 16; ++z) { resolutions[z] = maxResolution / Math.pow(2, z); } var tileOrigin = ol.extent.getTopLeft(projectionExtent); var layer = new ol.layer.Tile({ extent: [ -180, -90, 180, 90 ], source: new ol.source.TileImage({ tileUrlFunction: function(tileCoord) { var z = tileCoord[0]+1; var x = tileCoord[1]; var y = -tileCoord[2]-1; var n = Math.pow(2, z + 1); x = x % n; if (x * n < 0) { x = x + n; } return url.replace('{z}', z.toString()) .replace('{y}', y.toString()) .replace('{x}', x.toString()); }, projection: projection, tileGrid: new ol.tilegrid.TileGrid({ origin: tileOrigin, resolutions: resolutions, tileSize: 256 }) }) }); return layer; } $.get("data/province.geojson",null,function(result){ var image = new ol.style.Circle({ radius: 5, fill: null, stroke: new ol.style.Stroke({color: 'red', width: 1}) }); var styles = new ol.style.Style({ stroke: new ol.style.Stroke({ color: 'blue', width: 2 }), fill: new ol.style.Fill({ color: 'rgba(0, 0, 255, 0)' }) }); var vectorSource = new ol.source.Vector({ features: (new ol.format.GeoJSON()).readFeatures(result) }); var vector = new ol.layer.Vector({ source: vectorSource, style: styles }); map.addLayer(vector); var _styles = new ol.style.Style({ stroke: new ol.style.Stroke({ color: '#00ffff', lineDash: [4], width: 3 }), fill: new ol.style.Fill({ color: 'rgba(0, 0, 255, 0.1)' }) }); var select = new ol.interaction.Select({ condition: ol.events.condition.click, style: _styles }); map.addInteraction(select); var highlight; select.on('select', function (e) { var feature = e.target.getFeatures().item(0); if (feature) { highlight = feature; var _coord = feature.getGeometry().getCoordinates(); console.log(_coord); map.render(); map.on('precompose',clip) } }); var center,pixelScale,offsetX,offsetY,rotation; function clip(evt) { var canvas=evt.context; canvas.save(); var coords=highlight.getGeometry().getCoordinates(); var frameState = evt.frameState; var pixelRatio = frameState.pixelRatio; var viewState = frameState.viewState; center = viewState.center; var resolution = viewState.resolution; rotation = viewState.rotation; var size = frameState.size; var size1=map.getSize(); offsetX = Math.round(pixelRatio * size[0] / 2); offsetY = Math.round(pixelRatio * size[1] / 2); pixelScale = pixelRatio / resolution; canvas.beginPath(); if(highlight.getGeometry().getType() == 'MultiPolygon'){ for(var i=0;i<coords.length;i++){ createClip(coords[i][0], canvas); } } else if(highlight.getGeometry().getType() == 'Polygon'){ createClip(coords[0], canvas); } canvas.clip(); } function createClip(coords, canvas) { for (var i = 0, cout = coords.length; i < cout; i++) { var xLen = Math.round((coords[i][0] - center[0]) * pixelScale); var yLen = Math.round((center[1] - coords[i][1]) * pixelScale); var x = offsetX; var y = offsetY; if (rotation) { x = xLen * Math.cos(rotation) - yLen * Math.sin(rotation) + offsetX; y = xLen * Math.sin(rotation) + yLen * Math.cos(rotation) + offsetY; } else { x = xLen + offsetX; y = yLen + offsetY; } if (i == 0) { canvas.moveTo(x, y); } else { canvas.lineTo(x, y); } } canvas.closePath(); } map.on('postcompose', function(event) { var ctx = event.context; ctx.restore(); }); }); } </script> </head> <body onLoad="init()"> <div id="map"> </div> </body> </html>
--------------------------------------------------------------------------------------------------------------- 技术博客 CSDN:http://blog.csdn.NET/gisshixisheng 博客园:http://www.cnblogs.com/lzugis/ 在线教程 http://edu.csdn.Net/course/detail/799 Github https://github.com/lzugis/ 联系方式 q q:1004740957 e-mail:niujp08@qq.com 公众号:lzugis15 Q Q 群:452117357(webgis) 337469080(Android)
Openlayers3中实现地图的切割的更多相关文章
- iOS中的地图和定位
文章摘自http://www.cnblogs.com/kenshincui/p/4125570.html#location 如有侵权,请联系删除. 概览 现在很多社交.电商.团购应用都引入了地图和定 ...
- WP8中的地图和导航
原文 WP8中的地图和导航 代码示例 源文件: Media:MapSample.zip 测试基于 SDK: Windows Phone SDK 8.0 兼容于 平台: Windows Phone 8 ...
- EChart中使用地图方式总结(转载)
EChart中使用地图方式总结 2018年02月06日 22:18:57 来源:https://blog.csdn.net/shaxiaozilove/article/details/79274772 ...
- 在Microsoft Power BI中创建地图的10种方法
今天,我们来简单聊一聊“地图”. 在我们日常生活中,地图地位已经提升的越来越高,出门聚餐.驾驶.坐车.旅行......应运而生的就是各种Map APP. 作为数据分析师,我们今天不讲生活地图,要跟大家 ...
- PHP 中使用explode()函数切割字符串为数组
explode()函数的作用:使用一个字符串分割另一个字符串,打散为数组. 例如: 字符串 $pizza = "第1 第2 第3 第4 第5 第6"; 根据空格分割后:$piece ...
- openlayers3中应用proj4js
要在openlayers3中应用proj4js,需要在html中引用proj4js,然后在引用所需要的projection的js定义,如 http://epsg.io/21781-1753.js 然后 ...
- Vue3.0中引入地图(谷歌+高德+腾讯+百度)
1 概述 项目需求需要引入地图,对于目前最新的Vue3.0,无论是百度/高德/腾讯地图目前还没有适配,只有Vue 2.x版本的: 目前只有谷歌地图的Vue3.0适配: 但是没有适配并不代表不能使用,本 ...
- EXCEL技能 | EXCEL中实现地图快照,截大图、加水印、保存PNG、TIF、HTML文件
1 应用场景 本文分享笔者使用EXCEL制作地图的体验. 之前网上有人介绍使用小O地图EXCEL插件版能够在EXCEL中标注地图.绘制地图.可视化数据等操作.如下截图.笔者通过实验,其软件保存方式只能 ...
- 在网页中嵌入地图API
1.登录百度地图api,地址:http://api.map.baidu.com/lbsapi/creatmap/index.html 2.设置你的地理位置 3.简单设置下地图功能.地图的宽度和高度根据 ...
随机推荐
- [BZOJ4010]菜肴制作
Description 知名美食家小 A被邀请至ATM 大酒店,为其品评菜肴. ATM 酒店为小 A 准备了 N 道菜肴,酒店按照为菜肴预估的质量从高到低给予 1到N的顺序编号,预估质量最高的菜肴编号 ...
- LeetCode——Palindromic Substrings
Question Given a string, your task is to count how many palindromic substrings in this string. The s ...
- Lucene 的 Scoring 评分机制
转自: http://www.oschina.net/question/5189_7707 Lucene 评分体系/机制(lucene scoring)是 Lucene 出名的一核心部分.它对用户来 ...
- harbor 管理Helm Chart包
官方网站:https://github.com/goharbor/harbor官方用户手册:https://github.com/goharbor/harbor/blob/master/docs/us ...
- 安装mysql数据库中的技巧、错误排查
针对解压版本5.7.16(博主使用的这个版本.在某些低版本中部分命令失效) 一.安装.初始化data目录(解压版解压后没有data目录) 安装:配置path环境变量,然后管理员运行命令提示符cmd ...
- centos7安装ambari教程
ambari版本 :2.4.2 (不过各版本安装过程没啥差异) 目录: 为什么要用Ambari 概念概述 版本信息 原理简介 安装 创建集群 创建集群 手动修改配置 NameNode HA 安装Sma ...
- Centos 查看端口占用情况
netstat -ntlp 把相应PID kill掉即可
- Kafka消息文件存储
在对消息进行存储和缓存时,Kafka依赖于文件系统.(Page Cache) 线性读取和写入是所有使用模式中最具可预计性的一种方式,因而操作系统采用预读(read-ahead)和后写(write-be ...
- Selenium入门练习(二)
自动登录博客园并且退出登录 package TestNG; import org.testng.annotations.Test;import org.testng.annotations.Befor ...
- 20.并发容器之ArrayBlockingQueue和LinkedBlockingQueue实现原理详解
1. ArrayBlockingQueue简介 在多线程编程过程中,为了业务解耦和架构设计,经常会使用并发容器用于存储多线程间的共享数据,这样不仅可以保证线程安全,还可以简化各个线程操作.例如在“生产 ...