19.Three.js 技巧之画布
19 Three.js 技巧之画布
本文讲解一下 画布 Canvas 的一些实用技巧。
- 创建画布截屏(快照),并保存图片到本地
- 设置不清除画布内容
- 获取键盘事件
- 设置画布透明度
- 设置画布为背景
示例基本代码:HelloCanvas
先制作一个简单的、带动画的 Three.js 场景
为了演示各个功能,我们先创建一个基础的 Three.js 动画场景:场景上有 3 个不同颜色、不停旋转的立方体。
这个场景在之前多个示例中已经创建过多次,但是这次和之前的略微不同。
**不同点 1:**由于本文是讲解 canvas 的,本身和 Three.js 没太大的关联,所以这次我们会创建一个 useCreateScene 的自定义 hook,用来专门创建 3D 场景,这样我们的 index.tsx 代码可以更加简洁。
所谓
自定义 react hook
,本质上就是包含有 hook 的普通函数
**不同点 2:**由于讲解过程中需要用到一些按钮,所以我们这次将引入 react-dat-gui 这个组件,来添加一些示例相关按钮。
关于如何使用 react-dat-gui 这个,请参考我写的 React 中使用 GUI.md 这篇教程。
请务必学习一下 react-dat-gui 这个组件,在后续示例中我们会经常使用这个组件来作为调试面板。
具体的代码
我们创建一个专门存放本示例的目录 src/components/hello-canvas/
use-create-scene.ts:
import { useEffect } from 'react'
import * as Three from 'three'
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls'
const useCreateScene = (canvasRef: React.RefObject<HTMLCanvasElement>) => {
useEffect(() => {
if (canvasRef.current === null) { return }
const renderer = new Three.WebGLRenderer({ canvas: canvasRef.current })
const scene = new Three.Scene()
scene.background = new Three.Color(0x222222)
const camera = new Three.PerspectiveCamera(45, 2, 0.1, 100)
camera.position.set(0, 5, 10)
const light = new Three.DirectionalLight(0xFFFFFF, 1)
light.position.set(5, 10, 0)
scene.add(light)
const controls = new OrbitControls(camera, canvasRef.current)
controls.update()
const colors = ['blue', 'red', 'green']
const cubes: Three.Mesh[] = []
colors.forEach((color, index) => {
const mat = new Three.MeshPhongMaterial({ color })
const geo = new Three.BoxBufferGeometry(2, 2, 2)
const mesh = new Three.Mesh(geo, mat)
mesh.position.x = (index - 1) * 4
scene.add(mesh)
cubes.push(mesh)
})
const render = (time: number) => {
time *= 0.001
cubes.forEach((cube) => {
cube.rotation.x = cube.rotation.y = time
})
renderer.render(scene, camera)
window.requestAnimationFrame(render)
}
window.requestAnimationFrame(render)
const handleResize = () => {
if (canvasRef.current === null) { return }
const width = canvasRef.current.clientWidth
const height = canvasRef.current.clientHeight
camera.aspect = width / height
camera.updateProjectionMatrix()
renderer.setSize(width, height, false)
}
handleResize()
window.addEventListener('resize', handleResize)
return () => {
window.removeEventListener('resize', handleResize)
}
}, [canvasRef])
}
export default useCreateScene
index.scss:
.full-screen,
canvas {
display: block;
height: inherit;
width: inherit;
}
.dat-gui {
top: 16px !important;
font-size: 18px !important;
}
index.tsx:
import { useRef, useState } from 'react'
import DatGUI, { DatButton } from 'react-dat-gui'
import useCreateScene from './use-create-scene'
import './index.scss'
import 'react-dat-gui/dist/index.css'
const HelloCanvas = () => {
const canvasRef = useRef<HTMLCanvasElement>(null)
const [date, setDate] = useState<any>({})
useCreateScene(canvasRef)
const handleGUIUpdate = (newDate: any) => {
setDate(newDate)
}
const handleSaveClick = () => {
//编写点击之后的代码
}
return (
<div className='full-screen'>
<canvas ref={canvasRef} className='full-screen' />
<DatGUI data={date} onUpdate={handleGUIUpdate} className='dat-gui' >
<DatButton label='点击保存画布快照' onClick={handleSaveClick} />
</DatGUI>
</div>
)
}
export default HelloCanvas
补充说明:
-
所有创建 3D 场景的代码都转移到了 use-create-scene.ts 中,index.tsx 的代码终于看上去非常简洁了。
-
我们使用了 <DatGUI > 标签,但是由于我们本身只使用 按钮(<DatButton >),并未用到任何其他变量,所以
<DatGUI data={date} onUpdate={handleGUIUpdate} >
这行属性配置只是为了不让 DatGUI 报缺省错误,并无其他作用。DatGUI 的 date、onUpdate 为必填属性
至此,本示例所用到的基础场景代码已搭建好,接下来开始讲解 canvas 的使用技巧。
创建画布截屏(快照),并保存图片到本地
先说一下如何创建画布截屏(快照)
对于 HTML5 中的 canvas 来说,创建画布截屏(快照)有 2 种方式:canvas.toDataURL()、canvas.toBolb()
所谓截屏和快照,更加准确的说法应该是:获取画布当前图片的内容
第 1 种:canvas.toDataURL()
canvas.toDataURL() 可以创建一个临时的图片地址,该图片地址可以作为当前页面中的 <image >标签中的 src 属性值。或者可以创建一个下载链接,点击下载这个图片。
toDataURL()用法:
canvas.toDataURL(type?: string, quality?: any): string;
-
type:图片格式类型,值只能是 “image/png” 或 “image/jpeg”
除了上面 2 个固定值,若你填写其他值则不启作用也不报错,最终会使用 “image/png” 来作为默认值
对于 谷歌浏览器 Chrome ,还额外支持一个类型 “image/webp”
-
quality:jpeg 图片的压缩质量,取值范围 0 - 1,默认值为 0.92
quality 的值越大,图片清晰度越高,文件体积越大
如果 quality 的值不在 0-1 范围内,则会使用默认值 0.92
创建 PNG 图片:
const imgurl = canvas.toDataURL('image/png')
console.log(imgurl)
//输出以下内容
...
创建 JPEG 图片:
const imgurl = canvas.toDataURL('image/jpeg',quality)
console.log(imgurl)
//输出以下内容
...
请注意上面输出内容中,均包含了 ”base64“,这种格式的图片,是可以自动下载的
关于图片自动下载保存到本地我们会稍后讲解
第 2 种:canvas.toBlob()
canvas.toBlob() 可以创建 Blob 对象,该对象包含图片的数据内容。
注意:canvas.toDataURL() 是获得一个临时的图片地址,而 canvas.toBlob() 是获得图片数据内容。
canvas.toBlob()用法:
toBlob(callback: BlobCallback, type?: string, quality?: any): void;
-
callback:获得 Blob 对象的回调函数
通常为:canvas.toBlob( (blob: Blob|null) => {} )
-
type:图片的格式类型,默认值为 “image/png”
-
quality:若图片格式类型为 “image/jpeg”,quality 表示 JPEG 的压缩质量
type 的取值和默认值、quality 的取值范围和默认值 与 canvas.toDataURL() 完全相同
使用示例:
canvas.toBlob((blob) => {
console.log(blob)
})
或
canvas.toBlob((blob) => {
console.log(blob)
}, 'image/jpeg', 0.8)
注意:不同于 canvas.toDataURL(),canvas.toBlob() 这个函数是没有返回值的
另外,假设使用 jpeg 压缩质量为 0.8,文件体积有可能只有 png 格式的 1/3 。
补充说明:
关于分辨率:
按照 MDN 文档,无论哪种方式保存的图片分辨率都是 96,但是我在 PC 机上试验,将下载的图片保存到本地,并在 PhotoShop 软件中查看,发现图片分辨率依然是 72。
我怀疑 保存图片的分辨率其实是和 当前系统一致的。假设在手机上,有可能图片分辨率就是 96 了。
稍后我会在手机上验证一下 图片分辨率 这个问题。
最后,建议你去 MDN 上看 canvas 保存图片 的相关讲解作为本小节的补充。
https://developer.mozilla.org/zh-CN/docs/Web/API/Canvas_API/Tutorial/Pixel_manipulation_with_canvas
再说一下如何自动将图片下载到本地
图片自动下载到本地的思路:在 JS 中创建一个 a 链接,并且模拟出 a 点击事件
使用 toDataURL()函数:
const canvas = canvasRef.current
const imgurl = canvas.toDataURL('image/jpeg', 0.8)
const a = document.createElement('a')
a.href = imgurl
a.download = 'myimg.jpeg'
a.click()
注意:我原本以为还需要将 a 标签插入到网页 body 中才可以实现自动下载,但是经过试验发现根本不需要这样,以下为我原本写的代码:
const a = document.createElement('a') document.body.appendChild(a) //根本无需此行代码 a.style.display = 'none' //由于不需要添加到 body 中,因此也无需此行代码 a.href = imgurl a.download = 'myimg.jpeg' a.click() document.body.removeChild(a) //根本无需此行代码
使用 toBlob()函数:
const canvas = canvasRef.current
canvas.toBlob((blob) => {
const imgurl = window.URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = imgurl
a.download = 'myimg.jpeg'
a.click()
}, 'image/jpeg', 0.8)
小总结:
- 若使用 canvas.toDataURL(),则可以直接将得到的图片临时地址 赋值给 a.href
- 若使用 canvas.toBlob(),则需要通过 window.URL.createObjectURL() 这个函数将 Blob 数据转化得到对应的地址,然后再赋值给 a.href
好了,关于如何获取画布图片数据、如何保持图片到本地讲解完毕,来实践吧。
无论采用 canvas.toDataURL() 还是 canvas.toBlob() 都可以,本示例我们采用 toBlob() 。
我们将 index.stx 的代码修改如下:
import { useRef, useState } from 'react'
import DatGUI, { DatButton } from 'react-dat-gui'
import useCreateScene from './use-create-scene'
import './index.scss'
import 'react-dat-gui/dist/index.css'
const HelloCanvas = () => {
const canvasRef = useRef<HTMLCanvasElement>(null)
const [date, setDate] = useState<any>({})
useCreateScene(canvasRef)
const handleGUIUpdate = (newDate: any) => {
setDate(newDate)
}
const handleSaveClick = () => {
if (canvasRef.current === null) { return }
const canvas = canvasRef.current
//采用 toDataURL() 方式
// const imgurl = canvas.toDataURL('image/jpeg', 0.8)
// const a = document.createElement('a')
// a.href = imgurl
// a.download = 'myimg.jpeg' //我们定义下载图片的文件名
// a.click()
//采用 toBlob() 方式
canvas.toBlob((blob) => {
const imgurl = window.URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = imgurl
a.download = 'myimg.jpeg'
a.click()
}, 'image/jpeg', 0.8)
}
return (
<div className='full-screen'>
<canvas ref={canvasRef} className='full-screen' />
<DatGUI data={date} onUpdate={handleGUIUpdate} className='dat-gui' >
<DatButton label='点击保存画布快照' onClick={handleSaveClick} />
</DatGUI>
</div>
)
}
export default HelloCanvas
实际运行,点击右上角的 按钮,就会给画布创建图片快照,并且自动下载到本地。
然后你可以查看刚刚下载到本地的 myimg.jpeg 这个文件,打开它——你会发现???
怎么图片啥内容都没有?纯色的?3 个立方体呢?
what ?why ?
呵,马上讲解为什么。
问题出在了哪里?
首先我们容易想到,在 use-create-scene.ts 的 render() 函数中,不停的运行着每一帧都进行画布重新渲染的代码,莫非是我们截图那一瞬间刚好画布还未渲染完成?
好,我们先把那行代码删除掉,看是否就可以截图显示有内容了。
const render = (time: number) => {
time *= 0.001
cubes.forEach((cube) => {
cube.rotation.x = cube.rotation.y = time
})
renderer.render(scene, camera)
- window.requestAnimationFrame(render)
}
window.requestAnimationFrame(render)
再次运行,3 个立方体是静止状态,此时点击按钮保存截图。
查看该图,竟然依然是空白,没有内容的。
看来问题并不出在上面一行代码中,我们恢复刚才删除的 window.requestAnimationFrame(render)
,再去想其他原因。
真实的原因是:
- 我们所谓的针对画布截屏 创建快照,实际上是获取 canvas 中的数据
- 但这个数据并不是针对 DOM 中已显示的 canvas,而是针对 canvas 对象中缓冲区的数据
- 关键在于当 canvas 渲染完成后(DOM 中已显示出内容),默认会清空 缓冲区中的数据
- 所以,这就是我们为什么去 “获取 canvas 图像数据时得到是空白内容” 的原因
canvas 从计算到显示的过程:
- canvas 根据相应的 JS 规则,开始创建、计算画布内容数据
- canvas 将计算得到的画布内容数据填充到 canvas 缓冲区
- 当 canvas 画布内容计算完成,此时 canvas 缓冲区已有完整的画布内容数据后,将画布内容显示到 DOM 中
再说一遍:
我们之前的示例中,渲染并显示 canvas 内容的函数 render 和 创建画布快照 的函数是相互独立的,这就造成了当我们去获取 canvas 缓冲区数据时,canvas 已经将画布内容显示到 DOM 中并且清空了缓冲区。
解决办法:
解决办法就是当我们要创建画布快照,获取 canvas 缓冲区内容之前,在同一个函数体内,额外调用一次 render 函数,确保此时 canvas 缓冲区内是有内容的。
实际代码:
第 1:由于我们示例代码中,render 函数本身位于 useCreateScene 函数内部,因此我们需要创造一个 renderRef 的钩子(hook),将 renderRef 对外 return 出去,以便 index.stx 中可以获取 render 函数的引用。
type RenderType = () => void
...
const renderRef = useRef<RenderType | null>(null)
...
renderRef.current = render
...
return renderRef
第 2:这样做引申出另外一个问题,就是我们的 render 函数其实是有参数 time 的:
const render = (time: number) => {
time *= 0.001
cubes.forEach((cube) => {
cube.rotation.x = cube.rotation.y = time
})
renderer.render(scene, camera)
window.requestAnimationFrame(render)
}
window.requestAnimationFrame(render)
而我们希望 index.tsx 中调用 render() 是不传参数 time 的。因为 index.tsx 中根本不存在 time 这个变量,所以我们需要对 渲染 进行适当的改造。
我们将原本的 渲染函数 render() 拆分成 2 个函数:
- 单纯负责渲染的 render 函数
- 负责修改物体属性从而产生动画的 animate 函数
const render = () => {
renderer.render(scene, camera)
}
renderRef.current = render
const animate = (time: number) => {
time *= 0.001
cubes.forEach((cube) => {
cube.rotation.x = cube.rotation.y = time
})
render() //这样 render() 就是一个不需要参数的函数
window.requestAnimationFrame(animate)
}
window.requestAnimationFrame(animate)
经过这样改造后,完整的 use-create-screen.ts 代码如下:
import { useEffect, useRef } from 'react'
import * as Three from 'three'
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls'
type RenderType = () => void
const useCreateScene = (canvasRef: React.RefObject<HTMLCanvasElement>) => {
const renderRef = useRef<RenderType | null>(null)
useEffect(() => {
if (canvasRef.current === null) { return }
const renderer = new Three.WebGLRenderer({ canvas: canvasRef.current })
const scene = new Three.Scene()
scene.background = new Three.Color(0x222222)
const camera = new Three.PerspectiveCamera(45, 2, 0.1, 100)
camera.position.set(0, 5, 10)
const light = new Three.DirectionalLight(0xFFFFFF, 1)
light.position.set(5, 10, 0)
scene.add(light)
const controls = new OrbitControls(camera, canvasRef.current)
controls.update()
const colors = ['blue', 'red', 'green']
const cubes: Three.Mesh[] = []
colors.forEach((color, index) => {
const mat = new Three.MeshPhongMaterial({ color })
const geo = new Three.BoxBufferGeometry(2, 2, 2)
const mesh = new Three.Mesh(geo, mat)
mesh.position.x = (index - 1) * 4
scene.add(mesh)
cubes.push(mesh)
})
const render = () => {
renderer.render(scene, camera)
}
renderRef.current = render
const animate = (time: number) => {
time *= 0.001
cubes.forEach((cube) => {
cube.rotation.x = cube.rotation.y = time
})
render() //这样 render() 就是一个不需要参数的函数
window.requestAnimationFrame(animate)
}
window.requestAnimationFrame(animate)
const handleResize = () => {
if (canvasRef.current === null) { return }
const width = canvasRef.current.clientWidth
const height = canvasRef.current.clientHeight
camera.aspect = width / height
camera.updateProjectionMatrix()
renderer.setSize(width, height, false)
}
handleResize()
window.addEventListener('resize', handleResize)
return () => {
window.removeEventListener('resize', handleResize)
}
}, [canvasRef])
return renderRef
}
export default useCreateScene
index.tsx 完整代码如下:
import { useRef, useState } from 'react'
import DatGUI, { DatButton } from 'react-dat-gui'
import useCreateScene from './use-create-scene'
import './index.scss'
import 'react-dat-gui/dist/index.css'
const HelloCanvas = () => {
const canvasRef = useRef<HTMLCanvasElement>(null)
const [date, setDate] = useState<any>({})
const renderRef = useCreateScene(canvasRef) //获取自定义 hook 返回的 renderRef
const handleGUIUpdate = (newDate: any) => {
setDate(newDate)
}
const handleSaveClick = () => {
if (canvasRef.current === null || renderRef.current === null) { return }
const canvas = canvasRef.current
renderRef.current() //此时调用 render(),进行一次渲染,确保 canvas 缓冲区有数据
//采用 toDataURL() 方式
// const imgurl = canvas.toDataURL('image/jpeg', 0.8)
// const a = document.createElement('a')
// a.href = imgurl
// a.download = 'myimg.jpeg' //我们定义下载图片的文件名
// a.click()
//采用 toBlob() 方式
canvas.toBlob((blob) => {
const imgurl = window.URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = imgurl
a.download = 'myimg.jpeg'
a.click()
}, 'image/jpeg', 0.8)
}
return (
<div className='full-screen'>
<canvas ref={canvasRef} className='full-screen' />
<DatGUI data={date} onUpdate={handleGUIUpdate} className='dat-gui' >
<DatButton label='点击保存画布快照' onClick={handleSaveClick} />
</DatGUI>
</div>
)
}
export default HelloCanvas
调试运行,这次保存的画布快照图片,就不会再是空白,而是有具体内容了。
设置不清除画布内容
上面刚讲到 HTML5 中的 Canvas 每次渲染都存在一个 数据缓冲区的概念,而 Three.js 的渲染器 WebGLRenderer 也同样存在 数据缓冲区 这个概念。
WebGLRenderer 缓冲区内为每次渲染场景得到的画面数据,默认情况下每一次渲染都会清空(释放)上一次的渲染画面数据。
Canvas 数据缓冲区每次清空 这个我们没有办法修改,只能调用渲染函数,重新渲染一次。
但是 WebGLRenderer 的数据缓冲区却是可以通过设置让默认不清除的。
所谓不清除上一次数据缓冲区的内容,本质上就是保留上一次渲染画面内容
所谓不清除画布内容,本质上是让渲染器不清除之前的渲染内容
设置 WebGLRenderer 保留数据缓冲区中的历史数据:
我们只需要将 WebGLRender 的配置修改如下:
const renderer = new Three.WebGLRenderer({
canvas: canvasRef.current,
preserveDrawingBuffer: true,
alpha: true
})
renderer.autoClearColor = false
经过以上的修改之后,每次渲染都会继续保留之前渲染历史画面。
调试运行代码,你就能感受到和之前渲染的不一样效果了。
但是,存在一个问题:当浏览器窗口尺寸改变后,由于执行了 renderer.setSize(),则此时 渲染器中过往的渲染内容将会被清空。
渲染器中的渲染历史内容被清空后,画面就好像第一次刚开始那样,重新开始渲染。
补充说明:当用户在手机上浏览时,手机从竖屏变为横屏时,也会触发重新绘制。
真正的解决方案:离屏渲染
例如使用 WebGLRenderTarget,具体请回顾我们之前讲解的内容:15 Three.js 基础之离屏渲染.md
补充一个示例:PreserveDrawingBuffer
src/components/hello-canvas/preserve-drawing-buffer.tsx
示例目标:
- 创建一个由 6 个立方体,做相互缠绕运动的一个物体
- 创建一个正交镜头 OrthographicCamera
- 给 canvas 添加鼠标滑动监听、以及 手指滑动监听
- 当 鼠标或手指滑动画布时,更新 物体 在镜头中的位置
- 设置渲染器不清除历史画面
最终呈现出的效果:类似一个 画笔在画板上 画画 的效果。
PreserveDrawingBuffer 代码:
import { useEffect, useRef } from 'react'
import * as Three from 'three'
import './index.scss'
const state = { x: 0, y: 0, z: 0 }
const PreserveDrawingBuffer = () => {
const canvasRef = useRef<HTMLCanvasElement | null>(null)
useEffect(() => {
if (canvasRef.current === null) { return }
const renderer = new Three.WebGLRenderer({
canvas: canvasRef.current,
preserveDrawingBuffer: true,
alpha: true
})
renderer.autoClearColor = false
const camera = new Three.OrthographicCamera(-2, 2, 1, -1, -1, 1)
const scene = new Three.Scene()
scene.background = new Three.Color(0xFFFFFF)
const light = new Three.DirectionalLight(0xFFFFFF, 1)
light.position.set(-1, 2, 3)
scene.add(light)
const geometry = new Three.BoxBufferGeometry(1, 1, 1)
const base = new Three.Object3D()
scene.add(base)
base.scale.set(0.1, 0.1, 0.1)
const colors = ['#F00', '#FF0', '#0F0', '#0FF', '#00F', '#F0F']
const numArr = [-2, 2] //同一坐标轴上,对称 2 个立方体的坐标
colors.forEach((color, index) => {
const material = new Three.MeshPhongMaterial({ color })
const cube = new Three.Mesh(geometry, material)
const col = Math.floor(index / numArr.length)
const row = index % numArr.length
let result = [0, 0, 0]
result[col] = numArr[row]
cube.position.set(result[0], result[1], result[2])
base.add(cube)
})
const temp = new Three.Vector3()
const updatePosition = (x: number, y: number) => {
if (canvasRef.current === null) { return }
// const rect = canvasRef.current.getBoundingClientRect()
// const newX = (x - rect.left) * canvasRef.current.width / rect.width
// const newY = (y - rect.top) * canvasRef.current.height / rect.height
// const resX = newX / canvasRef.current.width * 2 - 1
// const resY = newY / canvasRef.current.height * -2 + 1
const resX = x / canvasRef.current.width * 2 - 1
const resY = y / canvasRef.current.height * -2 + 1
temp.set(resX, resY, 0).unproject(camera)
state.x = temp.x
state.y = temp.y
}
const handleMouseMove = (eve: MouseEvent) => {
updatePosition(eve.clientX, eve.clientY)
}
const handleTouchMove = (eve: TouchEvent) => {
eve.preventDefault()
const touche = eve.touches[0]
updatePosition(touche.clientX, touche.clientY)
}
canvasRef.current.addEventListener('mousemove', handleMouseMove)
canvasRef.current.addEventListener('touchmove', handleTouchMove, { passive: false })
const render = (time: number) => {
time = time * 0.001
base.position.set(state.x, state.y, state.z)
base.rotation.x = time
base.rotation.y = time * 1.11
renderer.render(scene, camera)
window.requestAnimationFrame(render)
}
window.requestAnimationFrame(render)
const handleResize = () => {
if (canvasRef.current === null) { return }
const width = canvasRef.current.clientWidth
const height = canvasRef.current.clientHeight
camera.right = width / height
camera.left = - camera.right
camera.updateProjectionMatrix()
renderer.setSize(width, height, false)
}
handleResize()
window.addEventListener('resize', handleResize)
return () => {
window.removeEventListener('resize', handleResize)
}
}, [canvasRef])
return (
<canvas ref={canvasRef} className='full-screen' />
)
}
export default PreserveDrawingBuffer
补充说明:
上面这段代码略微复杂、陌生。因为这段代码中有几个地方是我们之前示例中从来未接触过的:
-
使用的是正交镜头,而不是透视镜头
-
当窗口尺寸发生变化时,更新正交镜头
更新方式和我们之前习惯使用的 透视镜头(PerspectiveCamera) 大不同
-
监听鼠标滑动、手指滑动事件
-
更新 物体 在正交镜头中的 “投影位置”
额外补充:
上述代码中,6 个立方体 他们分别是:
-
在 x 轴对称的 2 个立方体
在示例中,对应的坐标分别为 (-2,0,0)、(2,0,0)
-
在 y 轴对称的 2 个立方体
在示例中,对应的坐标分别为 (0,-2,0)、(0,2,0)
-
在 z 轴堆成的 2 个立方体
在示例中,对应的坐标分别为 (0,0,-2)、(0,0,2)
这 6 个立方体他们依次对应的坐标,没有提前写死,而是通过一段特殊的 forEach 循环来计算得出的。
可以阅读下面这段通用的代码,帮助你理解 整个 forEach 循环是如何得到每个立方体坐标的。
//遍历出 目标长度为N,特殊值为 [xx, xx, ...] 的 多维数组
const numArr = [-2, 2] //定义特殊位置上出现的数字
const arrLength = 3 //定义目标数组长度
const total = arrLength * numArr.length //根据目标数组长度以及特殊数字的个数,计算得出目标数组的总个数
for (let i = 0; i < total; i++) {
const col = Math.floor(i / numArr.length) //计算出特殊位置的索引
const row = i % numArr.length //计算出特殊位置上数字值对应的索引
let result = new Array(arrLength) //得到一个 长度为 arrLenght 的数组
result.fill(0) //将数组每一项填充为 0
result[col] = numArr[row] //修改特殊位置上的值
console.log(result)
}
如果你对 useCreateScene 这个示例不太理解也没有关系,因为毕竟这个示例中出现了一些我们之前示例中从未用到的一些类,你可以先跳过这个示例,继续后面的学习。
随着日后对于 正交镜头 的多次使用,终归会熟练并理解的。
获取键盘事件
让 Canvas 获取键盘事件
必须同时满足以下 2 个条件后,canvas 才可以获得键盘事件。
-
canvas 当前获得焦点
-
<canvas \>
标签中必须添加 tabIndex 属性,属性值是 -1、0、1 都无所谓,建议设置为 0
补充说明:当 canvas 获得当前焦点后,会在四周出现一个蓝色边框,可以通过定义 css 样式来取消这个样式。
canvas:focus {
outline: none;
}
简单示例:
import { useEffect, useRef } from 'react'
import './index.scss'
const CanvasKeyboard = () => {
const canvasRef = useRef<HTMLCanvasElement | null>(null)
useEffect(() => {
if (canvasRef.current === null) { return }
canvasRef.current.focus() //自动获取焦点
const handleKeydown = (event: KeyboardEvent) => {
console.log(event)
}
canvasRef.current.addEventListener('keydown', handleKeydown)
return () => {
if (canvasRef.current === null) { return }
canvasRef.current.removeEventListener('keydown', handleKeydown)
}
}, [canvasRef])
return (
<canvas ref={canvasRef} className='full-screen' tabIndex={0} />
)
}
export default CanvasKeyboard
让 OrbitControl 获取键盘事件
默认 OrbitControl 对象就包含键盘方向键侦听。
键盘上的 上下左右 方向键 均可操控改变 镜头轨道视图。
但是我们之前的代码中,经常是这样写的:
const controls = new OrbitControls(camera, canvasRef.current)
这样存在的问题是,当 canvas 失去焦点后,就无法再获得键盘事件。
最简单的解决办法就是将代码修改为:
const controls = new OrbitControls(camera, document.body)
这样键盘事件就不容易丢失。
设置画布透明度
设置画布透明度,你可能会疑惑,这有什么好讲的,直接通过 css 给 canvas 添加透明度样式即可:
canvas {
opacity: 0.4;
}
这样做肯定没有问题,但是这里说的 “设置画布透明度” 实际上是指 给不同物体设置透明度。
例如我们之前示例中的立方体,那么所有的示例中立方体都不是半透明的。
给材质设置透明度:
-
需要给材质设置透明度
const mat = new Three.MeshPhongMaterial({ color, opacity: 0.4 })
-
渲染器需要开启透明度渲染
const renderer = new Three.WebGLRenderer({ canvas: canvasRef.current, alpha:true, premultipliedAlpha:false })
alpha:canvas 是否包含透明度,默认为 false
premultipliedAlpha:renderer 是否假设颜色有 premultiplied alpha (预乘 alpha),默认为 true
针对 预乘 Alpha 的补充说明:
premultiplied alpha:颜色值 预乘 alpha
这是传统 3D 绘制中的一个重要概念,你可以简单理解成如下:
假设我们要表示一个 透明度为 60% 的纯红色,采用 RGBA 的方式为 (255,0,0,0.6),通过 预乘 alpha,我们可以得到透明度为 60% 的纯红色如果放置在纯白色底上,实际上最终呈现出来的颜色和 RGB ( 255,102,102) 是完全相同的。
假设一个颜色使用 RGBA 来表示,透明度为 A、rgb 颜色为 C、纯白色(255,255,255)为 F,那么把 RGBA 转化为 RGB 的公式为:
C*A + ( 1-A ) * F
也就是说 rgba(255,0,0,0.6) 转化为对应的 rgb 过程为:
(255,0,0)_0.6 + (1-0.6) _ (255,255,255) = (255, 255 x 0.4, 255 x 0.4) = (255,102,102)
暂时看不懂没有关系,只需记住若想让渲染器将物体渲染出半透明,除了物体本身材质配置透明度以外,还需要将渲染器中的 alpha 设置为 true 、premultipliedAlpha 设置为 false
设置画布为背景
将 canvas 设置为网页背景,事实上也很简单,对应的样式:
canvas {
position: fixed;
top: 0;
left: 0;
z-index: -1;
}
上述 CSS 样式就让 canvas 位置固定,且层级最低,这样就成为当前网页背景了。
但是实际项目中,更加建议将 canvas 包含在一个 iframe 中后,再作为 网页的背景。
这样做有几个理由:
- 使用 iframe 后,可以将 canvas、Three.js 的相关代码独立出来
- 可以多个页面都引用这个 iframe
iframe 的相关示例:
<iframe id='background' src='xxx.html' >
<div>
Hello Three.js
</div>
#background {
position: fixed;
width:100%;
height:100%;
left:0;
top:0;
z-index:-1;
border:none;
pointer-events:none;
}
上述 css 样式中:
- position: fixed; 可以让 iframe 位置固定
- z-index: -1; 可以让 iframe 层级最低
- border: none; 可以让 iframe 不显示边框
- pointer-events: none; 让 iframe 永远不会成为鼠标事件的 target,意味着让 iframe 不接受鼠标交互事件
关于更多 canvas 的相关用法,建议阅读 MDN 上关于 canvas 的相关文档:
https://developer.mozilla.org/zh-CN/docs/Web/API/Canvas_API
至此,关于 Three.js 的一些常用技巧讲解完毕。
接下来开始讲解 Three.js 的一些性能优化。