程序员社区

三维世界中的坐标系

上篇文章中介绍了threejs中几个基本概念,例如场景、相机、渲染器以及组件等,并通过一个简单的案例向小伙伴展示了这些东西的用法,本文来看看threejs中的坐标体系。

本文是threejs系列的第二篇,阅读前面的文章有助于更好的理解本文:


1.一个简单的案例,理解threejs中几个基本概念


坐标体系

首先,threejs中坐标体系是右手坐标系,如下图:

三维世界中的坐标系插图

在此基础上,坐标体系分为世界坐标和本地坐标,相机默认位于世界坐标体系的(0,0,0)点,本地坐标则是一个组件内部的坐标。如下图,每个组件内部都会有一个坐标体系,这个就是本地坐标:

三维世界中的坐标系插图1

默认位置

按理说,场景是不需要坐标这个概念的,其他的组件和相机是有坐标的,在上文的案例中,读者可以在浏览器控制台打印出所有的坐标:

三维世界中的坐标系插图2

可以看到,相机的坐标是(0,0,5),其他的坐标则都是(0,0,0),相机默认坐标也是(0,0,0),只是由于我们在代码中配置了z轴坐标为5,不知读者是否还记得上文中如下一行代码:

  
  
  
  1. camera.position.z = 5;

经过上面的分析,我们直到,相机和组件默认都是在世界坐标的原点。

案例演示

还是上文的案例,接下来,我再添加一个坐标系到场景中,如下:

  
  
  
  1. var scene = new THREE.Scene();

  2. var camera = new THREE.PerspectiveCamera(60, window.innerWidth / window.innerHeight, 1, 1000);

  3. var render = new THREE.WebGLRenderer();

  4. render.setClearColor("#FFFFFF");

  5. render.setSize(window.innerWidth, window.innerHeight);

  6. document.body.appendChild(render.domElement);

  7. var geometry = new THREE.CubeGeometry(1,1,1);

  8. var material = new THREE.MeshBasicMaterial({color: 0xFFB6C1});

  9. var cube = new THREE.Mesh(geometry, material);

  10. scene.add(cube);

  11. var axesHelper = new THREE.AxesHelper(5);

  12. scene.add(axesHelper);

  13. camera.position.z = 5;

  14. function showCube() {

  15.    requestAnimationFrame(showCube)

  16.    cube.rotation.x += 0.01;

  17.    cube.rotation.y += 0.01;

  18.    cube.rotation.z += 0.01;

  19.    render.render(scene, camera);

  20. }

  21. showCube()

和上文相比,这里主要是多了AxesHelper,这是一个坐标系,我向场景中添加了这个坐标系:

三维世界中的坐标系插图3

不过读者看到,这里红色是x轴,绿色是y轴,好像没有看到z轴?这是因为z轴垂直于屏幕,而相机目前的位置是(0,0,5),因此看不到z轴,将相机在x轴方向上移动1个单位,即添加如下代码:

  
  
  
  1. camera.position.x = 1;

此时页面展示结果如下:

三维世界中的坐标系插图4

当然这样看起来三维的效果还是不太明显,那么可以将相机向上太高一点,即相机的y轴移动一个单位,此时,拍摄到的图像会相应的向下移动一个单位,为了使组件看起来依然在原点,这个时候需要调整下相机的方向,相机本来是查看正前方事物,现在让它低头看(0,0,0)点,就像你本来站着眺望远处的高山,手上的手机在视野的下方,现在让你低头45度看看手上的手机,此时手机就在视野的正中心,最终修改相机的代码如下:

  
  
  
  1. camera.position.z = 5;

  2. camera.position.x = 1;

  3. camera.position.y = 1;

  4. camera.lookAt(0, 0, 0);

其中lookAt表示一个方向,即相机对着哪个点看,如下:

三维世界中的坐标系插图5

不过此时的旋转只有立方体在旋转,坐标轴未旋转,要想使坐标轴旋转,修改showCube方法,如下:

  
  
  
  1. function showCube() {

  2.    requestAnimationFrame(showCube)

  3.    cube.rotation.x += 0.01;

  4.    cube.rotation.y += 0.01;

  5.    cube.rotation.z += 0.01;

  6.    axesHelper.rotation.x += 0.01;

  7.    axesHelper.rotation.y += 0.01;

  8.    axesHelper.rotation.z += 0.01;

  9.    render.render(scene, camera);

  10. }

修改之后,旋转效果如下:

三维世界中的坐标系插图6

另外,也可以将这两个组件放到一个Object3D对象中,作为一个整体旋转,如下:

  
  
  
  1. var scene = new THREE.Scene();

  2. var camera = new THREE.PerspectiveCamera(60, window.innerWidth / window.innerHeight, 1, 1000);

  3. var render = new THREE.WebGLRenderer({

  4.    antialias: true

  5. });

  6. var obj = new THREE.Object3D();

  7. render.setClearColor("#FFFFFF");

  8. render.setSize(window.innerWidth, window.innerHeight);

  9. document.body.appendChild(render.domElement);

  10. var geometry = new THREE.CubeGeometry(1, 1, 1);

  11. var material = new THREE.MeshBasicMaterial({color: 0xFFB6C1});

  12. var cube = new THREE.Mesh(geometry, material);

  13. obj.add(cube);

  14. var axesHelper = new THREE.AxesHelper(5);

  15. obj.add(axesHelper);

  16. scene.add(obj);

  17. camera.position.z = 5;

  18. camera.position.x = 1;

  19. camera.position.y = 1;

  20. camera.lookAt(0, 0, 0);

  21. function showCube() {

  22.    requestAnimationFrame(showCube)

  23.    obj.rotation.x += 0.01;

  24.    obj.rotation.y += 0.01;

  25.    obj.rotation.z += 0.01;

  26.    render.render(scene, camera);

  27. }

  28. showCube()

可以看到,立方体和坐标轴都是添加到Object3D中,而Object3D则添加到场景中,旋转时候,只需要Object3D旋转即可,效果与上图类似,这里不再赘述,另外,在创建WebGLRenderer渲染器时,还增加了antialias参数,表示抗锯齿。

好了,本文的介绍就说到这里,有问题欢迎留言讨论。

本文案例:https://github.com/lenve/threejsDemo


往期精彩回顾

Redis教程
SpringCloud教程
Git教程
MongoDB教程
SpringBoot+Vue前后端分离开源项目-微人事
SpringBoot+Vue前后端分离开源项目-V部落

三维世界中的坐标系插图7

本文分享自微信公众号 - 江南一点雨(a_javaboy)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

赞(0) 打赏
未经允许不得转载:IDEA激活码 » 三维世界中的坐标系

一个分享Java & Python知识的社区