BabylonJS 6.0文档 Deep Dive 摄像机(二):摄像机碰撞

摄像机、网格碰撞和重力

你玩过第一人称射击游戏(FPS)吗?在本教程中,我们将模拟FPS的摄影机移动:摄影机位于地板上,与地面碰撞,并可能与场景中的任何对象碰撞。

如何实现?

为了实现这一功能,我们必须执行3个简单步骤:

1.定义并应用重力

首先要做的是定义重力矢量,定义重力。

Babylon.js的Scene类具有重力属性,可以应用于您之前在代码中定义的任何相机。这将沿指定的方向和速度移动摄影机(Vector3对象),除非摄影机的椭球体(参见下面的第2步)在checkCollisions设置为true的情况下与该方向上的另一个网格(如地面网格)发生碰撞。

Scene设置重力属性

scene.gravity = new BABYLON.Vector3(0, -0.15, 0);

摄像机开启重力属性

camera.applyGravity = true;

在现实世界中,重力是一种向下施加的力,即沿Y轴的负方向施加的力。在地球上,这一力大约为9.81m/s²。下落物体在下落时会加速,因此需要1秒才能完全达到这一速度,然后在2秒后速度达到19.62m/s,在3秒后达到29.43m/s... 在大气层中,风阻力将最终与这一力相匹配,自由落体速度也会停止增加(达到“终极速度”)。

然而,Babylon.js遵循了一个简单得多的引力模型 - scene.gravity表示恒定的速度,而不是加速度,它是以单位/帧而不是米/秒来测量的。渲染每个帧时,应用该重力的摄影机将沿每个轴移动矢量值(通常x和z设置为0,但您可以在任何方向上使用“重力”!),直到检测到碰撞。

虽然Babylon.js单元距离(unit)没有直接对应物理世界的东西,但在默认的相机视场下,1个单元距离=1米的近似值是一个相当标准的假设。因此,如果想要模拟地球重力,你需要对每秒渲染的帧数做出一些假设,并计算一个合适的矢量:

const assumedFramesPerSecond = 60;
const earthGravity = -9.81;
scene.gravity = new BABYLON.Vector3(0, earthGravity / assumedFramesPerSecond, 0);

由于这是每帧计算一次,所以相机实际上并没有“移动”,而是沿着重力矢量的方向进行微小的“跳跃”。如果您依靠碰撞检测来确定相机(或者更确切地说,是为此目的附着在相机上的网格)是否“进入”或“退出”了其他网格(例如,“地面”层下的一个平面,用于感测下落的角色并重置游戏),这一点可能很重要。根据您选择的重力、起始高程以及“触发”网格的位置和高度,相机可能会直接跳过触发网格,而不会与之“相交”。因此请务必检查Math类,以确保添加到起始高程的场景重力的至少一个倍数会与触发网格相交。

如果你需要更准确地表示引力(或其他力),你可以使用Babylon集成的物理引擎,或者添加自定义的物理引擎。

警告:

如在一对象上添加physicsimpostor(自定义的外部物理引擎)并同时启用碰撞(collision),可能会导致意外行为。

2. 定义椭圆球体

下一个重要步骤是定义相机周围的椭圆球体。这个椭圆球体代表了我们玩家的尺寸:当对象网格与这个椭圆球体接触时,会引发碰撞事件,防止我们的相机离这个对象网格太近:

Babylon.js相机上的椭圆球属性默认为尺寸大小为(0.5、1、0.5)。更改这些值会使您变高、变大、变小、变瘦。在下面的示例中,我们将使相机的椭圆球体比默认的更胖:

//Set the ellipsoid around the camera (e.g. your player's size)
camera.ellipsoid = new BABYLON.Vector3(1, 1, 1);

请注意,摄像机的椭圆球体应该是偏移的,以便是视点(眼睛)始终位于椭球体的顶部。

可以通过更新椭球体偏移特性来控制此行为。

计算方法如下:

finalPosition = position - vec3(0, ellipsoid.y, 0) + ellipsoidOffset

3. 应用碰撞

我们的最后一步是声明我们对感知场景中的碰撞感兴趣:

scene.collisionsEnabled = true;
camera.checkCollisions = true;

以及要声明哪些网格可能与我们的相机发生碰撞(box和ground):

ground.checkCollisions = true;
box.checkCollisions = true;

非常简单,以下是案例代码:

4.物体与物体碰撞

你可以通过对物体网格(Mesh)的ellipsoid属性和moveWithCollisions(速度)函数的使用来达到相同的效果。此函数将尝试根据给定的速度移动Mesh,并检查当前模型与激活了checkCollisions的所有模型之间是否产生了碰撞。

也可以使用mesh.ellipsoidOffse来移动网格上的椭球体(默认情况下,椭球体以网格为中心)。

代码如下:

const speedCharacter = 8;
const gravity = 0.15;
const character = Your mesh;

character.ellipsoid = new BABYLON.Vector3(0.5, 1.0, 0.5);
character.ellipsoidOffset = new BABYLON.Vector3(0, 1.0, 0);

const forwards = new BABYLON.Vector3(parseFloat(Math.sin(character.rotation.y)) / speedCharacter, gravity, parseFloat(Math.cos(character.rotation.y)) / speedCharacter);
forwards.negate();
character.moveWithCollisions(forwards);
// or
const backwards = new BABYLON.Vector3(parseFloat(Math.sin(character.rotation.y)) / speedCharacter, -gravity, parseFloat(Math.cos(character.rotation.y)) / speedCharacter);
character.moveWithCollisions(backwards);

案例如下:

弧形旋转摄像机(Arc Rotate Camera)的碰撞

ArcRotateCamera也可以检查碰撞,但碰撞发生时,该相机不会移动。

如要激活碰撞,只需将camera.checkCollisions设置为true。还可以定义碰撞半径:

camera.collisionRadius = new BABYLON.Vector3(0.5, 0.5, 0.5);

其他类型的碰撞

太好了,现在你可以开发一款真正的FPS游戏了!但也许你想知道一个网格何时与另一个网格碰撞?如果你对此感兴趣,你可以在这里学习:网格碰撞。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/302330.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

CodeWave智能开发平台--03--目标:应用创建--07供应商数据表格02

摘要 本文是网易数帆CodeWave智能开发平台系列的第10篇,主要介绍了基于CodeWave平台文档的新手入门进行学习,实现一个完整的应用,本文主要完成07供应商数据表格下 CodeWave智能开发平台的10次接触 CodeWave参考资源 网易数帆CodeWave开发…

Hello 2024(A~D,F1)

新年坐大牢 A - Wallet Exchange 题意:共有俩钱包,每回合从其中一个钱包中拿走一块钱,谁拿走最后一块钱谁赢。 思路:奇偶讨论即可。 // Problem: A. Wallet Exchange // Contest: Codeforces - Hello 2024 // URL: https://cod…

josef 约瑟 数字式时间继电器 JS14P 0~20S AC220V 通电延时

JS14P系列时间继电器 JS14P系列数字式时间继电器是JS14、JS20等的更新换代产品采用集成电路,数字按键开关预置,它具有体积小、重量轻、精度高、寿命长、通用性强等优点,适用于交流50Hz,电压380V及以下和直流220w以下的自动控制电…

Pycharm安装numpy库失败解决办法

一、出现错误(以matplotlib为例): 二、解决办法: 方法一(失败):PyCharm中有一个安装库的方法是:Settings>>Python Interpreter>>点击右侧的加号 第二个图 失败原因&am…

C++每日一练(14):对称矩阵的判定

题目描述 输入矩阵的行数&#xff0c;再依次输入矩阵的每行元素&#xff0c;判断该矩阵是否为对称矩阵&#xff0c;若矩阵对称输出“yes"&#xff0c;不对称输出”no“。 输入 第一行输入一个正整数N&#xff08;N<20&#xff09;&#xff0c;表示矩阵的行数&#xff0…

深入了解pnpm:一种高效的包管理工具

✨专栏介绍 在当今数字化时代&#xff0c;Web应用程序已经成为了人们生活和工作中不可或缺的一部分。而要构建出令人印象深刻且功能强大的Web应用程序&#xff0c;就需要掌握一系列前端技术。前端技术涵盖了HTML、CSS和JavaScript等核心技术&#xff0c;以及各种框架、库和工具…

Python笔记03-判断和循环

文章目录 比较运算符if-else语句while语句for循环循环中断 比较运算符 字面量True表示真&#xff0c;字面量False表示假 if-else语句 if语句判断条件的结果一定要是布尔类型 不要忘记判断条件后的&#xff1a; 归属于if语句的代码块&#xff0c;需在前方填充4个空格缩进 age…

批量整理记账,让你的财务生活更轻松

每个月&#xff0c;每季度&#xff0c;每年&#xff0c;都需要花费大量的时间来整理、核对账单和记录每一笔支出。这种繁琐的工作真的太费时、费力了&#xff01;但今天&#xff0c;我要给你介绍一个神奇的工具——晨曦记账本。 所需工具&#xff1a; 一个【晨曦记账本】软件…

常用类型_日期..

1.Date java.util.Date是开发中常用的日期处理类(并非java.sql.Date类) 现在这么一个需求&#xff1a; 就是获取当前时区的时间 public class Main{public static void main(String[] args) {// d1和d2表示的时间都是一样的 所以推荐使用第一种写法 比较简洁Date d1 new Da…

全新热门电商API接口,实现闲鱼商品详细搜索功能

近年来&#xff0c;电商行业蓬勃发展&#xff0c;API&#xff08;Application Programming Interface&#xff09;接口已经成为电商平台的重要组成部分。API接口联讯数据不仅可以实现平台间的数据交互&#xff0c;还可以为开发者提供丰富的功能&#xff0c;满足用户多样化的需求…

多模态推荐系统综述:五、挑战

五、挑战 1、Multimodal Recommender Systems: A Survey 2023 •通用解决方案。 值得注意的是&#xff0c;尽管针对模型中的不同阶段提出了一些方法[24]&#xff0c;但没有提供这些技术组合的最新通用解决方案。 •模型可解释性。 多模态模型的复杂性会使系统生成的建议难以…

【数据结构篇】数据结构中的 R 树和 B 树

数据结构中的 R 树和 B 树 ✔️关于R树&#xff08;RTree&#xff09;✔️什么是B树&#xff08;B-tree&#xff09;✔️B树和B树的区别✔️B树和B树在数据存储方面的具体差异 ✔️拓展知识仓✔️R树和B树的区别✔️ 那在内存消耗上有什么区别&#xff1f;✔️ R树有哪些优点和…

【数据库】mysql事务

一、事务的基本概念 1、事务的定义 事务可由一条非常简单的SQL语句组成&#xff0c;也可以由一组复杂的SQL语句组成。。 在 MySQL 中只有使用了 Innodb 数据库引擎的数据库或表才支持事务。事务处理可以用来维护数据库的完整性&#xff0c;保证成批的 SQL 语句要么全部执行&…

海外代理IP在游戏中有什么作用?

随着科技的飞速发展&#xff0c;手机和电脑等电子产品已成为互联网连接万物的重要工具&#xff0c;深度融入我们的日常生活&#xff0c;我们借助互联网完成工作、休闲和购物等任务&#xff0c;以求提升生活质量。 不仅如此&#xff0c;网络游戏也是人们心中最爱&#xff0c;它…

2024.1.8每日一题

LeetCode 回旋镖的数量 447. 回旋镖的数量 - 力扣&#xff08;LeetCode&#xff09; 题目描述 给定平面上 n 对 互不相同 的点 points &#xff0c;其中 points[i] [xi, yi] 。回旋镖 是由点 (i, j, k) 表示的元组 &#xff0c;其中 i 和 j 之间的距离和 i 和 k 之间的欧式…

Prometheus Blackbox_exporter笔记

一、安装Promtheus 在 Prometheus 官网 Download | Prometheus 获取适用于 Linux 的 Prometheus 安 装包&#xff0c;这里我选择最新的 2.46.0 版本&#xff0c;我是 Linux 系统&#xff0c;选择下载 prometheus-2.46.0.linux-amd64.tar.gz 下载安装包&#xff1a; wget htt…

UI自动化测试神器:RunnerGo

UI自动化测试已经成为现代软件开发过程中不可或缺的一部分。它能够提供诸多优势&#xff0c;包括提高测试效率、减少人力成本、提升软件质量等。同时&#xff0c;可视化工具为UI自动化测试带来了更多便利和灵活性。然而&#xff0c;可视化工具也存在一些潜在的劣势。本文将探讨…

读元宇宙改变一切笔记01_起源

1. 元宇宙是我们下一个生存之地 1.1. 1968年&#xff0c;只有不到10%的美国家庭拥有彩色电视&#xff0c;但当年票房排名第二位的电影《2001&#xff1a;太空漫游》&#xff08;2001: A Space Odyssey&#xff09;设想了这样的未来 1.1.1. 斯坦利库布里克(Stanley Kubrick) …

二 数据查询

1、实验目的 理解SQL成熟设计基本规范&#xff0c;熟练运用SQL语言实现数据基本查询&#xff0c;包括但表查询、分组统计查询和连接查询。 2、实验内容及要求 针对数据库设计各种单表查询SQL语句、分组统计查询语句&#xff1b;设计单个表针对自身的连接查询&#xff0c;设计…

lc 140. 单词拆分 II

回溯算法查询匹配单词 class Solution { public:unordered_map<string, int> word_map;void mapping(vector<string>& wordDict){for(auto &a : wordDict)word_map[a];}vector<string> ret;// s: 原始字符串// tmp: 已查询到的单词// …