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

/* 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();
}