文档章节

一个第三人称游戏相机的实现(基于rust语言和cgmath库)

_loop
 _loop
发布于 2017/07/04 22:56
字数 1124
阅读 14
收藏 0

    我在学校时就对角色扮演类3D游戏感兴趣,毕业那会研究过第三人称视角的游戏相机的实现(基于D3D),但由于没有想到合适的计算方法,最后实现出来的程序有BUG,并且没找出原因。

    最近看到git有rust写游戏的示例,于是就想再尝试解决这个问题,发现cgmath库做矩阵运算非常方便,顺便也找到一个自认为比较不错的解决办法。

    第三人称游戏相机的特点是,无论相机移动或旋转,视点始终在屏幕中心的某个点(不会作图,只能先语言描述了)。即在游戏角色不移动的情况下旋转视角,相机始终是绕着游戏玩家在一个球面上运动。cgmath提供了一个计算取景矩阵的函数:

pub fn look_at(eye: Point3<S>, center: Point3<S>, up: Vector3<S>) -> Matrix4<S>

该函数接受3个参数,eye表示眼睛的位置,即相机的位置,是3维空间的一个点,center表示视线的焦点,在游戏里可以认为是游戏玩家的位置,是3维空间的一个点,up表示相机向上的方向,是一个向量。游戏视角的旋转变化,实际上就是要计算这3个值。一般游戏视角旋转由鼠标控制,我们假设鼠标在屏幕x轴方向移动了x,y轴方向移动了y,我们来计算旋转后的取景矩阵(假设鼠标移动的距离与相机旋转的角度之间的系数为1,并且假设玩家不移动,简化问题)。

    为了计算这3个参数,我首先由 center-eye 得到一个向量,即相机的视线向量sight。旋转的过程中,center点的位置是不变的,所以我们可以先对sight向量绕center点作旋转,得到新的sight方向,然后再由center点沿着sight的反方向平移一个距离,就可以得到新的eye坐标。这里sight向量绕center点旋转,实际是分别绕了两个轴作了旋转:一个是过center点平行于坐标系y轴的轴,另一个是过center点平行于right轴的轴,right轴是垂直与sight向量,平行于xoz平面的方向向量,right轴可以由世界座标轴y轴与sight向量叉乘得到:

let world_up: Vector3<S> = Vector3::unit_y();
let right: Vector3<S> = world_up.cross(sight).normalize();

这里要注意world_up与sight的位置不能反,否则算出来的方向就反了。

在计算新的sight向量时,为了计算方便,可以先将center点平移到世界坐标原点来计算,这样就变成了绕y轴和right轴分别作旋转。向量绕y轴旋转函数:

let matrix_up = Matrix4::from_angle_y(Deg(x));//得到旋转矩阵
let sight_new = matrix_up.transform_vector(self.sight);//旋转
self.sight = sight_new.normalize();//规则化

绕right轴旋转实现:

let matrix_right = Matrix4::from_axis_angle(right, Deg(y));
let sight_new = matrix_right.transform_vector(self.sight);
self.sight = sight_new.normalize();

这里需要注意,如果x,y都不为0,则将matrix_up * matrix_right的结果用于对sight作旋转

let sight_new = matrix.transform_vector(self.sight);
self.sight = sight_new.normalize();

因为浮点运算有误差,计算次数越少越好。

    游戏玩家移动就简单了,直接对center点作平移,最后计算新的eye坐标:

let eye: Point3<S> = Point3::new(
    -(sight.x * length) + focus.x,
    -(sight.y * length) + focus.y,
    -(sight.z * length) + focus.z
);

up向量的计算可以直接用sight向量与right向量叉乘得到:

let up: Vector3<S> = sight.cross(right).normalize();

 可以看到相机的变换只依据两个量:sight向量与center坐标。这里认为eye与center点的距离是常量length,当然真正游戏里是一个可以控制的变量。

    最后计算矩阵(真正取景矩阵还需要与一个相机设置的矩阵相乘)。

let view = Matrix4::look_at(eye, point, up)

    个人极少写东西,写的不好还请见谅。我只是对3D图形这些东西很感兴趣,但是又没有个很好的途径去学习,走了很多弯路,就比如这个相机的实现,我之前怎么都找不到相关资料(关于游戏开发的资料本来就很少)。如果我写的这点东西能对同样对游戏感兴趣的新手有帮助的话,我也是非常开心的。欢迎大家留言,吐槽。

© 著作权归作者所有

共有 人打赏支持
_loop
粉丝 5
博文 2
码字总数 2536
作品 0
成都
程序员
Unity Cinemachine插件学习笔记,实现单目标和多目标之间切换

Unity Cinemachine插件学习笔记,实现单目标和多目标之间切换

martins1994
05/10
0
0
Unity 算法功能 日記

Hello ,I am KitStar 1. 相机根据玩家与玩家位置 进行智能跟随 方法 在使用相机跟随玩家对象的时候。往往会使用多种模式进行。本文将介绍两种方式。 其实,就是就是transform.LookAt 不同参数...

KiTok
2017/08/10
0
0
基于 Rust 的操作系统--Redox

Redox 是一个用 Rust 语言编写的类 UNIX 操作系统 , 它的目标是把 Rust 语言的创新带入到一个现代的微内核和全系列的应用程序。 特性: Rust 语言实现 微内核设计 包括可选的 GUI 程序 - Or...

翟志军
2015/09/30
3.4K
3
Redox 0.0.6 发布,基于 Rust 的操作系统

Redox 是一个用 Rust 语言编写的类 UNIX 操作系统 , 它的目标是把 Rust 语言的创新带入到一个现代的微内核和全系列的应用程序。 特性: Rust 语言实现 微内核设计 包括可选的 GUI 程序 - Or...

王练
2017/01/05
3.1K
22
使用 Rust 构建分布式 Key-Value Store

欢迎大家前往腾讯云社区,获取更多腾讯海量技术实践干货哦~ 引子 构建一个分布式 Key-Value Store 并不是一件容易的事情,我们需要考虑很多的问题,首先就是我们的系统到底需要提供什么样的功...

腾讯云社区
2017/11/20
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Go语言_通神之路(2)

1、包 每个Go程序都是由包构成,从main包开始运行,就是我上一篇讲到的,都是从main函数开始执行,但是必须在main包下面! package mainimport ( "fmt" "math/rand")func ...

木九天
昨天
1
0
51.php-fpm的pool 慢日志 open_basedir 进程管理

12.21 php-fpm的pool 12.22 php-fpm慢执行日志(测试时报错) 12.23 open_basedir 12.24 php-fpm进程管理 12.21 php-fpm的pool: php-fpm里的pool也叫池子,咱们之前加入过www的配置,这个w...

王鑫linux
昨天
0
0
java内存模型概述

1、Java虚拟机运行时数据分区图 程序计数器:线程私有,是一块较小的内存空间,它是当前线程所执行的字节码文件的行号指示器 java虚拟机栈:线程私有,其生命周期与线程相同,这也就是我们平...

京一
昨天
0
0
shell学习之test语法

因为if-then语句不能测试退出状态码之外的条件,所以提供了test, 如果test命令中列出的条件成立,test命令就会退出并返回退出状态码0;如果条件不成立,test命令就会退出并返回非零的退出状态...

woshixin
昨天
0
0
openJDK之如何下载各个版本的openJDK源码

如果我们需要阅读openJDK的源码,那么需要下载,那么该去哪下载呢? 现在JDK已经发展到版本10了,11已经处于计划中,如果需要特定版本的openJDK,它们的下载链接在哪呢? 1.openJDK的项目 链接...

汉斯-冯-拉特
昨天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部