一个简单利用WebGL绘制频谱瀑布图示例
先看效果

还是比较节省性能的,这个还是包含了生成测试数据的性能,实际应用如果是直接通信获得数据应该还能少几毫秒吧!
准备工作
- 用了React,但是关系不大
- WebGL的基础用法(推荐看一看掘金里的一个教程:WebGL 入门与实践)
- 有兴趣应该读一读这个的源码GPU.JS,因为一开始偷学了一手flatten2dArrayTo这个方法,后来发现实现上述效果的功能的实现原理和GPU
.JS的加速原理差不多的 - 需要稍微写一个fragmentShader的代码;因为要把大多数耗时的计算扔到这个里面执行
重要代码
- 需要将测试数据生成的matrix二维数组通过flatten2dArrayTo转化为Uint8Array
const size = { width: 1400, height: 600 };
const sourceArr = new Float32Array(size.width * size.height).fill(-999.0);
const matrix = []; // 一个二维数组
setInterval(() => {
const res = [];
// TODO 按某些规则生成一行数据放入res
matrix.unshift(res);
if (matrix.length > size.height) {
matrix.pop();
}
flatten2dArrayTo(matrix, sourceArr);
}, 20);
// 二维数组转一维数组
const flatten2dArrayTo = (array, target) => {
let offset = 0;
for (let y = 0; y < array.length; y += 1) {
target.set(array[y], offset);
offset += array[y].length;
}
};
// ...实际给GL使用再将Float32Array转为Uint8Array
const d = sourceArr;
new Uint8Array(d.buffer);
// ...
- 需要给GL定义:调色板作为第0个纹理,数据转化为的Uint8Array当作第1个纹理
const drawDance = (palette, data, width, height, l) => {
if (canRef.current) {
const gl = canRef.current.getContext?.('webgl');
if (gl) {
// 这个暂时没用
const lengthHandle = gl.getUniformLocation(gl.program, 'length');
gl.uniform1f(lengthHandle, l);
// 纹理0:一个256*2的调色板
const paletteLoc = gl.getUniformLocation(gl.program, 'u_Palette');
gl.uniform1i(paletteLoc, 0);
// 纹理1:Float32Array转为Uint8Array的数据纹理
const samplerLoc = gl.getUniformLocation(gl.program, 'u_Sampler');
gl.uniform1i(samplerLoc, 1);
// 纹理0放入
gl.activeTexture(gl.TEXTURE0);
const texture1 = gl.createTexture();
gl.bindTexture(gl.TEXTURE_2D, texture1);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGB, 256 * 2, 1, 0, gl.RGB, gl.UNSIGNED_BYTE, palette);
// 纹理1放入
gl.activeTexture(gl.TEXTURE1);
const texture2 = gl.createTexture();
gl.bindTexture(gl.TEXTURE_2D, texture2);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, width, height, 0, gl.RGBA, gl.UNSIGNED_BYTE, bvbv(data));
// canvas那么大的一个网格
drawBuffer(gl, [-1.0, 1.0, -1.0, -1.0, 1.0, 1.0, 1.0, -1.0], 'a_Position');
// 释放纹理
gl.deleteTexture(texture1);
gl.deleteTexture(texture2);
}
}
};
- canvas尺寸变化的时候告诉GL {width,height} 方便在片元着色器中取到对应点的value
const onResize = useCallback(() => {
const { offsetHeight: height, offsetWidth: width } = containerRef.current || {};
const can = React.createElement('canvas', {
width,
height,
style: { background: '#90202020' },
ref: canRef,
});
containerRef.current && render(can, containerRef.current);
const gl = canRef.current.getContext('webgl');
if (gl) {
// 重新设置gl视口,gl的Oxy在canvas的中心
initShaders(gl, VSHADER_SOURCE, FSHADER_SOURCE);
gl.viewport(0, 0, width, height);
const widthHandle = gl.getUniformLocation(gl.program, 'width');
const heightHandle = gl.getUniformLocation(gl.program, 'height');
gl.uniform1f(widthHandle, width);
gl.uniform1f(heightHandle, height);
// 缩放相关的不重要
const x1H = gl.getUniformLocation(gl.program, 'x1');
const x2H = gl.getUniformLocation(gl.program, 'x2');
gl.uniform1f(x1H, 0);
gl.uniform1f(x2H, width);
}
}, []);
- 一个fragmentShader,主要是要把通过像素位置获取对应二维的数组中对应的数据值(重点是那个decode32函数,网上找了挺久的How do I convert a vec4 rgba value to a float? ),然后根据最小最大和调色板确定绘制的颜色
precision mediump float;
uniform float width;
uniform float height;
uniform float length;
uniform float x1;
uniform float x2;
uniform sampler2D u_Palette;
uniform sampler2D u_Sampler;
vec2 reslution =vec2(width,height);
highp float decode32(highp vec4 rgba) {
highp float Sign = 1.0 - step(128.0,rgba[0])*2.0;
highp float Exponent = 2.0 * mod(rgba[0],128.0) + step(128.0,rgba[1]) - 127.0;
highp float Mantissa = mod(rgba[1],128.0)*65536.0 + rgba[2]*256.0 +rgba[3] + float(0x800000);
highp float Result = Sign * exp2(Exponent) * (Mantissa * exp2(-23.0 ));
return Result;
}
void main(){
// gl_FragColor=vec4(gl_FragCoord.xyx/reslution.xyx,1.0);
if(gl_FragCoord.y < 10.0){
// 简单画个调色板示意图
gl_FragColor=texture2D(u_Palette,gl_FragCoord.xy/reslution.xy);
}
else{
// 绘制数据点
float xx=x1+gl_FragCoord.x/reslution.x*(x2-x1);
vec2 pos = vec2(xx,reslution.y-gl_FragCoord.y)/vec2(reslution.x,reslution.y);
vec4 color=texture2D(u_Sampler,pos)*255.0;
float val=decode32(color.abgr);
if(val < -99.0){
gl_FragColor=vec4(0.0,0.0,0.0,1.0);
}
else{
gl_FragColor=texture2D(u_Palette,vec2((val+20.0)/100.0,1.0));
}
}
}
源码
基本上述代码过程就能实现了;直接看源码吧:ctrlcv->AudioDance8 ctrlCV工程师
一个简单利用WebGL绘制频谱瀑布图示例的更多相关文章
- 一个简单的AXIS远程调用Web Service示例
我们通常都将编写好的Web Service发布在Tomcat或者其他应用服务器上,然后通过浏览器调用该Web Service,返回规范的XML文件.但是如果我们不通过浏览器调用,而是通过客户端程序调用 ...
- svg绘制一个简单地饼图
一个简单地svg绘制饼图的demo,代码如下 <!DOCTYPE html> <html> <head> <meta charset="UTF-8& ...
- ComplexHeatmap|根据excel表绘制突变景观图(oncoplot)
本文首发于“生信补给站”:https://mp.weixin.qq.com/s/8kz2oKvUQrCR2_HWYXQT4g 如果有maf格式的文件,可以直接oncoplot包绘制瀑布图,有多种展示和 ...
- IOS开发之小实例--使用UIImagePickerController创建一个简单的相机应用程序
前言:本篇博文是本人阅读国外的IOS Programming Tutorial的一篇入门文章的学习过程总结,难度不大,因为是入门.主要是入门UIImagePickerController这个控制器,那 ...
- 一个简单的通讯服务框架(大家发表意见一起研究)JAVA版本
最近研究下java语言,根据一般使用的情况,写了个连接通讯服务的框架: 框架结构 C-Manager-S; 把所有通讯内容抽取成三个方法接口:GetData,SetData,带返还的Get; 所有数据 ...
- 分享:计算机图形学期末作业!!利用WebGL的第三方库three.js写一个简单的网页版“我的世界小游戏”
这几天一直在忙着期末考试,所以一直没有更新我的博客,今天刚把我的期末作业完成了,心情澎湃,所以晚上不管怎么样,我也要写一篇博客纪念一下我上课都没有听,还是通过强大的度娘完成了我的作业的经历.(当然作业 ...
- Quartz2D之绘制一个简单的机器猫
学习iOS有一段时间了,在博客园也默默的潜水了两个月,见识了很多大神,收获不少. 今天整理笔记,发现忘记的不少,我感觉需要及时的整理一下了,同时也把做的小东西贴上来和大家分享一下. 最近学习了Quar ...
- 利用javascript和WebGL绘制地球 【翻译】
利用javascript和WebGL绘制地球 [翻译] 原翻译:利用javascript和WebGL绘制地球 [翻译] 在我们所有已知的HTML5API中,WebGL可能是最有意思的一个,利用这个AP ...
- maftools | 从头开始绘制发表级oncoplot(瀑布图)
本文首发于微信公众号 **“ 生信补给站 ”** ,期待您的关注!!! 原文链接:https://mp.weixin.qq.com/s/G-0PtaoO6bYuhx_D_Rlrlw 对于组学数据的分析 ...
- [Unity]利用Mesh绘制简单的可被遮挡,可以探测的攻击指示器
最近做一个小游戏的Demo,最终的效果是这样的 主要是利用Mesh绘制三角形作为显示,然后使用后处理来制作探灯,注意,性能一般,仅仅适合小游戏 分为3步 1:利用mesh绘制三角形,原理很简单,利用三 ...
随机推荐
- sh: vue-cli-service: command not found
mac环境下运行vue项目报错sh: vue-cli-service: command not found 解决方法:cd到项目目录下,执行命令sudo rm -rf node_modules pac ...
- Casdoor 开始
Casdoor 是一个基于 OAuth 2.0 / OIDC 的中心化的单点登录(SSO)身份验证平台,简单来说,就是 Casdoor 可以帮你解决用户管理的难题,你无需开发用户登录.注册等与用户鉴权 ...
- Selenium - 元素等待(1) - 强制等待/隐式等待
Selenium - 元素等待 浏览器的等待可以分为三种:强制等待.隐式等待.显式等待. 强制等待 强制等待使用python自带的 time 模块: time.sleep(second):强制等待,无 ...
- ERROR: Failed to install the following Android SDK packages as some licences have not been accepted.
android studio 配置sdk时提示如下错误 麻麻蛋~ 根据accepted 了解到是安装android-26时未被允许:于是执行如下步骤 1.cd 到sdk目录 D:\develop\An ...
- Hackathon 代码黑客马拉松采访复盘
AIGC Hackathon 2023 北京站 我参加了选手采访提纲,这里我感觉有些点可以分享给大家.之前复盘的链接: 下面是采访我的回答内容: 1. 请向大家简单介绍一下自己吧? 子木,社区名称为程 ...
- CSharp初体验
入门 初来乍到了解一门新的语言,它可能和熟悉的c/c++有不小差别,整体上需要首先了解下语法文件的整体结构.例如,源文件整体结构如何. 乍看CSharp源文件(compile unit)的结构,官网主 ...
- 【模型部署 01】C++实现分类模型(以GoogLeNet为例)在OpenCV DNN、ONNXRuntime、TensorRT、OpenVINO上的推理部署
深度学习领域常用的基于CPU/GPU的推理方式有OpenCV DNN.ONNXRuntime.TensorRT以及OpenVINO.这几种方式的推理过程可以统一用下图来概述.整体可分为模型初始化部分和 ...
- 最全面的Mybatis面试八股文
Mybatis是什么? MyBatis框架是一个开源的数据持久层框架. 它的内部封装了通过JDBC访问数据库的操作,支持普通的SQL查询.存储过程和高级映射,几乎消除了所有的JDBC代码和参数的手工设 ...
- 如何在2023年开启React项目
在这里,我想给你一个新的React项目入门的简要概述.我想反思一下优点和缺点,反思一下作为一个开发者所需要的技术水平,反思一下作为一个React开发者,每个启动项目都能为你提供哪些功能.最后,你将了解 ...
- 从Kafka中学习高性能系统如何设计
1 前言 相信各位小伙伴之前或多或少接触过消息队列,比较知名的包含Rocket MQ和Kafka,在京东内部使用的是自研的消息中间件JMQ,从JMQ2升级到JMQ4的也是带来了性能上的明显提升,并且J ...