【three.js】23. Raycaster and Mouse Events 投射射线(碰撞检测)和鼠标事件

本文主要是介绍【three.js】23. Raycaster and Mouse Events 投射射线(碰撞检测)和鼠标事件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

介绍

顾名思义,Raycaster 可以向特定方向投射(或发射)一条射线,并测试与它相交的对象。

您可以使用该技术来检测玩家前面是否有墙,测试激光枪是否击中了什么东西,测试当前鼠标下方是否有东西来模拟鼠标事件,以及许多其他事情。

设置

在我们的启动器中,我们有 3 个红色球体,我们将射出一条光线,看看这些球体是否相交。

创建光线投射器

实例化一个Raycaster:

/*** Raycaster*/
const raycaster = new THREE.Raycaster()

要改变光线投射的位置和方向,我们可以使用 set(...)方法。第一个参数是position,第二个参数是direction。它需要两个向量作为参数:一个是起点位置,另一个是方向。
两者都是Vector3,但direction必须进行归一化。归一化向量的长度为1. 别担心,你不必自己做数学运算,你可以调用normalize()向量上的方法:

const rayOrigin = new THREE.Vector3(- 3, 0, 0)
const rayDirection = new THREE.Vector3(10, 0, 0)
rayDirection.normalize()raycaster.set(rayOrigin, rayDirection)

在这个例子中,起点位置是(-3, 0, 0),方向是(10, 0, 0),我们通过normalize()方法将方向向量归一化处理,使其长度为1,表示这是一个单位向量。
然后将起点位置和方向向量设置为射线的起点和方向,最后使用raycaster.set()方法将它们传递给Raycaster对象。这个射线可以用于碰撞检测或者其他的渲染相关工作。
在这里,光线位置应该从我们场景的左侧开始,方向似乎向右。我们的光线应该穿过所有球体。

投射光线

要投射光线并获得相交的对象,我们可以使用两种方法,intersectObject(...)(单数)和intersectObjects(...)(复数)。
intersectObject(...)将测试一个对象并将intersectObjects(...)测试一组对象:

const intersect = raycaster.intersectObject(object2)
console.log(intersect)const intersects = raycaster.intersectObjects([object1, object2, object3])
console.log(intersects)

如果您查看日志,您会看到intersectObject(...)返回了一个包含一个对象的数组(可能是第二个球体)并且 intersectObjects(...)返回了一个包含三个对象的数组(可能是 3 个球体的集合)。

交集的结果

raycaster.intersectObject交集的结果始终是一个数组,即使您只测试了一个对象。那是因为一条光线可以多次穿过同一个物体。想象一个甜甜圈。光线将穿过环的第一部分,然后穿过中间的孔,然后再次穿过环的第二部分,这样交集的结果就是2个对象了。

返回数组的每一项都包含很多有用的信息:

  • distance:射线原点和碰撞点之间的距离。
  • face:几何体的哪个面被光线击中。
  • faceIndex: 那张脸的索引。
  • object: 碰撞涉及什么对象。
  • point:碰撞在 3D 空间中的确切位置的Vector3 。
  • uv:该几何体中的 UV 坐标。

使用哪个数据取决于您。如果你想测试玩家面前是否有墙,你可以测试distance的值. 如果要更改对象的颜色,可以更新 object的材质。如果你想在冲击点上显示爆炸特效,你可以在该point位置创建这个爆炸动画。

测试每一帧

目前,我们一开始只投射一条光线。如果我们想在物体移动时对其进行测试,我们必须在每一帧上进行测试。让我们为球体设置动画,并在光线与它们相交时将它们变成蓝色。
删除我们之前所做的代码,只保留 raycaster 实例化:

const raycaster = new THREE.Raycaster()

通过使用tick函数中的经过时间和经典Math.sin(...)来为球体制作动画:

const clock = new THREE.Clock()const tick = () =>
{const elapsedTime = clock.getElapsedTime()// Animate objectsobject1.position.y = Math.sin(elapsedTime * 0.3) * 1.5object2.position.y = Math.sin(elapsedTime * 0.8) * 1.5object3.position.y = Math.sin(elapsedTime * 1.4) * 1.5// ...
}

004.gif
您应该看到球体以不同的频率上下波动。
现在让我们在tick函数中 像以前一样更新我们的 raycaster

const clock = new THREE.Clock()const tick = () =>
{// ...// Cast a rayconst rayOrigin = new THREE.Vector3(- 3, 0, 0)const rayDirection = new THREE.Vector3(1, 0, 0)rayDirection.normalize()raycaster.set(rayOrigin, rayDirection)const objectsToTest = [object1, object2, object3]const intersects = raycaster.intersectObjects(objectsToTest)console.log(intersects)// ...
}

我们不需要规范化,因为rayDirection它的长度已经是1。但最好保留 normalize()归一化向量以防我们改变方向导致向量没有归一化。
我们还将要测试的对象数组放在一个objectsToTest变量中。因为接下来会派上用场的。
如果您查看控制台,您应该会得到一个包含交点的数组,并且这些交点会根据球体的位置不断变化。
我们现在可以object为数组的每一项更新属性的材质intersects

for(const intersect of intersects){intersect.object.material.color.set('#0000ff')}

005.gif
不幸的是,它们都变蓝了,但再也不会变红了。有很多方法可以将不相交的对象变回红色。我们可以做的是将所有球体变成红色,然后将相交的球体变成蓝色:

for(const object of objectsToTest){object.material.color.set('#ff0000')}for(const intersect of intersects){intersect.object.material.color.set('#0000ff')}

006.gif

用鼠标使用 raycaster

正如我们之前所说,我们还可以使用光线投射器来测试鼠标后面是否有物体。换句话说,如果你悬停在一个物体上就测试它。
从数学上讲,它有点复杂,因为我们需要从相机向鼠标方向投射光线,但幸运的是,Three.js 完成了所有繁重的工作。
现在,让我们在tick函数中注释与 raycaster 相关的代码。

徘徊悬停

首先,让我们处理悬停。
首先,我们需要鼠标的坐标。我们不能使用以像素为单位的基本原生 JavaScript 坐标。我们需要一个在水平轴和垂直轴上都从-1+1的范围,当鼠标向上移动时,垂直坐标为正。
这就是 WebGL 的工作原理,它与裁剪空间之类的东西有关,但我们不需要理解那些复杂的概念。
例子:

  • 鼠标在页面左上角:-1 / 1
  • 鼠标在页面左下方:-1 / - 1
  • 鼠标垂直居中,水平居右:1 / 0
  • 鼠标在页面中央:0 / 0

首先,让我们创建一个带有Vector2 的mouse变量,并在鼠标移动时更新该变量:

/*** Mouse*/
const mouse = new THREE.Vector2()window.addEventListener('mousemove', (event) =>
{mouse.x = event.clientX / sizes.width * 2 - 1mouse.y = - (event.clientY / sizes.height) * 2 + 1console.log(mouse)
})

查看日志并确保值与前面的示例匹配。
我们可以在mousemove事件回调中投射射线,但不建议这样做,因为mousemove对于某些浏览器,事件的触发速度可能超过帧率。我们将像以前一样在tick函数中投射射线进行碰撞检测。
为了将光线定向到正确的方向,我们可以使用Raycaster上setFromCamera()的方法。其余代码与之前相同。如果对象相交或不相交,我们只需将对象材料更新为红色或蓝色:

const tick = () =>
{// ...raycaster.setFromCamera(mouse, camera)const objectsToTest = [object1, object2, object3]const intersects = raycaster.intersectObjects(objectsToTest)for(const intersect of intersects){intersect.object.material.color.set('#0000ff')}for(const object of objectsToTest){if(!intersects.find(intersect => intersect.object === object)){object.material.color.set('#ff0000')}}// ...
}

如果光标在球体上方,球体应该变成蓝色。tutieshi_640x400_4s.gif

鼠标进入和鼠标离开事件

在three项目中'mouseenter''mouseleave'等鼠标事件也不支持。如果您想在鼠标“进入”一个对象或“离开”该对象时得到通知,您必须自己完成。
我们可以做的是重现mouseentermouseleave事件,即拥有一个包含当前悬停对象的变量。
如果有一个对象相交,但之前没有,则表示该a对象发生了mouseenter
如果没有对象相交,但之前有一个,则表示mouseleave发生了。
我们只需要保存当前相交的对象:

let currentIntersect = null

然后,测试并更新currentIntersect变量:

const tick = () =>
{// ...raycaster.setFromCamera(mouse, camera)const objectsToTest = [object1, object2, object3]const intersects = raycaster.intersectObjects(objectsToTest)if(intersects.length){if(!currentIntersect){console.log('mouse enter')}currentIntersect = intersects[0]}else{if(currentIntersect){console.log('mouse leave')}currentIntersect = null}// ...
}

鼠标点击事件

现在我们有了一个包含当前悬停对象的变量,我们可以轻松地实现一个click事件。
首先,我们需要监听click事件,不管它发生在哪里:

window.addEventListener('click', () =>
{})

然后,我们可以测试currentIntersect变量中是否有东西:

window.addEventListener('click', () =>
{if(currentIntersect){console.log('click')}
})

我们还可以测试点击关注的是什么对象:

window.addEventListener('click', () =>
{if(currentIntersect){switch(currentIntersect.object){case object1:console.log('click on object 1')breakcase object2:console.log('click on object 2')breakcase object3:console.log('click on object 3')break}}
})

重现本机事件需要时间,但一旦您理解了它,它就会非常简单。

使用模型进行光线投射

这一切都很好,但是我们可以将光线投射应用于导入的模型吗?
答案是肯定的,而且其实很容易。但我们将一起做,因为我们可以在此过程中学到一些有趣的东西。
首先,我们需要一个模型。

加载模型

我们在上一课中使用的 Duck 模型位于该static/models/Duck/文件夹中。
现在是尝试自行加载该模型并将其添加到场景中的好时机。
首先,我们将使用GLTFLoader。
引入GLTFLoaderthree/examples/jsm/loaders/GLTFLoader.js

import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader.js'

接下来,我们需要实例化它。
您可以将该代码放在函数实例化之后scenetick函数之前的任何位置:

/*** Model*/
const gltfLoader = new GLTFLoader()

我们现在可以调用该load方法。这两个参数是文件的路径和加载模型时应调用的函数。
我们将使用glTF-Binary,但请随意使用其他版本GLTFLoader。另外,不要忘记如果要使用Draco压缩版需要DracoLoader在实例中添加实例。
调用该方法并作为路径(没有路径)和一个带有控制台日志的函数load发送:'./models/Duck/glTF-Binary/Duck.glb'static/

gltfLoader.load('./models/Duck/glTF-Binary/Duck.glb',() =>{console.log('loaded')}
)


'loaded'您应该在控制台中看到。
我们现在可以将模型添加到场景中。首先,gltf向函数添加一个参数:

gltfLoader.load('./models/Duck/glTF-Binary/Duck.glb',(gltf) =>{console.log('loaded')}
)

现在,包含在您自己的scene属性中add gltf.scene整个加载场景:

gltfLoader.load('./models/Duck/glTF-Binary/Duck.glb',(gltf) =>{scene.add(gltf.scene)}
)

如您所见,有些地方不对劲。

灯灯

如果您尝试过自己做,您可能在执行这一步时遇到了一些困难。
好像场景里加了点什么,却是一片漆黑。原因是我们的 Duck 材质是 MeshStandardMaterial ,这种材质只有在灯光下才能看到。
让我们添加一个AmbientLight和一个DirectionalLight:

/*** Lights*/
// Ambient light
const ambientLight = new THREE.AmbientLight('#ffffff', 0.3)
scene.add(ambientLight)// Directional light
const directionalLight = new THREE.DirectionalLight('#ffffff', 0.7)
directionalLight.position.set(1, 2, 3)
scene.add(directionalLight)


现在我们可以看到 Duck,将它向下移动一点:

gltfLoader.load('./models/Duck/glTF-Binary/Duck.glb',(gltf) =>{gltf.scene.position.y = - 1.2scene.add(gltf.scene)}
)

与模型相交

让我们在模型上试试 raycaster
这个练习会很简单。我们希望 Duck 在光标进入时变大,在光标离开时恢复到正常大小。
我们将在每一帧上测试光标是否在 Duck 中,这意味着我们需要配置该tick功能。光线投射器已经通过鼠标设置,我们可以在与我们对球体进行的测试相关的代码之后立即进行相交测试。
以前,我们曾经针对 raycaster内的一组网格raycaster.intersectObjects进行测试。但是现在,我们测试的是一个gltf.scene, 是的,这个对象可能有多个孩子,更糟糕的是,孩子中也会有孩子,但你会发现这不是问题,我们仍在测试一整个对象。
我们不使用intersectObjects(复数),而是使用intersectObject(单数)。它的工作原理是一样的,也会返回一个交集数组,但我们必须向它发送一个对象而不是对象数组。
那么,你必须做什么?首先,创建一个modelIntersects变量(这样它就不会与intersects变量冲突),然后调用raycaster.intersectObject(单数)方法,最后将其发送gltf.scene(此代码无效):

const tick = () =>
{// ...// Test intersect with a modelconst modelIntersects = raycaster.intersectObject(gltf.scene)console.log(modelIntersects)// Update controls// ...
}


我们在这里犯了一个错误。如果你熟悉 JS,你就会知道我们无法从加载的回调函数访问外部gltf变量。我们称之为变量的“范围”。
此外,加载模型需要时间。是的,我们正在使用一个非常简单的模型在本地进行测试,但情况可能会有所不同,在线加载复杂的对象需要时间。
当您尝试与加载的模型进行交互或为加载的模型设置动画时,这些都是您将遇到的经典问题。
为了解决这两个问题,我们将在加载模型之前使用 let 创建一个model变量并将其设置为null(相当于 JavaScript 中的“无”):

let model = null
gltfLoader.load(// ...
)

由于我们model在函数之外创建了该变量,因此我们将能够在tick函数中使用它。
接下来,当加载模型时,我们将 分配gltf.scenemodel

let model = null
gltfLoader.load('./models/Duck/glTF-Binary/Duck.glb',(gltf) =>{model = gltf.scenegltf.scene.position.y = - 1.2scene.add(gltf.scene)}
)

你也可以在加载的函数中把gltf.scene替换成model因为它更佳语意化方便阅读,尽管它是可选的:

let model = null
gltfLoader.load('./models/Duck/glTF-Binary/Duck.glb',(gltf) =>{model = gltf.scenemodel.position.y = - 1.2scene.add(model)}
)

回到tick函数和我们的intersectObject:我们现在可以使用model变量而不是gltf.scene(这段代码现在还不能工作):

const tick = () =>
{// ...// Test intersect with a modelconst modelIntersects = raycaster.intersectObject(model)console.log(modelIntersects)// ...
}


再一次,我们得到一个错误。我们忘记了js的同步执行,因为加载模型需要时间,这意味着model变量将暂时已null存在。
model我们在这里可以做的只是测试语句中是否有内容if

const tick = () =>
{// ...if(model){const modelIntersects = raycaster.intersectObject(model)console.log(modelIntersects)}// ...
}

现在我们得到了相交数组。

笔记

在我们解决 Duck size 这个功能之前,有几件事需要注意。

递归

首先,我们调用intersectObjectmodel它是一个Group,而不是Mesh。
您可以在加载的回调函数中分配model之前通过记录来测试它:

let model = null
gltfLoader.load('./models/Duck/glTF-Binary/Duck.glb',(gltf) =>{model = gltf.sceneconsole.log(model)model.position.y = - 1.2scene.add(model)}
)


这不应该工作,因为 Raycaster 应该针对网格进行测试。它起作用的原因是,默认情况下,Raycaster 将检查对象的子对象。更好的是,它会递归地测试所有的内部孩子。
实际上,我们可以通过将intersectObjectintersectObjects方法的第二个参数设置为 false 来选择停用该选项,但我们可以接受默认行为。

相交数组

第二点要注意的是,当我们只测试一个对象时,我们收到了一组相交。
第一个原因是,由于 Raycaster 正在递归地测试子项,因此可能有多个与射线相交的网格。此处情况并非如此,因为 Duck 仅由一个网格构成,但我们本可以测试更复杂的模型。
第二个原因是,正如我们之前看到的,即使是一个网格也可以与一条射线相交多次,我们的鸭子就是这种情况。从一个非常特定的角度进行测试,您可以有多个相交点:

更新规模

我们快完成了。我们现在需要做的就是根据相交数组更新模型的scale
在调用 intersectObject之后,我们可以测试数组的length
0被认为是false,所以我们可以只使用modelIntersects.lengthas 条件。
如果在上方0,则为true,这意味着鼠标悬停在模型上,我们应该增加比例。否则,它将是false,这意味着鼠标没有悬停在模型上,我们应该将比例设置为1

const tick = () =>
{// ...if(model){const modelIntersects = raycaster.intersectObject(model)if(modelIntersects.length){model.scale.set(1.2, 1.2, 1.2)}else{model.scale.set(1, 1, 1)}}// ...
}

tutieshi_640x360_5s.gif

这篇关于【three.js】23. Raycaster and Mouse Events 投射射线(碰撞检测)和鼠标事件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/964350

相关文章

龙蜥操作系统Anolis OS-23.x安装配置图解教程(保姆级)

《龙蜥操作系统AnolisOS-23.x安装配置图解教程(保姆级)》:本文主要介绍了安装和配置AnolisOS23.2系统,包括分区、软件选择、设置root密码、网络配置、主机名设置和禁用SELinux的步骤,详细内容请阅读本文,希望能对你有所帮助... ‌AnolisOS‌是由阿里云推出的开源操作系统,旨

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

使用Vue.js报错:ReferenceError: “Vue is not defined“ 的原因与解决方案

《使用Vue.js报错:ReferenceError:“Vueisnotdefined“的原因与解决方案》在前端开发中,ReferenceError:Vueisnotdefined是一个常见... 目录一、错误描述二、错误成因分析三、解决方案1. 检查 vue.js 的引入方式2. 验证 npm 安装3.

Python中的异步:async 和 await以及操作中的事件循环、回调和异常

《Python中的异步:async和await以及操作中的事件循环、回调和异常》在现代编程中,异步操作在处理I/O密集型任务时,可以显著提高程序的性能和响应速度,Python提供了asyn... 目录引言什么是异步操作?python 中的异步编程基础async 和 await 关键字asyncio 模块理论

JS常用组件收集

收集了一些平时遇到的前端比较优秀的组件,方便以后开发的时候查找!!! 函数工具: Lodash 页面固定: stickUp、jQuery.Pin 轮播: unslider、swiper 开关: switch 复选框: icheck 气泡: grumble 隐藏元素: Headroom

禁止平板,iPad长按弹出默认菜单事件

通过监控按下抬起时间差来禁止弹出事件,把以下代码写在要禁止的页面的页面加载事件里面即可     var date;document.addEventListener('touchstart', event => {date = new Date().getTime();});document.addEventListener('touchend', event => {if (new

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

安卓链接正常显示,ios#符被转义%23导致链接访问404

原因分析: url中含有特殊字符 中文未编码 都有可能导致URL转换失败,所以需要对url编码处理  如下: guard let allowUrl = webUrl.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {return} 后面发现当url中有#号时,会被误伤转义为%23,导致链接无法访问

Node.js学习记录(二)

目录 一、express 1、初识express 2、安装express 3、创建并启动web服务器 4、监听 GET&POST 请求、响应内容给客户端 5、获取URL中携带的查询参数 6、获取URL中动态参数 7、静态资源托管 二、工具nodemon 三、express路由 1、express中路由 2、路由的匹配 3、路由模块化 4、路由模块添加前缀 四、中间件

韦季李输入法_输入法和鼠标的深度融合

在数字化输入的新纪元,传统键盘输入方式正悄然进化。以往,面对实体键盘,我们常需目光游离于屏幕与键盘之间,以确认指尖下的精准位置。而屏幕键盘虽直观可见,却常因占据屏幕空间,迫使我们在操作与视野间做出妥协,频繁调整布局以兼顾输入与界面浏览。 幸而,韦季李输入法的横空出世,彻底颠覆了这一现状。它不仅对输入界面进行了革命性的重构,更巧妙地将鼠标这一传统外设融入其中,开创了一种前所未有的交互体验。 想象