Bootstrap

openlayers 云动画 自定义渲染

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

;