openlayers 云层动画 自定义渲染
最后出来的效果 gif有点模糊
这里使用了ol_ext库 里面的云效果 然后对里面的一些代码进行了简单修改
线上示例代码 上面的gif模糊 可以看这里的实现效果
这个是根据 https://blog.csdn.net/qq_36287830/article/details/131844745改进来的
关键代码:
样式
<style>
.input {
position: fixed;
top: 10px;
right: 10px;
border-radius: 10px;
background: #fff;
display: flex;
flex-direction: column;
padding: 5px;
padding-bottom: 10px;
>* {
margin-top: 10px;
display: flex;
align-items: center;
}
}
</style>
用来控制的dom
<div class="input">
<div>
<span>透明程度:</span><input type="range" name="blur" id="blur" min="0" max="1" step="0.01" value="1" />
</div>
<div>
<span>密集程度:</span><input type="range" name="density" id="density" min="0" step="0.01" max="1" value="1" />
</div>
<div>
<span>运动速度:</span><input type="range" name="speed" id="speed" min="0" step="0.01" max="1000" value="5" />
</div>
<div>
<span>方向角度:</span><input type="range" name="angle" id="angle" min="0" step="1" max="360" value="90" />
</div>
</div>
关键使用插件的js代码
var Cloud= new ol.control.Cloud({
opacity: parseFloat(document.getElementById('blur').value),
density: parseFloat(document.getElementById('density').value),
speed: parseFloat(document.getElementById('speed').value),
angle: parseFloat(document.getElementById('angle').value) * Math.PI / 180,
bird: false
});
map.addControl(Cloud);
document.getElementById('blur').addEventListener("input", (target) => ctrl.set('opacity', parseFloat(target.target.value)))
document.getElementById('speed').addEventListener("input", (target) => ctrl.set('speed', parseFloat(target.target.value)))
document.getElementById('density').addEventListener("input", (target) => ctrl.set('density', parseFloat(target.target.value)))
document.getElementById('angle').addEventListener("input", (target) => ctrl.set('angle', parseFloat(target.target.value)))
关键修改ol_ext后的代码 可以按这个格式写自定义的渲染方法
/* Copyright (c) 2016 Jean-Marc VIGLINO,
released under the CeCILL-B license (French BSD license)
(http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.txt).
*/
/** ol.control.Cloud adds an old map effect on a canvas renderer.
* It colors the map, adds a parchment texture and compass onto the map.
* @constructor
* @param {Object} options
* @param {_ol_color_} options.hue color to set hue of the map, default #963
* @param {Number} options.saturation saturation of the hue color, default 0.6
* @param {Number} options.opacity opacity of the overimpose image, default 0.7
* @todo add effects on pan / zoom change
*/
ol.control.Cloud = class olcontrolCloud extends ol.control.Control {
constructor(options) {
options = options || {};
var div = document.createElement("div");
div.className = "ol-cloud ol-unselectable ol-control";
super({
element: div,
});
this.options = options;
this.updated();
}
updated() {
// Defaut cloud image
this.cloud = this.options.img;
if (!this.cloud) {
this.cloud = new Image();
//base64的云的图片
this.cloud.src ="";
}
this.bird = this.options.bird;
if (this.bird) {
this.bird = new Image();
//bse64的鸟的图片
this.bird.src ="";
}
// Parameters
for (let i of Object.keys(this.options)) {
this.set(i, this.options[i] || 0.3);
}
this.setWind(this.options);
}
/**
* Remove the control from its current map and attach it to the new map.
* Subclasses may set up event handlers to get notified about changes to
* the map here.
* @param {_ol_Map_} map Map.
* @api stable
*/
setMap(map) {
if (this._listener) ol.Observable.unByKey(this._listener);
this._listener = null;
super.setMap(map);
if (map) {
this._listener = map.on("postcompose", this.drawCloud_.bind(this));
}
}
/** Set wind direction / force
*/
setWind(options) {
options = options || {};
var rnd = Math.random;
var a = options.windAngle || rnd() * Math.PI;
var speed = options.windSpeed || rnd();
console.log("修改");
}
/**
* @private
*/
drawCloud_(event) {
if (!this.getMap()) return;
var ctx =
event.context || ol.ext.getMapCanvas(this.getMap()).getContext("2d");
var canvas = ctx.canvas;
// var ratio = event.frameState.pixelRatio;
// var m = Math.max(canvas.width, canvas.height);
// var res = view.getResolution()/ratio;
// var rot = view.getRotation();
// Not ready !
if (!this.cloud.width) return;
// Go!
var p = this.particules;
var rnd = Math.random;
var w = this.cloud.width;
var h = this.cloud.height;
var w2 = this.cloud.width / 2;
var h2 = this.cloud.height / 2;
var d =
((this.get("density") * 10 * canvas.width * canvas.height) / w / h) << 0;
var i;
function addClouds(nb) {
for (i = 0; i < nb; i++) {
p.push({ x: rnd() * canvas.width - w2, y: rnd() * canvas.height - h2 });
}
}
// First time: init clouds
if (!p) {
p = this.particules = [];
addClouds(d);
// Wind
this.width = canvas.width;
this.height = canvas.height;
// Birds
this.birds = [];
for (i = 0; i < 5; i++) {
var b = {
angle: rnd() * 2 * Math.PI,
x: rnd() * canvas.width,
y: rnd() * canvas.height,
rot: 0,
fly: 0,
};
b.cos = Math.cos(b.angle);
b.sin = Math.cos(b.angle);
this.birds.push(b);
}
} else if (d != p.length) {
// Parameters changed
if (this.width !== canvas.width || this.height !== canvas.height) {
p = this.particules = [];
addClouds(d);
this.width = canvas.width;
this.height = canvas.height;
} else if (d > p.length) {
addClouds(1);
} else if (d < p.length) {
p.splice((p.length * rnd()) << 0, 1);
}
}
var dx = Math.cos((this.get("angle") * Math.PI) / 180) * this.get("speed");
var dy = Math.sin((this.get("angle") * Math.PI) / 180) * this.get("speed");
for (i = 0; i < p.length; i++) {
p[i].x += dx;
p[i].y += dy;
// out!
if (p[i].x < -w) {
p[i].x = canvas.width;
p[i].y = rnd() * canvas.height - h2;
} else if (p[i].x > canvas.width) {
p[i].x = -w;
p[i].y = rnd() * canvas.height - h2;
}
if (p[i].y < -h) {
p[i].y = canvas.height;
p[i].x = rnd() * canvas.width - w2;
} else if (p[i].y > canvas.height) {
p[i].y = -h;
p[i].x = rnd() * canvas.width - w2;
}
}
// Draw clouds
ctx.globalAlpha = this.get("opacity");
for (i = 0; i < p.length; i++) {
ctx.drawImage(this.cloud, p[i].x, p[i].y);
}
ctx.globalAlpha = 1;
// Draw birds
var sc = 0.5;
var dw = canvas.width + w;
var dh = canvas.height + h;
if (this.bird) {
w = this.bird.width / 2;
h = this.bird.height / 2;
for (i = 0; i < this.birds.length; i++) {
var bi = this.birds[i];
// Animate birds
var sx = 0;
if (bi.fly) {
bi.fly = ++bi.fly % 5;
sx = -0.1;
} else if (rnd() < 0.01) bi.fly = 1;
// Rotate birds
if (bi.rot) {
bi.angle += bi.rot;
bi.cos = Math.cos(bi.angle);
bi.sin = Math.sin(bi.angle);
}
if (rnd() < 0.01) {
bi.rot = bi.rot ? 0 : (rnd() * Math.PI) / 200 - Math.PI / 400;
bi.cos = Math.cos(bi.angle);
bi.sin = Math.sin(bi.angle);
}
// Move birds
bi.x += bi.sin;
if (bi.x > dw) bi.x = -w;
if (bi.x < -w) bi.x = dw;
bi.y -= bi.cos;
if (bi.y > dh) bi.y = -h;
if (bi.y < -h) bi.y = dh;
// Draw birds
ctx.save();
ctx.translate(bi.x, bi.y);
ctx.rotate(bi.angle);
ctx.scale(sc + sx, sc);
ctx.drawImage(this.bird, -w, -h);
}
ctx.restore();
}
// Continue animation
this.getMap().render();
}
};