IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> 栅格数据灰度化并前端转换展示 -> 正文阅读

[JavaScript知识库]栅格数据灰度化并前端转换展示

概述

对于栅格数据,为提升网络的传输效率,通过一定的计算公式将栅格数据压缩成一个灰度图,在通过客户端进行解析渲染是常见的一种手段。本文将结合canvas实现此功能。

效果

灰度图转换

地图叠加效果

测试数据

测试数据来源于ventusky上的温度图,渲染的色标也用了该网站上面的。

实现代码

  const canvas = document.getElementById('canvas');
  canvas.width = window.innerWidth
  canvas.height = window.innerHeight
  const ctx = canvas.getContext('2d');

  const legends = [{"v":50,"c":"rgb(43, 0, 1)"},{"v":40,"c":"rgb(107, 21, 39)"},{"v":30,"c":"rgba(190, 48, 102, 0.92)"},{"v":25,"c":"rgba(229, 109, 83, 0.92)"},{"v":20,"c":"rgba(234, 164, 62, 0.92)"},{"v":15,"c":"rgba(235, 215, 53, 0.92)"},{"v":10,"c":"rgba(190, 228, 61, 0.92)"},{"v":5,"c":"rgba(89, 208, 73, 0.92)"},{"v":0,"c":"rgba(75, 182, 152, 0.92)"},{"v":-5,"c":"rgba(62, 121, 198, 0.92)"},{"v":-10,"c":"rgba(85, 78, 177, 0.92)"},{"v":-15,"c":"rgba(36, 24, 106, 0.92)"},{"v":-20,"c":"rgba(145, 9, 145, 0.92)"},{"v":-30,"c":"rgba(255, 170, 255, 0.92)"},{"v":-40,"c":"rgba(238, 238, 238, 0.92)"}]

  class Palette {
    constructor(legends) {
      this._init(legends)
    }

    _init(legends) {
      // 创建canvas
      let canvas = document.createElement("canvas");
      canvas.width = 10;
      canvas.height = 256;
      let ctx = canvas.getContext("2d");

      // 创建线性渐变色
      let linearGradient = ctx.createLinearGradient(0, 0, 0, canvas.height);
      const max = legends[0].v,
        min = legends[legends.length - 1].v - 10
      this.colorStops = {}
      legends.forEach(legend => {
        const k = (legend.v - min) / (max - min)
        linearGradient.addColorStop(k, legend.c);
      })

      // 绘制渐变色条
      ctx.fillStyle = linearGradient;
      ctx.fillRect(0, 0, canvas.width, canvas.height);

      // 读取像素数据
      this.imageData = ctx.getImageData(0, 0, 1, canvas.height).data;
      this.canvas = canvas;
    }

    colorPicker(position) {
      return this.imageData.slice(position * 4, position * 4 + 3);
    }
  }

  const palette = new Palette(legends)

  const img = new Image()
  img.src = './tem.jpg'
  let imgData = null
  img.onload = function () {
    // 离屏
    const w = img.width, h = img.height
    const _canvas = document.createElement('canvas');
    _canvas.width = w
    _canvas.height = h
    const _ctx = _canvas.getContext('2d');
    _ctx.drawImage(img, 0, 0)
    imgData = _ctx.getImageData(0, 0, w, h)
    ctx.putImageData(imgData, 0, 0)

    const data = imgData.data
    let val = []
    for (let i = 0; i < data.length; i+=4) {
      let pos = data[i];
      val.push(pos)
    }
    val = val.sort((a, b) => a - b)
    const min = val[0],
      max = val[val.length - 1]

    for (let i = 0; i < data.length; i+=4) {
      let pos = data[i];
      pos = Math.floor(((pos - min) / (max - min)) * 256)
      let color = palette.colorPicker(pos);
      data[i] = color[0];
      data[i + 1] = color[1];
      data[i + 2] = color[2];
      data[i + 3] = 255;
    }
    // 展示img
    ctx.putImageData(imgData, w + 10, 0)

    // 绘制图例
    const _w = palette.canvas.width, _h = palette.canvas.height / 2
    const x = 25, y = canvas.height - 20 - _h
    ctx.fillStyle = 'white'
    ctx.shadowBlur = 10
    ctx.shadowColor = '#ccc'
    ctx.shadowOffsetX = 2
    ctx.shadowOffsetY = 2
    ctx.textAlign = 'left'
    ctx.fillRect(x - 10, y - 10, _w + 40, _h + 20)
    ctx.drawImage(palette.canvas, x, y, _w, _h)
    // 绘制文字
    ctx.fillStyle = '#000'
    ctx.textBaseline = 'top'
    ctx.fillText(legends[legends.length - 1].v, x + _w + 6, y)
    ctx.textBaseline = 'bottom'
    ctx.fillText(legends[0].v, x + _w + 6, y + _h)
  }

结合到openlayers中:

  const legends = [{"v":50,"c":"rgb(43, 0, 1)"},{"v":40,"c":"rgb(107, 21, 39)"},{"v":30,"c":"rgba(190, 48, 102, 0.92)"},{"v":25,"c":"rgba(229, 109, 83, 0.92)"},{"v":20,"c":"rgba(234, 164, 62, 0.92)"},{"v":15,"c":"rgba(235, 215, 53, 0.92)"},{"v":10,"c":"rgba(190, 228, 61, 0.92)"},{"v":5,"c":"rgba(89, 208, 73, 0.92)"},{"v":0,"c":"rgba(75, 182, 152, 0.92)"},{"v":-5,"c":"rgba(62, 121, 198, 0.92)"},{"v":-10,"c":"rgba(85, 78, 177, 0.92)"},{"v":-15,"c":"rgba(36, 24, 106, 0.92)"},{"v":-20,"c":"rgba(145, 9, 145, 0.92)"},{"v":-30,"c":"rgba(255, 170, 255, 0.92)"},{"v":-40,"c":"rgba(238, 238, 238, 0.92)"}]

  class Palette {
    constructor(legends) {
      this._init(legends)
    }

    _init(legends) {
      // 创建canvas
      let canvas = document.createElement("canvas");
      canvas.width = 10;
      canvas.height = 256;
      let ctx = canvas.getContext("2d");

      // 创建线性渐变色
      let linearGradient = ctx.createLinearGradient(0, 0, 0, canvas.height);
      const max = legends[0].v,
        min = legends[legends.length - 1].v - 10
      this.colorStops = {}
      legends.forEach(legend => {
        const k = (legend.v - min) / (max - min)
        linearGradient.addColorStop(k, legend.c);
      })

      // 绘制渐变色条
      ctx.fillStyle = linearGradient;
      ctx.fillRect(0, 0, canvas.width, canvas.height);

      // 读取像素数据
      this.imageData = ctx.getImageData(0, 0, 1, canvas.height).data;
      this.canvas = canvas;
    }

    colorPicker(position) {
      return this.imageData.slice(position * 4, position * 4 + 3);
    }
  }

  const palette = new Palette(legends)

  const img = new Image()
  img.src = './data/tem.jpg'
  img.onload = function () {
    // 离屏
    const w = img.width, h = img.height
    const _canvas = document.createElement('canvas');
    _canvas.width = w
    _canvas.height = h
    const _ctx = _canvas.getContext('2d');
    _ctx.drawImage(img, 0, 0)
    const imgData = _ctx.getImageData(0, 0, w, h)
    const data = imgData.data
    let val = []
    for (let i = 0; i < data.length; i+=4) {
      let pos = data[i];
      val.push(pos)
    }
    val = val.sort((a, b) => a - b)
    const min = val[0],
      max = val[val.length - 1]

    for (let i = 0; i < data.length; i+=4) {
      let pos = data[i];
      pos = Math.floor(((pos - min) / (max - min)) * 256)
      let color = palette.colorPicker(pos);
      data[i] = color[0];
      data[i + 1] = color[1];
      data[i + 2] = color[2];
      data[i + 3] = 150;
    }
    // 展示img
    _ctx.putImageData(imgData, 0, 0)
    const imgLayer = new ol.layer.Image({
      source: new ol.source.ImageStatic({
        url: _canvas.toDataURL(),
        imageExtent: [-180, -90, 180, 90]
      })
    })
    map.addLayer(imgLayer);
}
  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2021-12-03 12:57:17  更:2021-12-03 12:59:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/6 14:12:30-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码