19.Three.js 技巧之画布
19 Three.js 技巧之画布
本文讲解一下 画布
- 创建画布截屏
( 快照) ,并保存图片到本地 - 设置不清除画布内容
- 获取键盘事件
- 设置画布透明度
- 设置画布为背景
示例基本代码:HelloCanvas
先制作一个简单的、带动画的Three.js 场景
为了演示各个功能,我们先创建一个基础的
这个场景在之前多个示例中已经创建过多次,但是这次和之前的略微不同。
所谓
自定义 react hook
,本质上就是包含有hook 的普通函数
关于如何使用
请务必学习一下
react-dat-gui 这个组件,在后续示例中我们会经常使用这个组件来作为调试面板。
具体的代码
我们创建一个专门存放本示例的目录
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 为必填属性
至此,本示例所用到的基础场景代码已搭建好,接下来开始讲解
创建画布截屏( 快照) ,并保存图片到本地
先说一下如何创建画布截屏( 快照)
对于
所谓截屏和快照,更加准确的说法应该是:获取画布当前图片的内容
第1 种:canvas.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
创建
const imgurl = canvas.toDataURL('image/png')
console.log(imgurl)
//输出以下内容
...
创建
const imgurl = canvas.toDataURL('image/jpeg',quality)
console.log(imgurl)
//输出以下内容
...
请注意上面输出内容中,均包含了 ”base64“,这种格式的图片,是可以自动下载的
关于图片自动下载保存到本地我们会稍后讲解
第2 种:canvas.toBlob()
注意:
canvas.toDataURL() 是获得一个临时的图片地址,而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 。
补充说明:
关于分辨率:
按照
我怀疑 保存图片的分辨率其实是和 当前系统一致的。假设在手机上,有可能图片分辨率就是
稍后我会在手机上验证一下 图片分辨率 这个问题。
最后,建议你去
https://developer.mozilla.org/zh-CN/docs/Web/API/Canvas_API/Tutorial/Pixel_manipulation_with_canvas
再说一下如何自动将图片下载到本地
图片自动下载到本地的思路:在
使用
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) //根本无需此行代码
使用
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
好了,关于如何获取画布图片数据、如何保持图片到本地讲解完毕,来实践吧。
无论采用
我们将
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
实际运行,点击右上角的 按钮,就会给画布创建图片快照,并且自动下载到本地。
然后你可以查看刚刚下载到本地的
怎么图片啥内容都没有?纯色的?
what ?why ?
呵,马上讲解为什么。
问题出在了哪里?
首先我们容易想到,在
好,我们先把那行代码删除掉,看是否就可以截图显示有内容了。
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)
再次运行,
查看该图,竟然依然是空白,没有内容的。
看来问题并不出在上面一行代码中,我们恢复刚才删除的 window.requestAnimationFrame(render)
,再去想其他原因。
真实的原因是:
- 我们所谓的针对画布截屏 创建快照,实际上是获取
canvas 中的数据 - 但这个数据并不是针对
DOM 中已显示的canvas ,而是针对canvas 对象中缓冲区的数据 - 关键在于当
canvas 渲染完成后(DOM 中已显示出内容) ,默认会清空 缓冲区中的数据 - 所以,这就是我们为什么去 “获取
canvas 图像数据时得到是空白内容” 的原因
canvas 根据相应的JS 规则,开始创建、计算画布内容数据canvas 将计算得到的画布内容数据填充到canvas 缓冲区- 当
canvas 画布内容计算完成,此时canvas 缓冲区已有完整的画布内容数据后,将画布内容显示到DOM 中
再说一遍:
我们之前的示例中,渲染并显示
解决办法:
解决办法就是当我们要创建画布快照,获取
实际代码:
第
type RenderType = () => void
...
const renderRef = useRef<RenderType | null>(null)
...
renderRef.current = render
...
return renderRef
第
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)
而我们希望
我们将原本的 渲染函数
- 单纯负责渲染的
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)
经过这样改造后,完整的
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
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
调试运行,这次保存的画布快照图片,就不会再是空白,而是有具体内容了。
设置不清除画布内容
上面刚讲到
但是
所谓不清除上一次数据缓冲区的内容,本质上就是保留上一次渲染画面内容
所谓不清除画布内容,本质上是让渲染器不清除之前的渲染内容
设置
我们只需要将
const renderer = new Three.WebGLRenderer({
canvas: canvasRef.current,
preserveDrawingBuffer: true,
alpha: true
})
renderer.autoClearColor = false
经过以上的修改之后,每次渲染都会继续保留之前渲染历史画面。
调试运行代码,你就能感受到和之前渲染的不一样效果了。
但是,存在一个问题:当浏览器窗口尺寸改变后,由于执行了
渲染器中的渲染历史内容被清空后,画面就好像第一次刚开始那样,重新开始渲染。
补充说明:当用户在手机上浏览时,手机从竖屏变为横屏时,也会触发重新绘制。
真正的解决方案:离屏渲染
例如使用
补充一个示例:PreserveDrawingBuffer
src/components/hello-canvas/preserve-drawing-buffer.tsx
示例目标:
- 创建一个由
6 个立方体,做相互缠绕运动的一个物体 - 创建一个正交镜头
OrthographicCamera - 给
canvas 添加鼠标滑动监听、以及 手指滑动监听 - 当 鼠标或手指滑动画布时,更新 物体 在镜头中的位置
- 设置渲染器不清除历史画面
最终呈现出的效果:类似一个 画笔在画板上 画画 的效果。
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) 大不同 -
监听鼠标滑动、手指滑动事件
-
更新 物体 在正交镜头中的 “投影位置”
额外补充:
上述代码中,
-
在
x 轴对称的2 个立方体在示例中,对应的坐标分别为
(-2,0,0) 、(2,0,0) -
在
y 轴对称的2 个立方体在示例中,对应的坐标分别为
(0,-2,0) 、(0,2,0) -
在
z 轴堆成的2 个立方体在示例中,对应的坐标分别为
(0,0,-2) 、(0,0,2)
这
可以阅读下面这段通用的代码,帮助你理解 整个
//遍历出 目标长度为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)
}
如果你对
随着日后对于 正交镜头 的多次使用,终归会熟练并理解的。
获取键盘事件
让Canvas 获取键盘事件
必须同时满足以下
-
canvas 当前获得焦点 -
<canvas \>
标签中必须添加tabIndex 属性,属性值是
-1 、0、1 都无所谓,建议设置为0
补充说明:当
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 获取键盘事件
默认
键盘上的 上下左右 方向键 均可操控改变 镜头轨道视图。
但是我们之前的代码中,经常是这样写的:
const controls = new OrbitControls(camera, canvasRef.current)
这样存在的问题是,当
最简单的解决办法就是将代码修改为:
const controls = new OrbitControls(camera, document.body)
这样键盘事件就不容易丢失。
设置画布透明度
设置画布透明度,你可能会疑惑,这有什么好讲的,直接通过
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
针对 预乘
premultiplied alpha:颜色值 预乘
这是传统
假设我们要表示一个 透明度为
假设一个颜色使用
C*A + ( 1-A ) * F
也就是说
(255,0,0)_0.6 + (1-0.6) _ (255,255,255) = (255, 255 x 0.4, 255 x 0.4) = (255,102,102)
暂时看不懂没有关系,只需记住若想让渲染器将物体渲染出半透明,除了物体本身材质配置透明度以外,还需要将渲染器中的
设置画布为背景
将
canvas {
position: fixed;
top: 0;
left: 0;
z-index: -1;
}
上述
但是实际项目中,更加建议将
这样做有几个理由:
- 使用
iframe 后,可以将canvas 、Three.js 的相关代码独立出来 - 可以多个页面都引用这个
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;
}
上述
position: fixed; 可以让iframe 位置固定z-index: -1; 可以让iframe 层级最低border: none; 可以让iframe 不显示边框pointer-events: none; 让iframe 永远不会成为鼠标事件的target ,意味着让iframe 不接受鼠标交互事件
关于更多
https://developer.mozilla.org/zh-CN/docs/Web/API/Canvas_API
至此,关于
接下来开始讲解