解决webgl画面虚问题

复现

vite搭建了新项目,设置canvas,然后加载3d模型很虚,然后创建一个点也很虚。

如图所示:

javascript 复制代码
#app {
        width: 100%;
        height: 100%;
      }

<canvas id="app"></canvas>

原因

画布分辨率问题确实是WebGL中导致图形模糊的最主要原因

实际发生的情况:

默认canvas尺寸:Canvas元素默认是300×150像素

CSS拉伸:CSS将300×150像素的画布拉伸到800×600像素

像素化拉伸:WebGL实际上是在300×150的像素缓冲区中渲染,然后被拉伸显示。

解决方案

javascript 复制代码
// 在获取WebGL上下文后,设置canvas的实际像素尺寸
const setCanvasSize = (canvas,gl) => {
  const dpr = window.devicePixelRatio || 1;
  const rect = canvas.getBoundingClientRect();
  
  // 设置canvas的实际像素尺寸(与CSS尺寸分离)
  canvas.width = rect.width * dpr;
  canvas.height = rect.height * dpr;
  
  // 设置WebGL视口
  gl.viewport(0, 0, canvas.width, canvas.height);
};
export {
  setCanvasSize
}

  // 初始化和窗口大小变化时都调用
setCanvasSize(canvas,gl);
window.addEventListener('resize', () => {
  setCanvasSize(canvas,gl)
});

试试鼠标点击生成点的方法中:

javascript 复制代码
  let ponitArr = []
  let pointColorArr = []
  // 鼠标点击
  canvas.addEventListener('click', (event) =>  {
  
    const rect = canvas.getBoundingClientRect();
  
  // 使用 clientWidth/clientHeight,而不是 width/height
    let x = event.clientX - rect.left;
    let y = event.clientY - rect.top;
    
    // 正确公式:
    x = (x / canvas.clientWidth) * 2 - 1;
    y = 1 - (y / canvas.clientHeight) * 2; // Y轴翻转
  
    let point = new Float32Array([x, y, 0]);
  
    let color = new Float32Array([0.0, 1.0, 0.0, 1.0]);
    if(x<0 && y<0) {
      color = new Float32Array([1.0, 0.0, 0.0, 1.0]); // 左下 红色
    } else if(x<0 && y>0){
      color = new Float32Array([0.0, 1.0, 0.0, 1.0]); // 左上 绿色
    } else if(x>0 && y<0){
      color = new Float32Array([0.0, 0.0, 1.0, 1.0]); // 右下 蓝色
    } else if(x>0 && y>0){
      color = new Float32Array([1.0, 1.0, 0.0, 1.0]); // 右上 黄色
    }
    ponitArr.push(point);
    pointColorArr.push(new Float32Array(color));
  
    gl.clear(gl.COLOR_BUFFER_BIT);
    console.log(ponitArr);
    
    ponitArr.forEach((point, index) => {
      gl.vertexAttrib3f(a_position, point[0], point[1], 0.0);
      gl.uniform4fv(u_FragColor, pointColorArr[index]);
      gl.drawArrays(gl.POINTS, 0, 1);
    });

    
  }); 
  

总结

记住这个黄金法则:

在WebGL中,canvas.width/height(像素尺寸)必须匹配CSS的显示尺寸乘以设备像素比,并且必须调用gl.viewport()设置相同的尺寸。

这是WebGL开发中最常见的"坑"之一!

相关推荐
cTz6FE7gA17 小时前
WebGL实战:用Three.js创建3D场景,实现沉浸式Web体验
前端·javascript·webgl
三维搬砖者6 天前
AI 解密大厂 Three.js 三维引擎开发 03|从经纬度到三维世界的坐标解码
webgl·three.js
threelab10 天前
引擎案例分析 02|GeoLayer 大厂地理可视化方案深度拆解
javascript·3d·webgl
山海鲸可视化10 天前
【山海鲸功能演示】如何设置选中按钮的时候其他按钮切换为默认样式?
webgl·可视化·数据可视化·数据表格·搜索框
kadog10 天前
GraphX:基于 WebGL 区间算术的 GPU 加速隐函数绘图器
前端·javascript·数学建模·webgl
ct97811 天前
Cesium的Primitive API
gis·webgl·cesium
sin°θ_陈14 天前
前馈式3D Gaussian Splatting 研究地图(路线二):几何优先的前馈式 3DGS——前馈式 3DGS 如何重新拥抱多视图几何
深度学习·3d·webgl·三维重建·空间计算·3dgs·空间智能
星河耀银海15 天前
3D效果:HTML5 WebGL结合AI实现智能3D场景渲染
前端·人工智能·深度学习·3d·html5·webgl
点量云实时渲染-小芹16 天前
Unity模型数字孪生虚拟仿真webgl推流卡实时云渲染推流
unity·webgl·数字孪生·实时云渲染·虚拟仿真·云推流
WebGISer_白茶乌龙桃17 天前
基于 Cesium 的 GLB 建筑模型分层分房间点击拾取技术实现
前端·javascript·vue.js·webgl·cesium