WebGL教程_Three.js教程_郭隆邦技术博客 郭隆邦_技术博客 Three.js免费视频教程

Three.js Canvas画布上局部渲染(剪裁方法.setScissor()和视口方法.setViewport())

通过Three.js渲染器WebGLRenderer的剪裁方法.setScissor()可以在canvas画布上定义一个局部矩形区域,这个矩形区可以称为剪裁框或剪裁区域。

剪裁方法.setScissor()功能简单说就是控制Threejs渲染器渲染方法.render()影响范围,通过前面课程学习你应该知道,每执行一次渲染器渲染方法.render()就会得到一帧图像,canvas画布上的像素就会更新。在执行渲染器渲染方法.render()之前,如果通过剪裁方法.setScissor()定义了一个剪裁区域,那么执行渲染方法.render()的时候,剪裁框内的像素数据会被清除掉,canvas画布上处于剪裁框范围外的像素不受影响,渲染结果的像素数据只会覆盖剪裁区域内像素,这也就是.setScissor()方法为什么被称为剪裁方法的原因。

剪裁方法.setScissor()

剪裁方法.setScissor()通过四个参数(x, y, width, height)定义一个矩形区域,这个矩形区域称为剪裁区域或剪裁框。

  • x:剪裁框相对canvas画布左侧像素。默认值:0。

  • y:剪裁框相对canvas画布顶部像素。默认值:0。

  • width:剪裁框区域宽度。默认值:canvas画布的宽度。

  • height:剪裁框区域高度。默认值:canvas画布的高度。

代码案例

下面代码在canvas画布上通过剪裁方法.setScissor()定义了一个矩形剪裁区域。

var width = window.innerWidth; //窗口宽度
var height = window.innerHeight; //窗口高度

//开启WebGL剪裁测试功能,如果不开启,.setScissor方法设置的范围不起作用
renderer.setScissorTest(true)
// 渲染函数
function render() {
  // 距离画布左侧50,顶部100   渲染区域宽高度都是画布宽高的三分之一
  renderer.setScissor(100, 200, width / 3, height / 3);
  // setViewport方法设置的视口变换区域和剪裁方法setScissor设置的渲染区域保持一致
  // 注意setScissor方法设置的是render影响范围,setViewport方法影响的是图形变换
  renderer.setViewport(100, 200, width / 3, height / 3)
  // 设置setScissor方法定义的渲染区域的背景颜色
  renderer.setClearColor(0xffff00, 1)
  renderer.render(scene, camera);

  mesh.rotateY(0.01);
  requestAnimationFrame(render);
}
render();

通过剪裁方法.setScissor()设置两个剪裁区域互不影响

//开启WebGL剪裁测试功能,如果不开启,.setScissor方法设置的范围不起作用
renderer.setScissorTest(true)
// 渲染函数
function render() {
  // 距离画布左侧50,顶部100   渲染区域宽高度都是画布宽高的三分之一
  renderer.setScissor(100, 200, width / 3, height / 3);
  // setViewport方法设置的视口变换区域和剪裁方法setScissor设置的渲染区域保持一致
  // 注意setScissor方法设置的是render影响范围,setViewport方法影响的是图形变换
  renderer.setViewport(100, 200, width / 3, height / 3)
  // 设置setScissor方法定义的渲染区域的背景颜色
  renderer.setClearColor(0xffff00, 1)
  renderer.render(scene, camera);

  // 新的剪裁区域
  renderer.setScissor(width / 3+200, 300, width / 3, height / 3);
  renderer.setViewport(width / 3+200, 300, width / 3, height / 3);
  // 新的剪裁区域可以设置一个新的背景颜色
  renderer.setClearColor(0x00ffff, 1)
  renderer.render(scene, camera);

  mesh.rotateY(0.01);
  requestAnimationFrame(render);
}
render();

通过剪裁方法.setScissor()设置两个剪裁区域,一个全屏区域,在全屏上一个角落设置一个局部剪裁区域,显示一些特定效果。

//开启WebGL剪裁测试功能,如果不开启,.setScissor方法设置的范围不起作用
renderer.setScissorTest(true)
// 渲染函数
function render() {
  // 全屏剪裁区
  renderer.setScissor(0, 0, width, height);
  renderer.setViewport(0, 0, width, height);
  renderer.setClearColor(0x000000, 1);//黑色背景
  renderer.render(scene, camera);

  // 右下角剪裁区
  renderer.setScissor(width-width / 3-10, height-height / 3-10, width / 3, height / 3);
  renderer.setViewport(width-width / 3-10, height-height / 3-10, width / 3, height / 3);
  renderer.setClearColor(0x202020, 1);//浅黑色背景
  renderer.render(scene, camera);

  mesh.rotateY(0.01);
  requestAnimationFrame(render);
}
render();

剪裁方法.setScissor()和视口方法.setViewport()功能区分

表面上看剪裁方法.setScissor()和视口方法.setViewport()好像都是定义一个渲染范围。不过要注意剪裁方法.setScissor()定义的是Threejs渲染结果像素覆盖影响的范围,剪裁方法.setScissor()并不影响Threejs场景中模型的平移缩放变换,而视口方法.setViewport()影响Threejs场景中模型的平移缩放变换,你可以通过注释上面代码中视口方法.setViewport()设置的代码,查看canvas画布上的渲染效果来体验视口方法.setViewport()的作用。

剪裁方法.setScissor()和视口方法.setViewport()用途

可以在网页上窗口上创建多个canvas元素,每一个canvas元素是一个渲染显示窗口,可以实现多窗口显示。但是每一个canvas画布对象对应一个独立的WebGL上下文, 不同canvas对象的三维场景渲染资源是不能共享的。开发产品有的时候需要同一份canvas资源显示在窗口的多个的区域中,比如在浏览器窗口的客户区分割出两个显示区域, 同一个三维场景以不同的比例分别显示在两个视口里面。同一个三维场景three.js程序是一样的,没有必要创建两个canvas对象,对于WebGL而言,使用原生WebGL APIgl.viewport()gl.scissor()可以实现, 对于three.js而言使用WebGL渲染器的剪裁方法.setScissor()和视口方法.setViewport()用途即可。

.setScissorTest()方法

Threejs渲染器的剪裁测试方法.setScissorTest()用于开启WebGL剪裁区域功能,如果不开启,.setScissor方法设置的范围不起作用该方法封装了原生WebGL的代码enable( gl.SCISSOR_TEST );disable( gl.SCISSOR_TEST );,具体可以查看目录src\renderers\webgl\WebGLState.js

//开启剪裁测试功能,等价于原生WebGL:enable( gl.SCISSOR_TEST );
renderer.setScissorTest(true)
//关闭剪裁测试功能,等价于原生WebGL:disable( gl.SCISSOR_TEST );
renderer.setScissorTest(false)

封装WebGL API

渲染器WebGLRenderer.setScissor()方法本质上是对原生WebGL APIgl.scissor()的封装。具体可以查看webgl目录的WebGLRenderer.js和WebGLState.js源码文件。

gl.scissor()方法设置了一个剪裁框,它将绘图区域限制为指定的矩形。

关于WebGL APIgl.scissor()的介绍:https://developer.mozilla.org/zh-CN/docs/Web/API/WebGLRenderingContext/scissor。

通过剪裁方法.setScissor()可以实现canvas画布每个剪裁区域渲染效果使用不同的背景颜色。

本站版权所有,本站任何内容未经允许不得转载   备案号:豫ICP备16004767号 QQ群:187740169 (WebGL-Three.js教程)   邮箱:guolongbang@163.com