如何使用threejs实现第一人称视角的移动

Posted learnhow

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何使用threejs实现第一人称视角的移动相关的知识,希望对你有一定的参考价值。

在数据可视化领域利用webgl来创建三维场景或VR已经越来越普遍,各种开发框架也应运而生。今天我们就通过最基本的threejs来完成第一人称视角的场景巡检功能。如果你是一位threejs的初学者或正打算入门,我强烈推荐你仔细阅读本文并在我的代码基础之上继续深入学习。因为它将是你能够在网上找到的最好的免费中文教程,通过本文你可以学习到一些基本的三维理论,threejs的api接口以及你应该掌握的数学知识。当然要想完全掌握threejs可能还有很长的路需要走,但至少今天我将带你入门并传授一些独特的学习技巧。

第一人称视角的场景巡检主要需要解决两个问题,人物在场景中的移动和碰撞检测。移动与碰撞功能是所有三维场景首先需要解决的基本问题。为了方便理解,首先需要构建一个简单的三维场景并在遇到问题的时候向你演示如何解决它。

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">
 5     <title>平移与碰撞</title>
 6     <script src="js/three.js"></script>
 7     <script src="js/jquery3.4.1.js"></script>
 8 </head>
 9 <body>
10 <canvas id="mainCanvas"></canvas>
11 </body>
12 <script>
13     let scene, camera, renderer, leftPress, cube;
14     init();
15     helper();
16     createBoxer();
17     animate();
18 
19     function init() 
20         // 初始化场景
21         scene = new THREE.Scene();
22         scene.background = new THREE.Color(0xffffff);
23 
24         // 创建渲染器
25         renderer = new THREE.WebGLRenderer(
26             canvas: document.getElementById("mainCanvas"),
27             antialias: true, // 抗锯齿
28             alpha: true
29         );
30         renderer.setSize(window.innerWidth, window.innerHeight);
31 
32 
33         // 创建透视相机
34         camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
35         camera.position.set(0, 40, 30);
36         camera.lookAt(0, 0, 0);
37 
38         // 参数初始化
39         mouse = new THREE.Vector2();
40         raycaster = new THREE.Raycaster();
41 
42         // 环境光
43         var ambientLight = new THREE.AmbientLight(0x606060);
44         scene.add(ambientLight);
45         // 平行光
46         var directionalLight = new THREE.DirectionalLight(0xBCD2EE);
47         directionalLight.position.set(1, 0.75, 0.5).normalize();
48         scene.add(directionalLight);
49     
50 
51     function helper() 
52         var grid = new THREE.GridHelper(100, 20, 0xFF0000, 0x000000);
53         grid.material.opacity = 0.1;
54         grid.material.transparent = true;
55         scene.add(grid);
56 
57         var axesHelper = new THREE.AxesHelper(30);
58         scene.add(axesHelper);
59     
60 
61     function animate() 
62         requestAnimationFrame(animate);
63         renderer.render(scene, camera);
64     
65 
66     function createBoxer() 
67         var geometry = new THREE.BoxGeometry(5, 5, 5);
68         var material = new THREE.MeshPhongMaterial(color: 0x00ff00);
69         cube = new THREE.Mesh(geometry, material);
70         scene.add(cube);
71     
72 
73     $(window).mousemove(function (event) 
74         event.preventDefault();
75         if (leftPress) 
76             cube.rotateOnAxis(
77                 new THREE.Vector3(0, 1, 0),
78                 event.originalEvent.movementX / 500
79             );
80             cube.rotateOnAxis(
81                 new THREE.Vector3(1, 0, 0),
82                 event.originalEvent.movementY / 500
83             );
84         
85     );
86 
87     $(window).mousedown(function (event) 
88         event.preventDefault();
89         leftPress = true;
90 
91     );
92 
93     $(window).mouseup(function (event) 
94         event.preventDefault();
95         leftPress = false;
96     );
97 </script>
98 </html>

很多js的开发人员非常熟悉jquery,我引用它确实让代码显得更加简单。首先我在init()方法里初始化了一个场景。我知道在大部分示例中包括官方提供的demo里都是通过threejs动态的在document下创建一个<canvas/>节点。我强烈建议你不要这样做,因为在很多单页面应用中(例如:Vue和Angular)直接操作DOM都不被推荐。接下来我使用helper()方法创建了两个辅助对象:一个模拟地面的网格和一个表示世界坐标系的AxesHelper。最后我利用createBoxer()方法在视角中央摆放了一个绿色的立方体以及绑定了三个鼠标动作用来控制立方地旋转。如图:

 技术图片

你可以尝试将代码复制到本地并在浏览器中运行,移动鼠标看看效果。接下来,为了让方块移动起来,我们需要添加一些键盘响应事件,以及给方块的“正面”上色。

  1 <!DOCTYPE html>
  2 <html lang="en">
  3 <head>
  4     <meta charset="UTF-8">
  5     <title>平移与碰撞</title>
  6     <script src="js/three.js"></script>
  7     <script src="js/jquery3.4.1.js"></script>
  8 </head>
  9 <body>
 10 <canvas id="mainCanvas"></canvas>
 11 </body>
 12 <script>
 13     let scene, camera, renderer, leftPress, cube;
 14     let left, right, front, back;
 15     init();
 16     helper();
 17     createBoxer();
 18     animate();
 19 
 20     function init() 
 21         // 初始化场景
 22         scene = new THREE.Scene();
 23         scene.background = new THREE.Color(0xffffff);
 24 
 25         // 创建渲染器
 26         renderer = new THREE.WebGLRenderer(
 27             canvas: document.getElementById("mainCanvas"),
 28             antialias: true, // 抗锯齿
 29             alpha: true
 30         );
 31         renderer.setSize(window.innerWidth, window.innerHeight);
 32 
 33 
 34         // 创建透视相机
 35         camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
 36         camera.position.set(0, 40, 30);
 37         camera.lookAt(0, 0, 0);
 38 
 39         // 参数初始化
 40         mouse = new THREE.Vector2();
 41         raycaster = new THREE.Raycaster();
 42 
 43         // 环境光
 44         var ambientLight = new THREE.AmbientLight(0x606060);
 45         scene.add(ambientLight);
 46         // 平行光
 47         var directionalLight = new THREE.DirectionalLight(0xBCD2EE);
 48         directionalLight.position.set(1, 0.75, 0.5).normalize();
 49         scene.add(directionalLight);
 50     
 51 
 52     function helper() 
 53         var grid = new THREE.GridHelper(100, 20, 0xFF0000, 0x000000);
 54         grid.material.opacity = 0.1;
 55         grid.material.transparent = true;
 56         scene.add(grid);
 57 
 58         var axesHelper = new THREE.AxesHelper(30);
 59         scene.add(axesHelper);
 60     
 61 
 62     function animate() 
 63         requestAnimationFrame(animate);
 64         renderer.render(scene, camera);
 65         if (front) 
 66             cube.translateZ(-1)
 67         
 68         if (back) 
 69             cube.translateZ(1);
 70         
 71         if (left) 
 72             cube.translateX(-1);
 73         
 74         if (right) 
 75             cube.translateX(1);
 76         
 77     
 78 
 79     function createBoxer() 
 80         var geometry = new THREE.BoxGeometry(5, 5, 5);
 81         var mats = [];
 82         mats.push(new THREE.MeshPhongMaterial(color: 0x00ff00));
 83         mats.push(new THREE.MeshPhongMaterial(color: 0xff0000));
 84         cube = new THREE.Mesh(geometry, mats);
 85         for (let j = 0; j < geometry.faces.length; j++) 
 86             if (j === 8 || j === 9) 
 87                 geometry.faces[j].materialIndex = 1;
 88              else 
 89                 geometry.faces[j].materialIndex = 0;
 90             
 91         
 92         scene.add(cube);
 93     
 94 
 95     $(window).mousemove(function (event) 
 96         event.preventDefault();
 97         if (leftPress) 
 98             cube.rotateOnAxis(
 99                 new THREE.Vector3(0, 1, 0),
100                 event.originalEvent.movementX / 500
101             );
102             cube.rotateOnAxis(
103                 new THREE.Vector3(1, 0, 0),
104                 event.originalEvent.movementY / 500
105             );
106         
107     );
108 
109     $(window).mousedown(function (event) 
110         event.preventDefault();
111         leftPress = true;
112 
113     );
114 
115     $(window).mouseup(function (event) 
116         event.preventDefault();
117         leftPress = false;
118     );
119 
120     $(window).keydown(function (event) 
121         switch (event.keyCode) 
122             case 65: // a
123                 left = true;
124                 break;
125             case 68: // d
126                 right = true;
127                 break;
128             case 83: // s
129                 back = true;
130                 break;
131             case 87: // w
132                 front = true;
133                 break;
134         
135     );
136 
137     $(window).keyup(function (event) 
138         switch (event.keyCode) 
139             case 65: // a
140                 left = false;
141                 break;
142             case 68: // d
143                 right = false;
144                 break;
145             case 83: // s
146                 back = false;
147                 break;
148             case 87: // w
149                 front = false;
150                 break;
151         
152     );
153 </script>
154 </html>

 

我们添加了keydown()事件和keyup()事件用来捕获键盘响应。我们还修改了createBoxer()方法,给朝向我们的那一面涂上红色。你一定发现了BoxGeometry所代表的立方体虽然只有6个面,可是为了给“1个面”上色我们却需要同时在“2个面”的材质上着色。这是因为在三维场景中,“面”的含义表示由空间中3个点所代表的区域,而一个矩形由两个三角形拼接而成。完成以后的样子如下:

技术图片

随意拖动几下鼠标,我们可能会得到一个类似的状态:

技术图片

设想一下在第一人称视角的游戏中,我们抬高视角观察周围后再降低视角,地平线是否依然处于水平状态。换句话说,无论我们如何拖动鼠标,红色的那面在朝向我们的时候都不应该倾斜。要解释这个问题,我们首先需要搞清楚三维场景中的坐标系概念。在threejs的世界中存在两套坐标体系:世界坐标系和自身坐标系。世界坐标系是整个场景的坐标系统,通过它可以定位场景中的物体。而自身坐标系就比较复杂,实际上一个物体的自身坐标系除了用来表示物体各个部分的相对关系以外主要用来表示物体的旋转。想象一下月球的自转和公转,在地月坐标系中,月球围绕地球公转,同时也绕着自身的Y轴旋转。在我们上面的场景中,立方体自身的坐标轴会随着自身的旋转而改变,当我们的鼠标自下而上滑动后,Y轴将不再垂直于地面。如果这时我们再横向滑动鼠标让立方体绕Y轴旋转,自然整个面都会发生倾斜。如果你还不理解可以在自己的代码中多尝试几次,理解世界坐标系和自身坐标系对于学习webgl尤其重要。很显然,要模拟第一人称的视角转动我们需要让视角上下移动的旋转轴为自身坐标系的X轴,左右移动的旋转轴固定为穿过自身中心的一条与世界坐标系Y轴保持平行的轴线。理解这个问题很不容易,可是解决它却非常简单。threejs为我们提供了方法,我们只需要修改mousemove()方法:

$(window).mousemove(function (event) 
        event.preventDefault();
        if (leftPress) 
            cube.rotateOnWorldAxis(
                new THREE.Vector3(0, 1, 0),
                event.originalEvent.movementX / 500
            );
            cube.rotateOnAxis(
                new THREE.Vector3(1, 0, 0),
                event.originalEvent.movementY / 500
            );
        
    );

有了控制视角的方式,接下来我们移动一下方块。新的问题又出现了:盒子的运动方向也是沿着自身坐标系的。就和我们看着月亮行走并不会走到月亮上去的情形一样,如果要模拟第一人称视角的移动,视角的移动方向应该永远和世界坐标系保持平行,那么我们是否可以通过世界坐标系来控制物体的移动呢:

 1  function animate() 
 2         requestAnimationFrame(animate);
 3         renderer.render(scene, camera);
 4         if (front) 
 5             // cube.translateZ(-1)
 6             cube.position.z -= 1;
 7         
 8         if (back) 
 9             // cube.translateZ(1);
10             cube.position.z += 1;
11         
12         if (left) 
13             // cube.translateX(-1);
14             cube.position.x -= 1;
15         
16         if (right) 
17             // cube.translateX(1);
18             cube.position.x += 1;
19         
20     

很显然也不行,原因是我们应该让物体的前进方向与物体面对的方向保持一致:

技术图片

尽管这个需求显得如此合理,可是threejs似乎并没有提供有效的解决方案,就连官方示例中提供的基于第一人称的移动也仅仅是通过固定物体Y轴数值的方法实现的。在射击游戏中不能蹲下或爬上屋顶实在不能让玩家接受。为了能够在接下来的变换中分解问题和测试效果,我们在模型上添加两个箭头表示物体的前后方向。

 1 let arrowFront, arrowBack;
 2 
 3 function animate() 
 4         requestAnimationFrame(animate);
 5         renderer.render(scene, camera);
 6         arrowFront.setDirection(cube.getWorldDirection(new THREE.Vector3()).normalize());
 7         arrowFront.position.copy(cube.position);
 8         arrowBack.setDirection(cube.getWorldDirection(new THREE.Vector3()).negate().normalize());
 9         arrowBack.position.copy(cube.position);
10         if (front) 
11             // cube.translateZ(-1)
12             cube.position.z -= 1;
13         
14         if (back) 
15             // cube.translateZ(1);
16             cube.position.z += 1;
17         
18         if (left) 
19             // cube.translateX(-1);
20             cube.position.x -= 1;
21         
22         if (right) 
23             // cube.translateX(1);
24             cube.position.x += 1;
25         
26     
27 
28 function createBoxer() 
29         var geometry = new THREE.BoxGeometry(5, 5, 5);
30         var mats = [];
31         mats.push(new THREE.MeshPhongMaterial(color: 0x00ff00));
32         mats.push(new THREE.MeshPhongMaterial(color: 0xff0000));
33         cube = new THREE.Mesh(geometry, mats);
34         for (let j = 0; j < geometry.faces.length; j++) 
35             if (j === 8 || j === 9) 
36                 geometry.faces[j].materialIndex = 1;
37              else 
38                 geometry.faces[j].materialIndex = 0;
39             
40         
41         scene.add(cube);
42         arrowFront = new THREE.ArrowHelper(cube.getWorldDirection(), cube.position, 15, 0xFF0000);
43         scene.add(arrowFront);
44         arrowBack = new THREE.ArrowHelper(cube.getWorldDirection().negate(), cube.position, 15, 0x00FF00);
45         scene.add(arrowBack);
46     

修改后的效果如下:

技术图片

有了箭头的辅助,我们能够以比较直观的方式测试算法是否有效。如果你能够认真读到这里,可能已经迫不及待想继续了,但是还请稍安勿躁。进入下个环节前,我们需要首先了解几个重要的概念。

  1. 三维向量(Vector3):可以表征三维空间中的点或来自原点(0,0,0)的矢量。需要注意,Vector3既可以表示空间中的一个点又可以表示方向。因此为了避免歧义,我建议在作为矢量的时候通过normalize()方法对向量标准化。具体api文档参考
  2. 欧拉角(Euler):表示一个物体在其自身坐标系上的旋转角度,欧拉角也是一个很常见的数学概念,优点是对于旋转的表述相对直观,不过我们在项目中并不常用。
  3. 四元数(Quaternion):四元数是一个相对高深的数学概念,几何含义与欧拉角类似。都可以用来表征物体的旋转方向,优点是运算效率更高。
  4. 四维矩阵(Matrix4):在threejs的世界中,任何一个对象都有它对应的四维矩阵。它集合了平移、旋转、缩放等操作。有时我们可以通过它来完成两个对象的动作同步。
  5. 叉积(.cross() ):向量叉积表示由两个向量所确定的平面的法线方向。叉积的用途很多,例如在第一人称的视角控制下,实现左右平移就可以通过当前视角方向z与垂直方向y做叉积运算获得:z.cross(y)。
  6. 点积(.dot()):与向量叉积不同,向量点积为一个长度数据。vect_a.dot(vect_b)表示向量b在向量a上的投影长度,具体如何使用我们马上就会看到

在理解了上面的概念以后,我们就可以实现沿视角方向平移的操作:我们知道,物体沿平面(XOZ)坐标系运动都可以分解为X方向上的运动分量和Z轴方向上的运动分量。首先获取视角的方向,以三维向量表示。接着我们需要以这个向量和X轴方向上的一个三维向量做点积运算,从而得到一个投影长度。这个长度即代表物体沿视角方向移动的水平x轴方向上的运动分量。同理,我们在计算与Z轴方向上的点积,又可以获得物体沿视角方向移动的z轴方向的运动分量。同时执行两个方向上的运动分量完成平移操作。

接下来,我们先通过实验观察是否能够获得这两个运动分量和投影长度。

  1 <!DOCTYPE html>
  2 <html lang="en">
  3 <head>
  4     <meta charset="UTF-8">
  5     <title>平移与碰撞</title>
  6     <script src="js/three.js"></script>
  7     <script src="js/jquery3.4.1.js"></script>
  8 </head>
  9 <body>
 10 <canvas id="mainCanvas"></canvas>
 11 </body>
 12 <script>
 13     let scene, camera, renderer, leftPress, cube, arrowFront, arrowFrontX, arrowFrontZ;
 14     let left, right, front, back;
 15     init();
 16     // helper();
 17     createBoxer();
 18     animate();
 19 
 20     function init() 
 21         // 初始化场景
 22         scene = new THREE.Scene();
 23         scene.background = new THREE.Color(0xffffff);
 24 
 25         // 创建渲染器
 26         renderer = new THREE.WebGLRenderer(
 27             canvas: document.getElementById("mainCanvas"),
 28             antialias: true, // 抗锯齿
 29             alpha: true
 30         );
 31         renderer.setSize(window.innerWidth, window.innerHeight);
 32 
 33 
 34         // 创建透视相机
 35         camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
 36         camera.position.set(0, 40, 30);
 37         camera.lookAt(0, 0, 0);
 38 
 39         // 参数初始化
 40         mouse = new THREE.Vector2();
 41         raycaster = new THREE.Raycaster();
 42 
 43         // 环境光
 44         var ambientLight = new THREE.AmbientLight(0x606060);
 45         scene.add(ambientLight);
 46         // 平行光
 47         var directionalLight = new THREE.DirectionalLight(0xBCD2EE);
 48         directionalLight.position.set(1, 0.75, 0.5).normalize();
 49         scene.add(directionalLight);
 50     
 51 
 52     function helper() 
 53         var grid = new THREE.GridHelper(100, 20, 0xFF0000, 0x000000);
 54         grid.material.opacity = 0.1;
 55         grid.material.transparent = true;
 56         scene.add(grid);
 57 
 58         var axesHelper = new THREE.AxesHelper(30);
 59         scene.add(axesHelper);
 60     
 61 
 62     function animate() 
 63         requestAnimationFrame(animate);
 64         renderer.render(scene, camera);
 65         arrowFront.setDirection(cube.getWorldDirection(new THREE.Vector3()).normalize());
 66         arrowFront.position.copy(cube.position);
 67 
 68         let vect = cube.getWorldDirection(new THREE.Vector3());
 69         arrowFrontX.setDirection(new THREE.Vector3(1, 0, 0));
 70         arrowFrontX.setLength(vect.dot(new THREE.Vector3(15, 0, 0)));
 71         arrowFrontX.position.copy(cube.position);
 72 
 73         arrowFrontZ.setDirection(new THREE.Vector3(0, 0, 1));
 74         arrowFrontZ.setLength(vect.dot(new THREE.Vector3(0, 0, 15)));
 75         arrowFrontZ.position.copy(cube.position);
 76         if (front) 
 77             // cube.translateZ(-1)
 78             cube.position.z -= 1;
 79         
 80         if (back) 
 81             // cube.translateZ(1);
 82             cube.position.z += 1;
 83         
 84         if (left) 
 85             // cube.translateX(-1);
 86             cube.position.x -= 1;
 87         
 88         if (right) 
 89             // cube.translateX(1);
 90             cube.position.x += 1;
 91         
 92     
 93 
 94     function createBoxer() 
 95         var geometry = new THREE.BoxGeometry(5, 5, 5);
 96         var mats = [];
 97         mats.push(new THREE.MeshPhongMaterial(color: 0x00ff00));
 98         mats.push(new THREE.MeshPhongMaterial(color: 0xff0000));
 99         cube = new THREE.Mesh(geometry, mats);
100         for (let j = 0; j < geometry.faces.length; j++) 
101             if (j === 8 || j === 9) 
102                 geometry.faces[j].materialIndex = 1;
103              else 
104                 geometry.faces[j].materialIndex = 0;
105             
106         
107         scene.add(cube);
108         arrowFront = new THREE.ArrowHelper(cube.getWorldDirection(), cube.position, 15, 0xFF0000);
109         scene.add(arrowFront);
110 
111         let cubeDirec = cube.getWorldDirection(new THREE.Vector3());
112         arrowFrontX = new THREE.ArrowHelper(cubeDirec.setY(0), cube.position, cubeDirec.dot(new THREE.Vector3(0, 0, 15)), 0x0000ff);
113         scene.add(arrowFrontX);
114 
115         arrowFrontZ = new THREE.ArrowHelper(cubeDirec.setY(0), cube.position, cubeDirec.dot(new THREE.Vector3(15, 0, 0)), 0xB5B5B5)
116         scene.add(arrowFrontZ);
117     
118 
119     $(window).mousemove(function (event) 
120         event.preventDefault();
121         if (leftPress) 
122             cube.rotateOnWorldAxis(
123                 new THREE.Vector3(0, 1, 0),
124                 event.originalEvent.movementX / 500
125             );
126             cube.rotateOnAxis(
127                 new THREE.Vector3(1, 0, 0),
128                 event.originalEvent.movementY / 500
129             );
130         
131     );
132 
133     $(window).mousedown(function (event) 
134         event.preventDefault();
135         leftPress = true;
136 
137     );
138 
139     $(window).mouseup(function (event) 
140         event.preventDefault();
141         leftPress = false;
142     );
143 
144     $(window).keydown(function (event) 
145         switch (event.keyCode) 
146             case 65: // a
147                 left = true;
148                 break;
149             case 68: // d
150                 right = true;
151                 break;
152             case 83: // s
153                 back = true;
154                 break;
155             case 87: // w
156                 front = true;
157                 break;
158         
159     );
160 
161     $(window).keyup(function (event) 
162         switch (event.keyCode) 
163             case 65: // a
164                 left = false;
165                 break;
166             case 68: // d
167                 right = false;
168                 break;
169             case 83: // s
170                 back = false;
171                 break;
172             case 87: // w
173                 front = false;
174                 break;
175         
176     );
177 </script>
178 </html>

 

通过箭头的辅助,我们很容易获得以下图形:

技术图片技术图片

红色箭头表示物体的朝向,蓝色表示物体沿x轴上的投影方向和长度。灰色表示沿z轴上的投影方向和长度。在确认方法可行以后,我们继续实现平移操作。完整代码如下,这个运算的方式很重要,读者应该仔细比较两段代码的差别。

  1 <!DOCTYPE html>
  2 <html lang="en">
  3 <head>
  4     <meta charset="UTF-8">
  5     <title>平移与碰撞</title>
  6     <script src="js/three.js"></script>
  7     <script src="js/jquery3.4.1.js"></script>
  8 </head>
  9 <body>
 10 <canvas id="mainCanvas"></canvas>
 11 </body>
 12 <script>
 13     let scene, camera, renderer, leftPress, cube, arrowFront, arrowFrontX, arrowFrontZ;
 14     let left, right, front, back;
 15     init();
 16     helper();
 17     createBoxer();
 18     animate();
 19 
 20     function init() 
 21         // 初始化场景
 22         scene = new THREE.Scene();
 23         scene.background = new THREE.Color(0xffffff);
 24 
 25         // 创建渲染器
 26         renderer = new THREE.WebGLRenderer(
 27             canvas: document.getElementById("mainCanvas"),
 28             antialias: true, // 抗锯齿
 29             alpha: true
 30         );
 31         renderer.setSize(window.innerWidth, window.innerHeight);
 32 
 33 
 34         // 创建透视相机
 35         camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
 36         camera.position.set(0, 40, 30);
 37         camera.lookAt(0, 0, 0);
 38 
 39         // 参数初始化
 40         mouse = new THREE.Vector2();
 41         raycaster = new THREE.Raycaster();
 42 
 43         // 环境光
 44         var ambientLight = new THREE.AmbientLight(0x606060);
 45         scene.add(ambientLight);
 46         // 平行光
 47         var directionalLight = new THREE.DirectionalLight(0xBCD2EE);
 48         directionalLight.position.set(1, 0.75, 0.5).normalize();
 49         scene.add(directionalLight);
 50     
 51 
 52     function helper() 
 53         var grid = new THREE.GridHelper(100, 20, 0xFF0000, 0x000000);
 54         grid.material.opacity = 0.1;
 55         grid.material.transparent = true;
 56         scene.add(grid);
 57 
 58         var axesHelper = new THREE.AxesHelper(30);
 59         scene.add(axesHelper);
 60     
 61 
 62     function animate() 
 63         requestAnimationFrame(animate);
 64         renderer.render(scene, camera);
 65         arrowFront.setDirection(cube.getWorldDirection(new THREE.Vector3()).normalize());
 66         arrowFront.position.copy(cube.position);
 67         let vect = cube.getWorldDirection(new THREE.Vector3());
 68         if (front) 
 69             cube.position.z += vect.dot(new THREE.Vector3(0, 0, 15)) * 0.01;
 70             cube.position.x += vect.dot(new THREE.Vector3(15, 0, 0)) * 0.01;
 71         
 72     
 73 
 74     function createBoxer() 
 75         var geometry = new THREE.BoxGeometry(5, 5, 5);
 76         var mats = [];
 77         mats.push(new THREE.MeshPhongMaterial(color: 0x00ff00));
 78         mats.push(new THREE.MeshPhongMaterial(color: 0xff0000));
 79         cube = new THREE.Mesh(geometry, mats);
 80         for (let j = 0; j < geometry.faces.length; j++) 
 81             if (j === 8 || j === 9) 
 82                 geometry.faces[j].materialIndex = 1;
 83              else 
 84                 geometry.faces[j].materialIndex = 0;
 85             
 86         
 87         scene.add(cube);
 88         arrowFront = new THREE.ArrowHelper(cube.getWorldDirection(), cube.position, 15, 0xFF0000);
 89         scene.add(arrowFront);
 90     
 91 
 92     $(window).mousemove(function (event) 
 93         event.preventDefault();
 94         if (leftPress) 
 95             cube.rotateOnWorldAxis(
 96                 new THREE.Vector3(0, 1, 0),
 97                 event.originalEvent.movementX / 500
 98             );
 99             cube.rotateOnAxis(
100                 new THREE.Vector3(1, 0, 0),
101                 event.originalEvent.movementY / 500
102             );
103         
104     );
105 
106     $(window).mousedown(function (event) 
107         event.preventDefault();
108         leftPress = true;
109 
110     );
111 
112     $(window).mouseup(function (event) 
113         event.preventDefault();
114         leftPress = false;
115     );
116 
117     $(window).keydown(function (event) 
118         switch (event.keyCode) 
119             case 65: // a
120                 left = true;
121                 break;
122             case 68: // d
123                 right = true;
124                 break;
125             case 83: // s
126                 back = true;
127                 break;
128             case 87: // w
129                 front = true;
130                 break;
131         
132     );
133 
134     $(window).keyup(function (event) 
135         switch (event.keyCode) 
136             case 65: // a
137                 left = false;
138                 break;
139             case 68: // d
140                 right = false;
141                 break;
142             case 83: // s
143                 back = false;
144                 break;
145             case 87: // w
146                 front = false;
147                 break;
148         
149     );
150 </script>
151 </html>

向后和左右平移的操作留给大家自己实现。有了以上基础,如何控制Camera移动就很简单了。几乎就是将cube的操作替换成camera即可:

  1 <!DOCTYPE html>
  2 <html lang="en">
  3 <head>
  4     <meta charset="UTF-8">
  5     <title>第一人称视角移动</title>
  6     <script src="js/three.js"></script>
  7     <script src="js/jquery3.4.1.js"></script>
  8 </head>
  9 <body>
 10 <canvas id="mainCanvas"></canvas>
 11 </body>
 12 <script>
 13     let scene, camera, renderer, leftPress, cube, arrowFront, arrowFrontX, arrowFrontZ;
 14     let left, right, front, back;
 15     init();
 16     helper();
 17     animate();
 18 
 19     function init() 
 20         // 初始化场景
 21         scene = new THREE.Scene();
 22         scene.background = new THREE.Color(0xffffff);
 23 
 24         // 创建渲染器
 25         renderer = new THREE.WebGLRenderer(
 26             canvas: document.getElementById("mainCanvas"),
 27             antialias: true, // 抗锯齿
 28             alpha: true
 29         );
 30         renderer.setSize(window.innerWidth, window.innerHeight);
 31 
 32 
 33         // 创建透视相机
 34         camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
 35         camera.position.set(0, 10, 30);
 36 
 37         // 参数初始化
 38         mouse = new THREE.Vector2();
 39         raycaster = new THREE.Raycaster();
 40 
 41         // 环境光
 42         var ambientLight = new THREE.AmbientLight(0x606060);
 43         scene.add(ambientLight);
 44         // 平行光
 45         var directionalLight = new THREE.DirectionalLight(0xBCD2EE);
 46         directionalLight.position.set(1, 0.75, 0.5).normalize();
 47         scene.add(directionalLight);
 48     
 49 
 50     function helper() 
 51         var grid = new THREE.GridHelper(100, 20, 0xFF0000, 0x000000);
 52         grid.material.opacity = 0.1;
 53         grid.material.transparent = true;
 54         scene.add(grid);
 55 
 56         var axesHelper = new THREE.AxesHelper(30);
 57         scene.add(axesHelper);
 58     
 59 
 60     function animate() 
 61         requestAnimationFrame(animate);
 62         renderer.render(scene, camera);
 63         let vect = camera.getWorldDirection(new THREE.Vector3());
 64         if (front) 
 65             camera.position.z += vect.dot(new THREE.Vector3(0, 0, 15)) * 0.01;
 66             camera.position.x += vect.dot(new THREE.Vector3(15, 0, 0)) * 0.01;
 67         
 68     
 69     
 70     $(window).mousemove(function (event) 
 71         event.preventDefault();
 72         if (leftPress) 
 73             camera.rotateOnWorldAxis(
 74                 new THREE.Vector3(0, 1, 0),
 75                 event.originalEvent.movementX / 500
 76             );
 77             camera.rotateOnAxis(
 78                 new THREE.Vector3(1, 0, 0),
 79                 event.originalEvent.movementY / 500
 80             );
 81         
 82     );
 83 
 84     $(window).mousedown(function (event) 
 85         event.preventDefault();
 86         leftPress = true;
 87 
 88     );
 89 
 90     $(window).mouseup(function (event) 
 91         event.preventDefault();
 92         leftPress = false;
 93     );
 94 
 95     $(window).keydown(function (event) 
 96         switch (event.keyCode) 
 97             case 65: // a
 98                 left = true;
 99                 break;
100             case 68: // d
101                 right = true;
102                 break;
103             case 83: // s
104                 back = true;
105                 break;
106             case 87: // w
107                 front = true;
108                 break;
109         
110     );
111 
112     $(window).keyup(function (event) 
113         switch (event.keyCode) 
114             case 65: // a
115                 left = false;
116                 break;
117             case 68: // d
118                 right = false;
119                 break;
120             case 83: // s
121                 back = false;
122                 break;
123             case 87: // w
124                 front = false;
125                 break;
126         
127     );
128 </script>
129 </html>

解决了平移操作以后,碰撞检测其实就不那么复杂了。我们可以沿着摄像机的位置向上下前后左右六个方向做光线投射(Raycaster),每次移动首先检测移动方向上的射线是否被阻挡,如果发生阻挡且距离小于安全距离,即停止该方向上的移动。后面的部分我打算放在下一篇博客中介绍,如果大家对这篇文章敢兴趣或有什么建议欢迎给我留言或加群讨论。

以上是关于如何使用threejs实现第一人称视角的移动的主要内容,如果未能解决你的问题,请参考以下文章

Unity第一人称视角开发

如何在 Roblox 第一人称视图中以编程方式调整旋转量?

unity 第一人称角色控制器手机虚拟双摇杆

第一人称玩家移动 摄像机跟随

在场景中加入第一人称视角运行后一直往下掉的解决方案

游戏开发实战Unity手游第一人称视角,双摇杆控制,FPS射击游戏Demo(教程 | 含Demo工程源码)