本文实例为大家分享了利用射线Raycaster进行碰撞检测的具体代码,供大家参考,具体内容如下
学习碰撞检测之前,我们先了解一下Raycaster类
Raycaster 应该翻译为“光线投射”,顾名思义,就是投射出去的一束光线。
Raycaster的构造函数如下
Raycaster( origin, direction, near, far ) { origin — 射线的起点向量。 direction — 射线的方向向量,应该归一化。 near — 所有返回的结果应该比 near 远。Near不能为负,默认值为0。 far — 所有返回的结果应该比 far 近。Far 不能小于 near,默认值为无穷大。
使用Raycaster进行碰撞检测
用Raycaster来检测碰撞的原理很简单,我们需要以物体的中心为起点,向各个顶点(vertices)发出射线,然后检查射线是否与其它的物体相交。如果出现了相交的情况,检查最近的一个交点与射线起点间的距离,如果这个距离比射线起点至物体顶点间的距离要小,则说明发生了碰撞。
这个方法有一个 缺点 ,当物体的中心在另一个物体内部时,是不能够检测到碰撞的。而且当两个物体能够互相穿过,且有较大部分重合时,检测效果也不理想。
还有需要 注意 的一点是:在Three.js中创建物体时,它的顶点(veritces)数目是与它的分段数目相关的,分段越多,顶点数目越多。为了检测过程中的准确度考虑,需要适当增加物体的分段。
检测光线是否与物体相交使用的是 intersectObject 或 intersectObjects 方法:
.intersectObject ( object, recursive ) //object — 检测该物体是否与射线相交。 //recursive — 如果设置,则会检测物体所有的子代。
相交的结果会以一个数组的形式返回,其中的元素依照距离排序,越近的排在越前.
这样通过对数组中的元素进行处理,就能得出想要的结果。
intersectObjects 与 intersectObject 类似,除了传入的参数是一个数组之外,并无大的差别。
/** * 功能:检测 movingCube 是否与数组 collideMeshList 中的元素发生了碰撞 * */ var originPoint = movingCube.position.clone(); for (var vertexIndex = 0; vertexIndex < movingCube.geometry.vertices.length; vertexIndex++) { // 顶点原始坐标 var localVertex = movingCube.geometry.vertices[vertexIndex].clone(); // 顶点经过变换后的坐标 var globalVertex = localVertex.applyMatrix4(movingCube.matrix); // 获得由中心指向顶点的向量 var directionVector = globalVertex.sub(movingCube.position); // 将方向向量初始化 var ray = new THREE.Raycaster(originPoint, directionVector.clone().normalize()); // 检测射线与多个物体的相交情况 var collisionResults = ray.intersectObjects(collideMeshList); // 如果返回结果不为空,且交点与射线起点的距离小于物体中心至顶点的距离,则发生了碰撞 if (collisionResults.length > 0 && collisionResults[0].distance < directionVector.length()) { crash = true; // crash 是一个标记变量 } }
在Three.js中是使用矩阵来记录3D转换的,每一个Object3D的实例都有一个矩阵,存储了位置position,旋转rotation和伸缩scale。
var globalVertex = localVertex.applyMatrix4(movingCube.matrix);
这一句代码将物体的本地坐标乘以变换矩阵,得到了这个物体在世界坐标系中的值,处理之后的值才是我们所需要的。
下面是一个测试的完整实例:
index.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <script src="/UploadFiles/2021-04-02/three.js">Main.js
var scene,camera,controls,renderer,cube,originPoint; var WIDTH,HEIGHT; var objects = []; //创建渲染器 function initRenderer(){ WIDTH = window.innerWidth; HEIGHT = window.innerHeight; renderer = new THREE.WebGLRenderer({ antialias:true, }); renderer.setSize(WIDTH,HEIGHT); renderer.setPixelRatio(WIDTH/HEIGHT); document.getElementById('canvas-frame').appendChild(renderer.domElement); } //创建场景 function initScene(){ scene = new THREE.Scene(); scene.background = new THREE.Color( 0xf0f0f0 ); } //创建相机 function initCamera(){ camera = new THREE.PerspectiveCamera(50,WIDTH/HEIGHT,1,10000); camera.position.set(0,0,1000); camera.lookAt(0,0,0); } //创建光源 function initLight(){ // 方向光 var directionalLight = new THREE.DirectionalLight( 0xffffff, 0.5 ); scene.add( directionalLight ); // 环境光 scene.add( new THREE.AmbientLight( 0x505050 ) ); } //创建对象 function initObject(){ var geometry = new THREE.BoxBufferGeometry( 40, 40, 40 ); for ( var i = 0; i < 2; i ++ ) { var object = new THREE.Mesh( geometry, new THREE.MeshLambertMaterial( { color: Math.random() * 0xffffff } ) ); //随机位置 object.position.x = Math.random() * 1000 - 500; object.position.y = Math.random() * 600 - 300; object.position.z = Math.random() * 800 - 400; //随机角度 object.rotation.x = Math.random() * 2 * Math.PI; object.rotation.y = Math.random() * 2 * Math.PI; object.rotation.z = Math.random() * 2 * Math.PI; //随机大小 object.scale.x = Math.random() * 2 + 1; object.scale.y = Math.random() * 2 + 1; object.scale.z = Math.random() * 2 + 1; //开启阴影 object.castShadow = true; object.receiveShadow = true; scene.add( object ); // 放入数组 objects.push( object ); } // var geometry = new THREE.BoxGeometry( 80, 80, 80 ); var material = new THREE.MeshLambertMaterial( {color: 0xfff000} ); cube = new THREE.Mesh( geometry, material ); scene.add( cube ); /** * .clone () : Vector3 * 返回一个新的Vector3,其具有和当前这个向量相同的x、y和z。 */ originPoint = cube.position.clone(); } //创建控制器 function initControls(){ // TrackballControls 轨迹球控件,最常用的控件,可以使用鼠标轻松的移动、平移,缩放场景。 controls = new THREE.TrackballControls( camera ); controls.rotateSpeed = 1.0;// 旋转速度 controls.zoomSpeed = 1.2;// 缩放速度 controls.panSpeed = 0.8;// 平controls controls.noZoom = false; controls.noPan = false; controls.staticMoving = true;// 静止移动,为 true 则没有惯性 controls.dynamicDampingFactor = 0.3;// 阻尼系数 越小 则滑动越大 // DragControls 初始化拖拽控件 var dragControls = new THREE.DragControls( objects, camera, renderer.domElement ); // 开始拖拽 dragControls.addEventListener( 'dragstart', function () { controls.enabled = false; } ); // 拖拽结束 dragControls.addEventListener( 'dragend', function () { controls.enabled = true; } ); } function initThree(){ initRenderer(); initScene(); initCamera(); initLight(); initObject(); initControls(); animation(); } //循环 function animation(){ requestAnimationFrame(animation); renderer.render(scene,camera); // 更新控制器 controls.update(); // 循环碰撞检测 for (var i = 0; i < cube.geometry.vertices.length; i++) { // 顶点原始坐标 var localVertex = cube.geometry.vertices[i].clone(); // 顶点经过变换后的坐标 // matrix 局部变换矩阵。 applyMatrix4 并返回新Matrix4(4x4矩阵)对象. var globalVertex = localVertex.applyMatrix4(cube.matrix); // 获得由中心指向顶点的向量 var directionVector = globalVertex.sub(cube.position); // 将方向向量初始化 var ray = new THREE.Raycaster(originPoint, directionVector.clone().normalize()); // 检测射线与多个物体的相交情况 var collisionResults = ray.intersectObjects(objects); // 如果返回结果不为空,且交点与射线起点的距离小于物体中心至顶点的距离,则发生了碰撞 if (collisionResults.length > 0 && collisionResults[0].distance < directionVector.length()) { console.log('碰撞!'); } } }以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 凤飞飞《我们的主题曲》飞跃制作[正版原抓WAV+CUE]
- 刘嘉亮《亮情歌2》[WAV+CUE][1G]
- 红馆40·谭咏麟《歌者恋歌浓情30年演唱会》3CD[低速原抓WAV+CUE][1.8G]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[320K/MP3][193.25MB]
- 【轻音乐】曼托凡尼乐团《精选辑》2CD.1998[FLAC+CUE整轨]
- 邝美云《心中有爱》1989年香港DMIJP版1MTO东芝首版[WAV+CUE]
- 群星《情叹-发烧女声DSD》天籁女声发烧碟[WAV+CUE]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[FLAC/分轨][748.03MB]
- 理想混蛋《Origin Sessions》[320K/MP3][37.47MB]
- 公馆青少年《我其实一点都不酷》[320K/MP3][78.78MB]
- 群星《情叹-发烧男声DSD》最值得珍藏的完美男声[WAV+CUE]
- 群星《国韵飘香·贵妃醉酒HQCD黑胶王》2CD[WAV]
- 卫兰《DAUGHTER》【低速原抓WAV+CUE】
- 公馆青少年《我其实一点都不酷》[FLAC/分轨][398.22MB]
- ZWEI《迟暮的花 (Explicit)》[320K/MP3][57.16MB]