当前位置:   article > 正文

Three.js机器人与星系动态场景(二):强化三维空间认识_three.js模型点击显示信息

three.js模型点击显示信息

在上篇博客中介绍了如何快速利用react搭建three.js平台,并实现3D模型的可视化。本文将在上一篇的基础上强化坐标系的概念。引入AxesHelper辅助工具及文本绘制工具,带你快速理解camer、坐标系、position、可视区域。

Three.js机器人与星系动态场景:实现3D渲染与交互式控制-CSDN博客

 AxesHelper辅助坐标系

three.js提供了 AxesHelper方法,在3D场景中建立一个,x、y、z两两垂直的坐标系。由于在网页中通常用z-index表示高度或者层级,在3D场景中z表示离屏幕的距离,z越大,物体离实现越近。所以在3D中的坐标系默认是y轴朝上。

辅助坐标系便于开发者或用户在视觉上识别和定位场景中的物体。坐标轴辅助器通常在开发阶段用于调试,但在最终的产品中也可以保留,以帮助用户理解3D空间

使用方法

  1. const axesHelper = new THREE.AxesHelper(150);
  2. scene.add(axesHelper);

AxesHelper是Three.js库中的一个类,用于创建一个可视化的坐标轴(X轴、Y轴和Z轴)。括号中的150是一个参数,表示坐标轴的长度为150个单位。 通过THREE的AxesHelper方法创建,并通过scene添加到场景中。

 通过这个方法可以看到在3D中生成了坐标轴辅助线。

three.js坐标轴颜色R绿GB分别对应坐标系的xyz

绘制文本并让文字始终朝向用户 

在旋转过程中坐标系可能会混乱,虽然你强迫自己记住xyz对应红绿蓝,但是用着用着就蒙了。可以借助文字辅助理解。然而在threeJS中文本也是当物体绘制出来。不是几行代码就能实现的。这里文字需要通过load进行加载,并且文字有自己的position,在循环动画时添加文本的lookAt为camera的position,让文本始终看着相机,即可实现文本朝向用户。

 创建文本

import { TextGeometry } from "three/examples/jsm/geometries/TextGeometry";
  1. //创建文本
  2. function createText(content: string, font: any) {
  3. const textGeometry = new TextGeometry(content, {
  4. font: font,
  5. size: 1,
  6. height: 0.1,
  7. curveSegments: 1,
  8. });
  9. textGeometry.center();
  10. const textMaterial = new THREE.MeshPhongMaterial({ color: 0xffffff, flatShading: true }); // front
  11. const mesh = new THREE.Mesh(textGeometry, textMaterial);
  12. return mesh;
  13. }

 解析字体的json文件

  1. //坐标系添加文字
  2. const loader = new FontLoader();
  3. let meshX = new THREE.Mesh();
  4. let meshY = new THREE.Mesh();
  5. let meshZ = new THREE.Mesh();
  6. loader.load("fonts/optimer_regular.typeface.json", function (font) {
  7. meshX = createText("X", font);
  8. meshY = createText("Y", font);
  9. meshZ = createText("Z", font);
  10. meshX.position.x = 12;
  11. meshY.position.y = 12;
  12. meshZ.position.z = 12;
  13. scene.add(meshX);
  14. scene.add(meshY);
  15. scene.add(meshZ);
  16. });

 这个json后缀的文件是用来解析字体的,load的第一个参数是json的url。json数据从哪获取?

打卡node_modules或者你下的threejs源码,找到examples/fonts,将其复制到public文件夹。就可以引入了

 

 

在学习threeJS过程中,我也是不断参考example示例,找所需的资源 

 相机camera

camera相机

 相机在3D场景的作用可以想象成是你手中的一个真实的相机或者你的眼睛。在现实生活中,你站在某个地方,你的位置就是相机的位置。比如说,你站在一个公园的角落,从这个位置你可以看到公园里的不同景物。在Three.js中,camera代表了观察者(可以是你的眼睛或者一个虚拟的摄像头)在3D空间中的位置和视角。

 PerspectiveCamera 相机

three.js 里有几种不同的相机,在这里,我们使用的是 PerspectiveCamera(透视摄像机)这个相机类型最接近我们现实生活中观察世界的方式,因为它使用透视投影来渲染场景。透视投影的特点是近大远小,这使得远处的物体看起来比实际小,而近处的物体则显得更大。

关键参数

当你创建一个 PerspectiveCamera 时,通常需要设置以下几个参数:

  1. 视野(Field of View, FOV):这是相机视野的角度,通常以度数表示。较大的 FOV 会使场景看起来更宽广,而较小的 FOV 则会使场景看起来更狭窄。

  2. 长宽比(Aspect Ratio):这是相机视口的宽度与高度的比率。通常设置为渲染窗口的宽度除以高度。

  3. 近裁剪面(Near Clipping Plane):这是相机能够看到的最近的距离。任何比这个距离更近的物体都不会被渲染。

  4. 远裁剪面(Far Clipping Plane):这是相机能够看到的最远的距离。任何比这个距离更远的物体也不会被渲染。

   

想象一下你正在用眼睛看世界。PerspectiveCamera 就像你的眼睛,它有一个视野范围(FOV),决定了你能看到多宽或多窄的场景。长宽比(Aspect Ratio)决定了你看到的画面是宽屏还是窄屏。近裁剪面和远裁剪面则决定了你能看到的最近和最远的物体。 

 在three.js中通常通过下面的方式创建透视相机,通常设置宽高比就用视口可视区域的宽高比。

  1. // 创建一个Three.js相机,包括透视投影、宽高比、近裁剪面和远裁剪面
  2. const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);

 position位置

position就是在3D场景中的物体的坐标信息,在Three.js中,设置camera.position就是确定你的“观察点”在哪里

 可以对具有postion属性的物体通过set方法设置x,y,z坐标信息

  camera.position.set(15, 12, 8);

 也可以对Three.js的Object3D对象通过position.x和position.y单独设置坐标轴的值

lookAt方法

lookAt方法就像是你在决定你的相机或者你的眼睛要聚焦在哪个点上。当你调整相机对准某个特定对象时,你实际上是在使用lookAt方法。在现实生活中,即使你站在一个固定的位置,你的头可以转向不同的方向,去看向不同的物体。

 lookAt看向的是3D空间中的具体点,相机的位置和lookAt看向的点覆盖的区域就是网页渲染初始时的视野

举例:继续上面的操场例子,即使你站在同一个角落,你可以选择看向操场的中心点,也可以看向操场上的某个运动员或者某个特定的物体。当你看向某个点时,你的注意力就集中在了那个方向上,就像是在Three.js中调用lookAt方法,让相机镜头指向那个点。

在本文中相机的位置position和lookAt坐标系

  1. camera.position.set(15, 12, 8);
  2. camera.lookAt(0, 0, 0);

可以看到,你的眼睛是能够看到x、y、z三个坐标的正向,并且视线正对坐标原点。也就是第一个机器人。 

  1. robot2.position.x = 6;
  2. robot2.position.z = 6;

camera位置和lookAt对可视物体的影响 

1.物体放在camera后面

 示例的第一个机器人位于坐标原点,第二个机器人x轴和z轴分别偏移了6个单位。因为camera的位置是x轴偏移15,y轴偏移12,因此能够看到两个机器人。如果我将第二个机器人的x和y都大于camera呢?比如如下的设置

  1. robot2.position.x = 20;
  2. robot2.position.z = 20;

 可以看到默认页面加载是看不到第二个机器人的。但是相机位置和看向点不变,我们旋转3D坐标系,可以看到第二个机器人。说明此时机器人“站在了你的后面”,因为你没看向它,但它存在吗?当然存在。

 2.切换camera的lookAt

机器人2的位置信息 

  1. robot2.position.x = 20;
  2. robot2.position.z = 20;

切换相机的视角使其看到机器人2

  1. camera.position.set(15, 12, 8);
  2. camera.lookAt(20, 0, 20);

 可以看到页面初始时只能看到机器人2,坐标原点在后面,所以看不到原点上的机器人。但是旋转坐标系能够看到。

 OrbitControls旋转坐标系

前面提了很多,旋转坐标系。默认坐标系是不能旋转的,在空间中位置信息确认了就是相对静止的。那我们怎么可以拖动屏幕实现立体观察的效果呢?

在 Three.js 中,如果你希望通过拖动屏幕来旋转场景中的坐标轴,通常需要使用一些交互控制器,例如 OrbitControlsOrbitControls 是一个控制器,它允许用户通过拖动、滚动和按键来控制相机的位置和方向。

 使用OrbitControls

import { OrbitControls } from "three/examples/jsm/controls/OrbitControls";

可以通过给controls对象添加一个事件监听,当orbitControls改变了,调用renderer重新渲染scene和camera

  1. // 设置相机控件轨道控制器OrbitControls
  2. const controls = new OrbitControls(camera, renderer.domElement);
  3. // 如果OrbitControls改变了相机参数,重新调用渲染器渲染三维场景
  4. controls.addEventListener('change', function () {
  5. renderer.render(scene, camera); //执行渲染操作
  6. });//监听鼠标、键盘事件

但在我们的代码中并没有显示的调用change的监听,而是在设置循环动画的时候通过renderer.render方法重新渲染每一帧 

  1. const update = () => {
  2. requestAnimationFrame(update);
  3. camera.lookAt(20, 0, 20);
  4. robot.rotation.y -= 0.005; //机器人旋转
  5. robot2.rotation.y -= 0.005;
  6. // 粒子旋转
  7. starts.rotation.y -= 0.001;
  8. starts.rotation.z += 0.001;
  9. starts.rotation.x += 0.001;
  10. renderer.render(scene, camera);
  11. };
  12. update(); //自动更新

虽然在 update 函数中没有显式调用 controls.update(),但是 OrbitControls 在内部已经通过事件监听器在每次交互时更新了相机的位置和方向。当你移动鼠标或触摸屏幕时,这些交互事件会被监听器捕获,OrbitControls 会相应地调整相机,然后 renderer.render(scene, camera) 会使用新的相机状态来渲染场景。

简而言之,OrbitControls 的设计允许它不需要显式的 change 事件监听就能工作。它会在用户交互时自动更新相机,然后在你的渲染函数中,通过 renderer.render(scene, camera) 来反映这些更新。

如果你的场景中确实需要响应用户交互以外的事件来触发渲染,或者你想要在特定条件下禁用某些控制行为,那么监听 change 事件会很有用。但在大多数情况下,对于基本的交互控制,OrbitControls 已经提供了所需的功能。

OrbitControls本质 

OrbitControls本质上就是改变相机的参数,比如相机的位置属性,改变相机位置也可以改变相机拍照场景中模型的角度,实现模型的360度旋转预览效果  

 完整版代码

  1. import { useEffect, useRef } from "react";
  2. import * as THREE from "three";
  3. import { OrbitControls } from "three/examples/jsm/controls/OrbitControls";
  4. import { FontLoader } from "three/examples/jsm/loaders/FontLoader";
  5. import { TextGeometry } from "three/examples/jsm/geometries/TextGeometry";
  6. //机器人脑袋
  7. function createHead() {
  8. //SphereGeometry创建球形几何体
  9. const head = new THREE.SphereGeometry(4, 32, 16, 0, Math.PI * 2, 0, Math.PI * 0.5);
  10. const headMaterial = new THREE.MeshStandardMaterial({
  11. color: 0x43b988,
  12. roughness: 0.5,
  13. metalness: 1.0,
  14. });
  15. const headMesh = new THREE.Mesh(head, headMaterial);
  16. return headMesh;
  17. }
  18. //触角
  19. function generateHorn(y: number, z: number, angle: number) {
  20. //触角 CapsuleGeometry 创建胶囊形状的几何体。胶囊形状可以看作是一个圆柱体两端加上半球体
  21. const line = new THREE.CapsuleGeometry(0.1, 2);
  22. const lineMaterial = new THREE.MeshStandardMaterial({
  23. color: 0x43b988,
  24. roughness: 0.5,
  25. metalness: 1.0,
  26. });
  27. const lineMesh = new THREE.Mesh(line, lineMaterial);
  28. lineMesh.position.y = y;
  29. lineMesh.position.z = z;
  30. lineMesh.rotation.x = angle;
  31. return lineMesh;
  32. }
  33. //机器人眼睛
  34. function generateEye(x: number, y: number, z: number) {
  35. //SphereGeometry创建球形几何体
  36. const eye = new THREE.SphereGeometry(0.5, 32, 16, 0, Math.PI * 2, 0, Math.PI * 2);
  37. const eyeMaterial = new THREE.MeshStandardMaterial({
  38. color: 0x212121,
  39. roughness: 0.5,
  40. metalness: 1.0,
  41. });
  42. const eyeMesh = new THREE.Mesh(eye, eyeMaterial);
  43. eyeMesh.position.x = x;
  44. eyeMesh.position.y = y;
  45. eyeMesh.position.z = z;
  46. return eyeMesh;
  47. }
  48. //机器人身体
  49. function generateBody() {
  50. //CylinderGeometry第一个参数是上部分圆的半径,第二个参数是下部分圆的半径,第三个参数是高度,材质使用的跟腿一样
  51. const body = new THREE.CylinderGeometry(4, 4, 6);
  52. const bodyMaterial = new THREE.MeshStandardMaterial({
  53. color: 0x43b988,
  54. roughness: 0.5,
  55. metalness: 1.0,
  56. });
  57. const bodyMesh = new THREE.Mesh(body, bodyMaterial);
  58. return bodyMesh;
  59. }
  60. //胳膊、腿
  61. function generateLegs(y: number, z: number) {
  62. const leg1 = new THREE.CapsuleGeometry(1, 4);
  63. const legMaterial1 = new THREE.MeshStandardMaterial({
  64. color: 0x43b988,
  65. roughness: 0.5,
  66. metalness: 1.0,
  67. });
  68. const leg1Mesh = new THREE.Mesh(leg1, legMaterial1);
  69. leg1Mesh.position.y = y;
  70. leg1Mesh.position.z = z;
  71. return leg1Mesh;
  72. }
  73. //创建机器人
  74. function generateRobot() {
  75. // 创建一个Three.js对象,用于存放机器人
  76. const robot = new THREE.Object3D();
  77. const headMesh = createHead();
  78. headMesh.position.y = 6.5;
  79. robot.add(headMesh);
  80. //眼睛
  81. const leftEye = generateEye(3, 8, -2);
  82. const rightEye = generateEye(3, 8, 2);
  83. robot.add(leftEye);
  84. robot.add(rightEye);
  85. const leftHorn = generateHorn(11, -1, (-Math.PI * 30) / 180);
  86. const rightHorn = generateHorn(11, 1, (Math.PI * 30) / 180);
  87. robot.add(leftHorn);
  88. robot.add(rightHorn);
  89. const body = generateBody();
  90. body.position.y = 4;
  91. robot.add(body);
  92. // 生成机器人左腿
  93. robot.add(generateLegs(0, -2));
  94. // 生成机器人右腿
  95. robot.add(generateLegs(0, 2));
  96. //胳膊
  97. robot.add(generateLegs(3, 5));
  98. robot.add(generateLegs(3, -5));
  99. //物体缩放
  100. robot.scale.x = 0.3;
  101. robot.scale.y = 0.3;
  102. robot.scale.z = 0.3;
  103. return robot;
  104. }
  105. //创建粒子星星
  106. function generateStarts(num: number) {
  107. //制作粒子特效
  108. const starts = new THREE.Object3D();
  109. const obj = new THREE.SphereGeometry(0.2, 3, 3);
  110. const material = new THREE.MeshStandardMaterial({
  111. color: 0x43b988,
  112. roughness: 0.5,
  113. metalness: 5,
  114. });
  115. const mesh = new THREE.Mesh(obj, material);
  116. for (let i = 0; i < num; i++) {
  117. const target = new THREE.Mesh();
  118. target.copy(mesh);
  119. target.position.x = Math.floor(Math.random() * 18 + Math.floor(Math.random() * -18));
  120. target.position.y = Math.floor(Math.random() * 18 + Math.floor(Math.random() * -18));
  121. target.position.z = Math.floor(Math.random() * 18 + Math.floor(Math.random() * -18));
  122. starts.add(target);
  123. }
  124. return starts;
  125. }
  126. //创建文本
  127. function createText(content: string, font: any) {
  128. const textGeometry = new TextGeometry(content, {
  129. font: font,
  130. size: 1,
  131. height: 0.1,
  132. curveSegments: 1,
  133. });
  134. textGeometry.center();
  135. const textMaterial = new THREE.MeshPhongMaterial({ color: 0xffffff, flatShading: true }); // front
  136. const mesh = new THREE.Mesh(textGeometry, textMaterial);
  137. return mesh;
  138. }
  139. /**
  140. * 创建一个Three.js场景,包括相机和渲染器
  141. */
  142. function Robot() {
  143. // 创建一个div容器,用于存放渲染的Three.js场景
  144. const containerRef = useRef<HTMLDivElement>(null);
  145. const scene = new THREE.Scene();
  146. // 创建一个Three.js相机,包括透视投影、宽高比、近裁剪面和远裁剪面
  147. const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
  148. camera.position.set(15, 12, 8);
  149. camera.lookAt(0, 0, 0);
  150. // 创建一个Three.js渲染器,包括抗锯齿
  151. const renderer = new THREE.WebGLRenderer({ antialias: true });
  152. renderer.setSize(window.innerWidth, window.innerHeight);
  153. //添加坐标系
  154. const axisHelper = new THREE.AxesHelper(150);
  155. scene.add(axisHelper);
  156. //坐标系添加文字
  157. const loader = new FontLoader();
  158. let meshX = new THREE.Mesh();
  159. let meshY = new THREE.Mesh();
  160. let meshZ = new THREE.Mesh();
  161. loader.load("fonts/optimer_regular.typeface.json", function (font) {
  162. meshX = createText("X", font);
  163. meshY = createText("Y", font);
  164. meshZ = createText("Z", font);
  165. meshX.position.x = 12;
  166. meshY.position.y = 12;
  167. meshZ.position.z = 12;
  168. scene.add(meshX);
  169. scene.add(meshY);
  170. scene.add(meshZ);
  171. });
  172. const robot = generateRobot();
  173. const robot2 = generateRobot();
  174. robot2.position.x = 6;
  175. robot2.position.z = 6;
  176. // 将机器人身体添加到场景中
  177. scene.add(robot);
  178. scene.add(robot2);
  179. // 创建一个Three.js方向光,包括颜色、强度
  180. const straightLight = new THREE.DirectionalLight(0xffffff, 5);
  181. // 设置方向光的位置
  182. straightLight.position.set(20, 20, 20);
  183. // 将方向光添加到场景中
  184. scene.add(straightLight);
  185. const starts = generateStarts(200);
  186. scene.add(starts);
  187. //轨道控制器
  188. const controls = new OrbitControls(camera, renderer.domElement);
  189. controls.update();
  190. const animate = () => {
  191. requestAnimationFrame(animate);
  192. robot.rotation.y -= 0.005; //机器人旋转
  193. robot2.rotation.y -= 0.005;
  194. // 粒子旋转
  195. starts.rotation.y -= 0.001;
  196. starts.rotation.z += 0.001;
  197. starts.rotation.x += 0.001;
  198. //
  199. meshX.lookAt(camera.position);
  200. meshY.lookAt(camera.position);
  201. meshZ.lookAt(camera.position);
  202. renderer.render(scene, camera);
  203. };
  204. animate(); //添加动画
  205. // 监听组件挂载和卸载
  206. useEffect(() => {
  207. // 如果div存在,将渲染器dom元素添加到div中
  208. if (containerRef.current) {
  209. containerRef.current.appendChild(renderer.domElement);
  210. // 渲染场景
  211. renderer.render(scene, camera);
  212. }
  213. }, [containerRef]);
  214. // 返回div容器,用于存放渲染的Three.js场景
  215. return <div ref={containerRef} style={{ width: "100vw", height: "100vh" }}></div>;
  216. }
  217. // 导出Robot组件
  218. export default Robot;

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/笔触狂放9/article/detail/801435
推荐阅读
相关标签
  

闽ICP备14008679号