From edb29deb247945dc6106d8e4a94e1c13184c0a0d Mon Sep 17 00:00:00 2001 From: puxiao Date: Thu, 15 Sep 2022 09:06:53 +0800 Subject: [PATCH 01/10] Add files via upload --- ...06\347\202\271\347\254\224\350\256\260.md" | 174 ++++++++++++++++++ 1 file changed, 174 insertions(+) create mode 100644 "Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" diff --git "a/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" new file mode 100644 index 0000000..d9c7317 --- /dev/null +++ "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" @@ -0,0 +1,174 @@ +# Three.js实用知识点笔记 + +从今天开始,在本文中记录实际 Three.js 开发过程中所遇到的知识点。 + + + +
+ +**关于示例代码的一般约定说明:** + +1. 举例的时候,很多都是伪代码 +2. 为了保证代码简洁,所以在实用某些类时没有添加 `Three.` 前缀 +3. 绝大多数时候都使用箭头函数 +4. 使用 Xxx 泛指同一类型,例如 XxxCamer 泛指各类相机 + + + +
+ +#### 01、每一个Object3D对象都只能有一个父级 + +这里说的 Object3D 实际上包括所有继承于 Object3D 的子类,例如 Mesh、Camera、Group 等 + +举例说明: + +``` +const mesh = new Mesh(geometry,material) + +const sceneA = new Scene() +const sceneB = new Scene() + +sceneA.add(mesh) +sceneB.add(mesh) +``` + +由于 mesh 只能有一个父类,所以当 sceneB 也执行 .add(mesh) 后,sceneA.children 中会自动删除掉 mesh。 + + + +
+ +#### 02、克隆或复制 Mesh 不会在内存中真正复制出一份 顶点(geometry)和材质(material),它们使用的是引用,而不是复制 + +Object3D 拥有 .clone() 和 .copy() 两个方法,Mesh 继承于 Object3D,所以也拥有这两个方法。 + +在 Mesh 类中扩展了 .copy() 方法,但是针对 Mesh 内部的属性 顶点和材质,实用的是引用而不是真正内存中的复制。 + +``` +const meshB = meshA.clone() +``` + +上述代码中新复制得到的 meshB 仅仅复制了 meshA 的一些变换相关的属性,例如 matrix 等,但是对于占内存大头的 顶点和材质 这两项实用的是引用。 + +也就是说此时 meshA 和 meshB 它们共用了一份 geometry 和 material。 + +> 不用担心因为多复制了几份 mesh 而增加很多内存。 + + + +
+ +#### 03、添加场景(scene)或其他Object3D渲染之前和渲染之后的回调函数 + +场景 scene 继承于 Object3D,而 Object3D 可以配置 2 个渲染之前或之后的回调函数: + +``` +scene.onBeforeRender = () => { ...} +scnet.onAfterRende = () => { ... } +``` + +使用场景举例:假设我们希望不渲染场景上的某一类元素,那么我们可以在 renderer.render() 之前通过上面 2 个回调函数进行设置 + +``` +scene.onBeforeRender = () => { + scene.children.forEach(item => { + if(item.type === 'Points'){ + item.visible = false + } + }) +} + +scene.onAfterRender = () => { + scene.children.forEach(item => item.visible = true) +} + +renderer.render(scene,camera) +``` + + + +
+ +#### 04、通过 .layers 控制物体是否被渲染 + +在 Three.js 中 .layers 对应的是 Layers 这个类,Three.js 规定 Layers 级别的值取值范围为 0 - 32。 + +> 你可以把 layers 翻译成 “级别”,也可以称呼为 “层级” + +任何继承于 Object3D 的类,例如 相机、物体 等都具有 .layers 属性。 + +它们的 .layers 默认级别都为 0。 + +不能通过直接给 .layers 赋值的方式修改级别,而是应该通过 .set(value) 这种形式。 + +``` +mymesh.layers.set(1) +camera.layers.set(1) +``` + + + +
+ +对于相机而言,它只能渲染出同一级别的物体元素。 + +``` +const meshA = new Mesh(...) +//meshA.layers.set(0) //默认就是 0 + +const meshB = new Mesh(...) +meshB.layers.set(1) + +const scene = new Scene() +scene.add(meshA) +scene.add(meshB) + +const cameraA = new XxxCamera() +//cameraA.layers.set(0) //默认就是 0 + +const cameraB = new XxxCamera() +cameraB.layers.set(1) +``` + +在上面代码中: + +1. 我们按照默认的形式添加了 meshA、cameraA,它们默认层级为 0 +2. 手动修改了 meshB、cameraB 的 .layers 层级为 1 + +
+ +那么当执行下面的代码: + +``` +renderer.render(scene, cameraA) +renderer.render(scene. cameraB) +``` + +1. cameraA 只会渲染出场景中同一级别的 meshA +2. cameraB 只会渲染出场景中同一级别的 meshB + + + +
+ +也可以选择随时修改 meshA 的 .layers 值,这样 cameraB 就可以渲染到它了。 + +``` +meshA.layers.set(1) +renderer.render(secen, cameraB) +``` + + + +
+ +换句话说,假设我们希望控制是否渲染场景中某些元素,那么有 2 种途径: + +1. 设置其 .visible 的值来决定是否渲染 +2. 设置其 .layers 的值来决定只被同一层级的相机渲染 + + + +
+ From 40e7fbc6752ef4274ed64c2a6d426761662f4bdf Mon Sep 17 00:00:00 2001 From: puxiao Date: Thu, 15 Sep 2022 13:11:51 +0800 Subject: [PATCH 02/10] =?UTF-8?q?Update=20Three.js=E5=AE=9E=E7=94=A8?= =?UTF-8?q?=E7=9F=A5=E8=AF=86=E7=82=B9=E7=AC=94=E8=AE=B0.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...06\347\202\271\347\254\224\350\256\260.md" | 38 +++++++++++++++++++ 1 file changed, 38 insertions(+) diff --git "a/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" index d9c7317..09c6315 100644 --- "a/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" +++ "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" @@ -170,5 +170,43 @@ renderer.render(secen, cameraB) +
+ +#### 05、手工修改Object3D实例的.matrix时切记要设置 .matrixAutoUpdate=false + +默认情况下 Object3D 实例的 .matrixAutoUpdate 的值为 true,也就是说当通过 .applyMatrix4()、.applyQuaternion() 等修改实例的变换时,默认会自动更新其他所有相关属性值,例如 position、quaternion、scale、rotation。 + +但是,如果直接通过修改 Object3D 实例的 .matrix 值时,生效的前提是: + +1. 先把 .matrixAutoUpdate 设置为 false +2. 不去调用 .updateMatrix() + +``` +const mesh = new Mesh(...) + +mesh.matrixAutoUpdate = false +mesh.matrix.copy(otherMatrix) +``` + + + +
+ +但是上面的代码存在另外一个问题:尽管 .matrix 值更新了,可是 mesh 的其他属性值 例如 .position,.quaternion,scale,rotation 却没有自动更新。 + +解决方式很简单,可以通过 Matrix 的 .decompose() 方法优雅更新它们。 + +举例说明:假设现在有 meshA、meshB 两个对象,需要将 meshB 的各种变换属性值设置成和 meshA 完全相同 + +``` +meshB.matrixAutoUpdate = false +meshB.matrix.copy(meshA.matrix) +meshB.matrix.decompose(meshB.position, meshB.quaternion, meshB.scale) +``` + +> 当修改 meshB.quaternion 值后会自动修改 meshB.rotation 的值 + + +
From aa8647bfe4cfec0e578d321d863b01a29b34f2d8 Mon Sep 17 00:00:00 2001 From: puxiao Date: Fri, 16 Sep 2022 08:59:47 +0800 Subject: [PATCH 03/10] =?UTF-8?q?Update=20Three.js=E5=AE=9E=E7=94=A8?= =?UTF-8?q?=E7=9F=A5=E8=AF=86=E7=82=B9=E7=AC=94=E8=AE=B0.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...\257\206\347\202\271\347\254\224\350\256\260.md" | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git "a/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" index 09c6315..869be56 100644 --- "a/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" +++ "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" @@ -208,5 +208,18 @@ meshB.matrix.decompose(meshB.position, meshB.quaternion, meshB.scale) +
+ +#### 06、绘制三角形的顶点顺序决定了该三角形是正面(顺时针)还是反面(逆时针) + +一个三角形有 3 个顶点,假定为 a、b、c,那么: + +1. 假定 a b c 连接顺序为 逆时针,那么最终形成的三角形为 正面 +2. 假定 a b c 连接顺序为 顺时针,那么最终形成的三角形为 反面(背面) + +另外一种判定形式是:右手握住沿着两个顶点添加顺序的连接线,此时大拇指指示方向即为正面 + + +
From 2d0b38095b161b690fb96fc22ac6b87801c1949c Mon Sep 17 00:00:00 2001 From: puxiao Date: Sat, 17 Sep 2022 08:46:34 +0800 Subject: [PATCH 04/10] =?UTF-8?q?Update=20Three.js=E5=AE=9E=E7=94=A8?= =?UTF-8?q?=E7=9F=A5=E8=AF=86=E7=82=B9=E7=AC=94=E8=AE=B0.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...06\347\202\271\347\254\224\350\256\260.md" | 39 +++++++++++++++++++ 1 file changed, 39 insertions(+) diff --git "a/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" index 869be56..2658cc2 100644 --- "a/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" +++ "b/Three.js\345\256\236\347\224\250\347\237\245\350\257\206\347\202\271\347\254\224\350\256\260.md" @@ -12,6 +12,7 @@ 2. 为了保证代码简洁,所以在实用某些类时没有添加 `Three.` 前缀 3. 绝大多数时候都使用箭头函数 4. 使用 Xxx 泛指同一类型,例如 XxxCamer 泛指各类相机 +5. 由于我们讲解的是 Three.js,所以全部使用的是 右手坐标系 @@ -223,3 +224,41 @@ meshB.matrix.decompose(meshB.position, meshB.quaternion, meshB.scale)
+#### 07、保持外观和位置的前提下,将立方体的顶点坐标 "归一化" + +这里说的立方体是指基于 BoxGeometry 而创建的立方体。 + +这里说的 顶点坐标 “归一化” 是指将立方体顶点坐标修改成 1x1x1 规格的立方体顶点坐标。 + +这里说的 保持外观和位置 是指通过修改其 变换矩阵 .matrix 来实现。 + +实现思路: + +1. 凡是基于 BoxGeometry 的立方体,其顶点坐标信息都是统一规范的,尽管其值可能不同 +2. 所以我们就根据其值来确定这个立方体与 1x1x1 立方体的 宽、高、深 比例 +3. 将这个缩放比例应用到立方体本身的矩阵中即可 +4. 同时将这个立方体的顶点信息修改成 1x1x1 立方体的顶点信息 + +``` +const boxGeometryNormalize = (box) => { + const originX = box.geometry.attributes.position.array[0] + const originY = box.geometry.attributes.position.array[1] + const originZ = box.geometry.attributes.position.array[2] + + const scaleX = originX / 0.5 + const scaleY = originY / 0.5 + const scaleZ = originZ / 0.5 + + box.geometry = new BoxGeometry() + box.matrixAutoUpdate = false + box.matrix.makeScale(scaleX, scaleY, scaleZ) + box.matrix.decompose(box.position, box.quaternion, box.scale) +} +``` + + + +
+ + + From c050d1137737ee804de6f2d3b17c977732907eab Mon Sep 17 00:00:00 2001 From: puxiao Date: Sat, 17 Sep 2022 10:34:50 +0800 Subject: [PATCH 05/10] =?UTF-8?q?Update=2012=20Three.js=E5=9F=BA=E7=A1=80?= =?UTF-8?q?=E4=B9=8B=E9=95=9C=E5=A4=B4.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git "a/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" "b/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" index e1a28f7..4f625df 100644 --- "a/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" +++ "b/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" @@ -518,7 +518,7 @@ OrthographicCamera( left : Number, right : Number, top : Number, bottom : Number **从实际的角度来看,一定要注意以下几点:** -1. left 的值不能大于 right,同理 top 的值不能大于 bottom +1. left 的值不能大于 right,同理 bottom 的值不能大于 top。 如果没有按照这个约定,例如 bottom 大于 top 相当于颠倒了相机。 2. near 设置越小,投影的映像越大 3. left 与 right 之间的距离、top 与 bottom 之间的距离的比例一定要和 canvas 比例相同,否则会导致投影的物体形状变形 From 53a7a813fc5cc2ea0e270d57cdcb2e30477ec95a Mon Sep 17 00:00:00 2001 From: xsstomy <674371968@qq.com> Date: Wed, 15 Feb 2023 16:25:17 +0800 Subject: [PATCH 06/10] =?UTF-8?q?Update=2004=20=E6=B7=BB=E5=8A=A0=E4=B8=80?= =?UTF-8?q?=E4=BA=9B=E8=87=AA=E9=80=82=E5=BA=94.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 拼写错误 --- ...344\272\233\350\207\252\351\200\202\345\272\224.md" | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git "a/04 \346\267\273\345\212\240\344\270\200\344\272\233\350\207\252\351\200\202\345\272\224.md" "b/04 \346\267\273\345\212\240\344\270\200\344\272\233\350\207\252\351\200\202\345\272\224.md" index 9038822..7c768eb 100644 --- "a/04 \346\267\273\345\212\240\344\270\200\344\272\233\350\207\252\351\200\202\345\272\224.md" +++ "b/04 \346\267\273\345\212\240\344\270\200\344\272\233\350\207\252\351\200\202\345\272\224.md" @@ -95,7 +95,7 @@ const HelloThreejs: React.FC = () => { #### 目前存在的问题: -可以观察到 canvase 是被硬生生由原本的 高150、像素 宽 300 像素给硬生生拉伸成 100%。 +可以观察到 canvas 是被硬生生由原本的 高150、像素 宽 300 像素给硬生生拉伸成 100%。 所以立方体出现了 扭曲、模糊、锯齿。 @@ -107,14 +107,14 @@ const HelloThreejs: React.FC = () => { #### 第4处修改: -修改 src/components/hello-threejs/index.stx 中 render 函数的代码,让镜头宽高比跟随着 canvase 宽高比,确保立方体不变形。 +修改 src/components/hello-threejs/index.stx 中 render 函数的代码,让镜头宽高比跟随着 canvas 宽高比,确保立方体不变形。 ```diff ... const render = (time: number) => { time = time * 0.001 -+ const canvas = renderer.domElement //获取 canvase ++ const canvas = renderer.domElement //获取 canvas + camera.aspect = canvas.clientWidth / canvas.clientHeight //设置镜头宽高比 + camera.updateProjectionMatrix() //通知镜头更新视椎(视野) @@ -138,7 +138,7 @@ const HelloThreejs: React.FC = () => { const render = (time: number) => { time = time * 0.001 - const canvas = renderer.domElement //获取 canvase + const canvas = renderer.domElement //获取 canvas camera.aspect = canvas.clientWidth / canvas.clientHeight //设置镜头宽高比 camera.updateProjectionMatrix() //通知镜头更新视椎(视野) @@ -531,4 +531,4 @@ https://github.com/mdn/translated-content/pull/817 那么接下来,会系统学习一下 Three.js 的一些基础理论。 -**大楼究竟能改多高,取决于地基有多深,加油!** \ No newline at end of file +**大楼究竟能改多高,取决于地基有多深,加油!** From 6cdca5fb8c31133ec1a72b553f6afddf25281aea Mon Sep 17 00:00:00 2001 From: xsstomy <674371968@qq.com> Date: Thu, 23 Feb 2023 10:47:11 +0800 Subject: [PATCH 07/10] =?UTF-8?q?Update=2012=20Three.js=E5=9F=BA=E7=A1=80?= =?UTF-8?q?=E4=B9=8B=E9=95=9C=E5=A4=B4.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git "a/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" "b/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" index 4f625df..2c94190 100644 --- "a/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" +++ "b/12 Three.js\345\237\272\347\241\200\344\271\213\351\225\234\345\244\264.md" @@ -534,7 +534,7 @@ OrthographicCamera( left : Number, right : Number, top : Number, bottom : Number -### 假设我们在 HelloCamer 示例中使用 OrthographicCamera +### 假设我们在 HelloCamera 示例中使用 OrthographicCamera **需要修改的地方为:** From 4e8c6f56806cad27d91cfeabab5c6af242fc0679 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E6=9F=A0=E9=85=B8?= Date: Wed, 13 Dec 2023 16:25:20 +0800 Subject: [PATCH 08/10] =?UTF-8?q?Update=2006=20=E5=9B=BE=E5=85=83=E7=BB=83?= =?UTF-8?q?=E4=B9=A0=E7=A4=BA=E4=BE=8B.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- ...\205\203\347\273\203\344\271\240\347\244\272\344\276\213.md" | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git "a/06 \345\233\276\345\205\203\347\273\203\344\271\240\347\244\272\344\276\213.md" "b/06 \345\233\276\345\205\203\347\273\203\344\271\240\347\244\272\344\276\213.md" index 5f29df9..a9bc94a 100644 --- "a/06 \345\233\276\345\205\203\347\273\203\344\271\240\347\244\272\344\276\213.md" +++ "b/06 \345\233\276\345\205\203\347\273\203\344\271\240\347\244\272\344\276\213.md" @@ -112,7 +112,7 @@ const HelloPrimitives = () => { scene.add(light0) const light1 = new Three.DirectionalLight(0xFFFFFF, 1) - light0.position.set(1, -2, -4) + light1.position.set(1, -2, -4) scene.add(light1) //获得各个 solid 类型的图元实例,并添加到 solidPrimitivesArr 中 From b74c77c3e9e86bc53bb6f1b49819e7a2db2e67a9 Mon Sep 17 00:00:00 2001 From: puxiao Date: Fri, 9 Feb 2024 18:47:36 +0800 Subject: [PATCH 09/10] Update README.md --- README.md | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 3aeafc1..39ed2db 100644 --- a/README.md +++ b/README.md @@ -2,6 +2,13 @@ 从今天 2020年11月27日 开始学习和探索 Three.js 。 + +
+

+wechat.jpg +

+ +
> 以下内容更新于 2021.04.16 @@ -417,4 +424,3 @@ https://www.bilibili.com/video/BV1X7411F744
**大家都是 Three.js 小白新手,一起加油!** - From 6d89d9ce43db564046af487485121b9dce3919ba Mon Sep 17 00:00:00 2001 From: puxiao Date: Tue, 5 Mar 2024 01:09:58 +0800 Subject: [PATCH 10/10] =?UTF-8?q?Update=2003=20=E7=BC=96=E5=86=99HelloThre?= =?UTF-8?q?ejs.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- "03 \347\274\226\345\206\231HelloThreejs.md" | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git "a/03 \347\274\226\345\206\231HelloThreejs.md" "b/03 \347\274\226\345\206\231HelloThreejs.md" index 9992898..3a5fd8d 100644 --- "a/03 \347\274\226\345\206\231HelloThreejs.md" +++ "b/03 \347\274\226\345\206\231HelloThreejs.md" @@ -65,7 +65,7 @@ const scene = new Scene() 1. 渲染器 > 本文示例采用的渲染器是 WebGLRenderer 2. 透视镜头 > 本文示例采用的是 PerspectiveCamera -3. 场景 > Scen +3. 场景 > Scene **场景可见元素:** @@ -76,7 +76,7 @@ const scene = new Scene() **补充说明:** -你应该发现,除了 场景(Scen)、网格(Mesh) 之外,其他的元素我都注明 “本文示例采用的是...”。 +你应该发现,除了 场景(Scene)、网格(Mesh) 之外,其他的元素我都注明 “本文示例采用的是...”。 因为无论渲染器,还是几何体,以及其他元素,Three.js 都内置了非常多不同种类的元素构造函数,这个会在以后学习中逐渐详细说明举例。