快速使用 ¶
Getting Started
TIP
在这里假设你已经了解你所使用的前端 WebGIS
类库,例如开源产品 maptalks
、 mapbox-gl
、openlayers
、 leaflet
等,以及国内的百度地图和高德地图。
介绍 ¶
wind-layer 是一个专注于气象格点数据可视的插件,设计之处是参考了 earth cambecc 的一个气象数据的展示,他使用了流体场的方式去展示了全球的风速和风向,富有很强的 表现力, 这个插件的早期的很多核心代码也是来源于此。当然现在它不仅仅是做风场的展示,常规的气象数据都可以依赖此插件进行可视化。
特性 (1.x 通用版本相对于原始 windy.js) ¶
- 易于配置粒子数量,原始 windy.js 只能给定一个系数,会根据地图元素的大小进行计算粒子数量;现在可以支持系数方式和固定粒子数量以及回调函数的的三种方式。
- 颜色配置支持三种方式:
String
:固定颜色值Function
: 通过回调函数的风速值设定颜色(但是会有一定的性能损失)String[]
: 按照风速值范围等间隔渲染,无法做到精确匹配对应值的颜色。
- 线条宽度支持动态设置。
- 抽离了核心渲染库,便于扩展到其他地图渲染库。
2.x webgl 版本 ¶
gl-core
扩展目前只针对 mapbox
、maplibre
和 maptalks
做了相关适配,其他引擎适配欢迎 PR。
目前计划支持的图层如下:
- [x] 常规 raster 图层,为了解决多时次瓦片序列播放问题。
- [x] 常规 Image 图层,单张 raster 图层。
- [x] 色斑图渲染,支持瓦片和单张。
- [x] 多数据源支持(geotiff、灰度图-可解析带 exif 信息、png-多通道浮点数压缩)。
- geotiff 的支持需要配置
configDeps
,exif
默认支持,因为在safari
浏览器下configDeps
有兼容性问题,所以默认打包了 exif 的解析库。
tsmapboxWind.configDeps(['https://unpkg.com/geotiff/dist-browser/geotiff.js']);
- geotiff 的支持需要配置
- [x] TimelineSource(时序数据源)支持。
- [x] 粒子渲染,支持瓦片和单张。
- [x] 箭头图层,支持瓦片和单张(矢量数据:风或洋流)。
- [x] 图层拾取。
- [x] 图层掩膜。
TIP: 注意 2.0 之后不再支持 jsonArray
数据和 EPSG:4326
的图片数据,如果有需求请使用 1.x 版本
https://github.com/sakitam-fdd/wind-layer/assets/19517451/b36b7eea-c647-42ed-91a4-e1f182d0343c
https://github.com/sakitam-fdd/wind-layer/assets/19517451/bf27d98e-68ed-4f9c-b1e4-812764665bff
https://github.com/sakitam-fdd/wind-layer/assets/19517451/90542837-b80c-450e-857d-4df6f52b6a49
https://github.com/sakitam-fdd/wind-layer/assets/19517451/064f0ea4-f72f-4e9a-80e7-7a0097f60013
示例图片 ¶
扩展 ¶
Project | Version | Npm | CDN | Description |
---|---|---|---|---|
wind-core | 风场核心渲染,可扩展不可以直接使用 | |||
wind-gl-core | 色斑图核心渲染,可扩展不可以直接使用 | |||
ol-wind | openlayers 6+ 风场扩展插件 | |||
ol5-wind | openlayers 5 风场扩展插件 | |||
openlayers-wind | openlayers 3/4 风场扩展插件 | |||
@sakitam-gis/maptalks-wind | maptalks 风场扩展插件 | |||
amap-wind | 高德地图风场扩展插件 | |||
bmap-wind | 百度地图风场扩展插件 | |||
mapbox-wind | mapbox-gl 风场扩展插件 | |||
maplibre-wind | maplibre-gl 风场扩展插件 | |||
leaflet-wind | Leaflet风场扩展插件 |
特殊说明 ¶
cesium 相关集成请查看 cesium-wind
安装 ¶
使用 pnpm 或 yarn 安装 ¶
TIP
我们推荐使用 pnpm 或 yarn 的方式进行开发, 不仅可在开发环境轻松调试,也可放心地在生产环境打包部署使用, 享受整个生态圈和工具链带来的诸多好处。
相关插件:
# pnpm
pnpm install wind-core
pnpm install wind-gl-core
pnpm install src-wind
pnpm install ol5-wind
pnpm install openlayers-wind
pnpm install @sakitam-gis/maptalks-wind
pnpm install amap-wind
pnpm install bmap-wind
pnpm install leaflet-wind
pnpm install @sakitam-gis/mapbox-wind
pnpm install @sakitam-gis/maplibre-wind
# yarn
yarn add wind-core
yarn add wind-gl-core
yarn add src-wind
yarn add ol5-wind
yarn add openlayers-wind
yarn add @sakitam-gis/maptalks-wind
yarn add amap-wind
yarn add bmap-wind
yarn add leaflet-wind
yarn add @sakitam-gis/mapbox-wind
yarn add @sakitam-gis/maplibre-wind
部分插件亦可以通过浏览器引入 ¶
在浏览器中使用 script
标签直接引入文件,并使用全局变量。
我们在仓库发布包内的 dist
目录下提供了 xxx.js
以及 xxx.min.js
;
基础 ¶
基础使用可以分为三步:
- 引入相应的
WebGIS
地图类库,引入对应的可视化图层扩展插件。 - 正常初始化一个地图。
- 创建一个
WindLayer
,设置风场格点的U V
数据和图层参数 并添加到地图上。
示例 ¶
以下以 maptalks 为例:
npm + es6 ¶
<template>
<div class="demo-content">
<div class="demo-content-datgui"></div>
<div class="map-warp" ref="map"></div>
</div>
</template>
<script>
import 'maptalks/dist/maptalks.css';
import {
Map,
TileLayer,
} from 'maptalks';
import { WindLayer } from 'maptalks-wind';
export default {
name: 'maptalks-wind-base',
data() {
return {};
},
watch: {},
methods: {
initMap() {
const map = new Map(this.$refs.map, {
center: [113.53450137499999, 34.44104525],
zoom: 3,
baseLayer: new TileLayer('base', {
urlTemplate: '//{s}.basemaps.cartocdn.com/dark_all/{z}/{x}/{y}.png',
// urlTemplate: 'https://{s}.basemaps.cartocdn.com/light_all/{z}/{x}/{y}.png',
subdomains: ['a', 'b', 'c', 'd'],
})
});
fetch('/data/wind.json')
.then(res => res.json())
.then(res => {
const windLayer = new WindLayer('wind', res, {
windOptions: {
// colorScale: scale,
velocityScale: 1 / 20,
paths: 5000,
// eslint-disable-next-line no-unused-vars
colorScale: [
"rgb(36,104, 180)",
"rgb(60,157, 194)",
"rgb(128,205,193 )",
"rgb(151,218,168 )",
"rgb(198,231,181)",
"rgb(238,247,217)",
"rgb(255,238,159)",
"rgb(252,217,125)",
"rgb(255,182,100)",
"rgb(252,150,75)",
"rgb(250,112,52)",
"rgb(245,64,32)",
"rgb(237,45,28)",
"rgb(220,24,32)",
"rgb(180,0,35)"
],
// colorScale: scale,
},
// map: map,
// projection: 'EPSG:4326'
});
console.log(map, windLayer);
map.addLayer(windLayer);
});
}
},
mounted() {
this.initMap();
},
};
</script>
<style lang="less">
.demo-content {
width: 100%;
height: 100%;
position: relative;
background-color: #cbe0ff;
&-datgui {
position: absolute;
top: 10px;
left: 10px;
z-index: 1;
pointer-events: auto;
}
.map-warp {
width: 100%;
height: 100%;
}
}
</style>
cdn ¶
<!DOCTYPE html>
<html>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Map - Display a map</title>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/maptalks/dist/maptalks.css">
<style type="text/css">
html, body {
margin: 0;
height: 100%;
width: 100%
}
.container {
width: 100%;
height: 100%
}
</style>
<body>
<div id="map" class="container"></div>
<script src="https://cdn.jsdelivr.net/npm/maptalks/dist/maptalks.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/@sakitam-gis/maptalks-wind/dist/maptalks-wind.js"></script>
<script>
const map = new maptalks.Map('map', {
center: [113.53450137499999, 34.44104525],
zoom: 5,
baseLayer: new maptalks.TileLayer('base', {
// urlTemplate: 'https://{s}.basemaps.cartocdn.com/dark_all/{z}/{x}/{y}.png',
urlTemplate: 'https://{s}.basemaps.cartocdn.com/light_all/{z}/{x}/{y}.png',
subdomains: ['a', 'b', 'c', 'd'],
})
});
fetch('data.json')
.then(res => res.json())
.then(res => {
const windLayer = new MaptalksWind.WindLayer('wind', res, {
windOptions: {
// colorScale: (m) => {
// // console.log(m);
// return '#fff';
// },
colorScale: [
"rgb(36,104, 180)",
"rgb(60,157, 194)",
"rgb(128,205,193 )",
"rgb(151,218,168 )",
"rgb(198,231,181)",
"rgb(238,247,217)",
"rgb(255,238,159)",
"rgb(252,217,125)",
"rgb(255,182,100)",
"rgb(252,150,75)",
"rgb(250,112,52)",
"rgb(245,64,32)",
"rgb(237,45,28)",
"rgb(220,24,32)",
"rgb(180,0,35)"
],
// velocityScale: 1 / 20,
// paths: 5000,
frameRate: 16,
maxAge: 60,
globalAlpha: 0.9,
velocityScale: 1 / 30,
// paths: 10000,
paths: () => { // can be number or function
const zoom = map.getZoom();
return zoom * 1000;
},
},
});
console.log(map, windLayer);
map.addLayer(windLayer);
});
</script>
</body>
</html>