You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
188 lines
24 KiB
JavaScript
188 lines
24 KiB
JavaScript
2 weeks ago
|
/* 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 = function(options) {
|
||
|
options = options || {};
|
||
|
var div = document.createElement('div');
|
||
|
div.className = "ol-cloud ol-unselectable ol-control";
|
||
|
ol.control.Control.call(this, {
|
||
|
element: div
|
||
|
});
|
||
|
// Defaut cloud image
|
||
|
this.cloud = options.img;
|
||
|
if (!this.cloud) {
|
||
|
this.cloud = new Image();
|
||
|
this.cloud.src = "
|
||
|
}
|
||
|
this.bird = options.bird;
|
||
|
if (!this.bird) {
|
||
|
this.bird = new Image();
|
||
|
this.bird.src = "";
|
||
|
}
|
||
|
// Parameters
|
||
|
this.set('opacity', options.opacity||0.3);
|
||
|
this.set('density', options.density||0.5);
|
||
|
this.setWind(options);
|
||
|
};
|
||
|
ol.ext.inherits(ol.control.Cloud, ol.control.Control);
|
||
|
/**
|
||
|
* 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
|
||
|
*/
|
||
|
ol.control.Cloud.prototype.setMap = function (map) {
|
||
|
if (this._listener) ol.Observable.unByKey(this._listener);
|
||
|
this._listener = null;
|
||
|
ol.control.Control.prototype.setMap.call(this, map);
|
||
|
if (map) {
|
||
|
this._listener = map.on('postcompose', this.drawCloud_.bind(this));
|
||
|
}
|
||
|
};
|
||
|
/** Set wind direction / force
|
||
|
*/
|
||
|
ol.control.Cloud.prototype.setWind= function (options) {
|
||
|
options = options || {};
|
||
|
var rnd = Math.random;
|
||
|
var a = options.windAngle || rnd()*Math.PI;
|
||
|
var speed = options.windSpeed || rnd();
|
||
|
this.wind = { angle: a, cos: Math.cos(a), sin: Math.sin(a), speed: speed };
|
||
|
}
|
||
|
/**
|
||
|
* @private
|
||
|
*/
|
||
|
ol.control.Cloud.prototype.drawCloud_ = function (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);
|
||
|
}
|
||
|
}
|
||
|
// Draw clouds
|
||
|
var dx = this.wind.cos * this.wind.speed;
|
||
|
var dy = this.wind.sin * this.wind.speed;
|
||
|
for (i=0; i<p.length; i++) {
|
||
|
p[i].x += dx + rnd()*2-1;
|
||
|
p[i].y += dy + rnd()*2-1;
|
||
|
// 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
|
||
|
w = this.bird.width/2;
|
||
|
h = this.bird.height/2;
|
||
|
var sc = 0.5;
|
||
|
var dw = canvas.width+w;
|
||
|
var dh = canvas.height+h;
|
||
|
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();
|
||
|
}
|