/* 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). */ import {unByKey as ol_Observable_unByKey} from 'ol/Observable.js' import ol_control_Control from 'ol/control/Control.js' import ol_ext_getMapCanvas from '../util/getMapCanvas.js' /** 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 */ var 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 }); // 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); } /** * 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(); this.wind = { angle: a, cos: Math.cos(a), sin: Math.sin(a), speed: speed }; } /** * @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); } } // 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(); } } export default ol_control_Cloud