摄像机相关类的设计与实现

2024-06-21 11:38
文章标签 实现 设计 相关 摄像机

本文主要是介绍摄像机相关类的设计与实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、cCamera基类

SPE现在支持两种相机,模型预览和第一人称相机,它们都从cCamera类继承,cCamera类一来是提供所有摄像机的一些公共接口,二来是实现相机的一些公共功能。下面是它提供的一些关键的函数:

void LoadAllMatrix();void SetViewport(int vx,int vy,int width,int height);void SetProjParams(float fovY,float znear,float zfar);
void SetProjParams(ProjType prjtype,float left,float right,float bottom,float top,float znear,float zfar);void SetViewParams(const VECTOR3D &pos,const VECTOR3D &cent,const VECTOR3D &up){SetViewParams(pos.x,pos.y,pos.z,cent.x,cent.y,cent.z,up.x,up.y,up.z);
}
void SetViewParams(float x,float y,float z,float cx,float cy,float cz,float upx,float upy,float upz);// key & mouse handlers
virtual bool HandleMouse(MouseState state,int x,int y) {return false;}
virtual bool HandleMouseMove(MouseState state,int x,int y) {return false;}virtual void Update(float appTime,float elapsedTime) {}

上面的一些非虚函数实现了所有相机的公共功能,如设置视口,设置视图参数和投影参数,ProjType是自定义的枚举类型,指示正交投影或透视投影;其他三个虚函数是处理鼠标事件和每帧更新事件的,提供给子类实现;另外cCamra类也维护了一组成员,如对应的视图矩阵,投影矩阵,视点位置,以及视点坐标系的三根轴向量坐标等。参数设置好之后,调用LoadAllMatrix()这个函数使Camera生效,它会设定视口,更新当前的视图矩阵和投影矩阵,一般在绘制相关的几何体之前被调用。

2、模型预览(ModelView)摄像机

顾名思义,模型预览相机是用来360度观察三维模型的,这种相机有一个固定的注视点,然后它可以以该注视点为球心,以注视点到相机位置的矩离为半径的球面上自由旋转;旋转方向是由鼠标控制的。在SPE中cModelViewCamera类表示这种相机,主要有下面三个函数实现:

void Fit2Box(const VECTOR3D &min_, const VECTOR3D &max_);
virtual bool HandleMouse(MouseState state,int x,int y);
virtual bool HandleMouseMove(MouseState state,int x,int y);

它重写了父类鼠标处理的两个函数,并提供了一个新函数Fit2Box(),它的功能是让摄像机聚焦到某个包围盒;鼠标处理函数的作用是根据鼠标位置更新视图参数;关于如何更新,请看图1,黑色的点表示相机的聚焦点,红色的点表示视点,相机根据两次鼠标位置在x坐标的偏移来确定绕y轴旋转的角度,根据在y坐标的偏移来确定绕x轴旋转的角度;为了计算机新的视图参数,先要确定旋转轴,y轴即为(0,1,0)即camera的up方向,x轴为相机坐标系中的x轴(side方向,图1左);根据轴方向和角度可以计算出绕轴旋转矩阵,根据它可以计算机视点的新位置,之后调用基类的SetViewParams()重新设定相机参数即可。

图1:模型预览相机x,y旋转轴的确定,左:俯视图,右:前视图

实现中要注意的一个问题是,当绕x轴旋转时,look向量可能会与up向量(0,1,0)重合,这会导致视图矩阵计算错误,解决办法是检查新的look向量的y坐标是大于某个阈值,SPE设定为0.95,若大于则禁止本次操作。

3、第一人称(FirstPerson)相机

角色扮演游戏里经常用这种相机来跟踪角色,使角色总是处于视野中;这种相机把所有移动限制在xz平面,但可以左右上下观看;在SPE中用cFirstPersonCamera实现,由下面这几个关键函数组成:

void Update(float appTime, float elapsedTime);bool HandleMouseMove( MouseState state,int x,int y );
bool HandleMouse( MouseState state,int x,int y );

这三个函数都是对父类虚函数的重写,鼠标处理函数用来绕side轴旋转look向量(上下观看),Update函数每帧被调用,里面根据按键的状态来决定沿look轴前进后退,或者绕up轴旋转look向量(左右观看),其中side,up,look分别是视点坐标系的三根轴,由基类cCamera维护。

实现中有两个问题需要注意,一是up轴与look向量重合的问题,解决方法跟ModelView方法一样,二是如何使camera的移动速度跟帧率无关,这里是直接用固定的移动速度,然后一帧中移动的距离结合elapsedTime计算出来,这样在帧率越高时移动越平滑,但帧率太低时可能会有抖动现象。

 

这篇关于摄像机相关类的设计与实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.

Python+PyQt5实现多屏幕协同播放功能

《Python+PyQt5实现多屏幕协同播放功能》在现代会议展示、数字广告、展览展示等场景中,多屏幕协同播放已成为刚需,下面我们就来看看如何利用Python和PyQt5开发一套功能强大的跨屏播控系统吧... 目录一、项目概述:突破传统播放限制二、核心技术解析2.1 多屏管理机制2.2 播放引擎设计2.3 专

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

idea中创建新类时自动添加注释的实现

《idea中创建新类时自动添加注释的实现》在每次使用idea创建一个新类时,过了一段时间发现看不懂这个类是用来干嘛的,为了解决这个问题,我们可以设置在创建一个新类时自动添加注释,帮助我们理解这个类的用... 目录前言:详细操作:步骤一:点击上方的 文件(File),点击&nbmyHIgsp;设置(Setti

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

MySQL大表数据的分区与分库分表的实现

《MySQL大表数据的分区与分库分表的实现》数据库的分区和分库分表是两种常用的技术方案,本文主要介绍了MySQL大表数据的分区与分库分表的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有... 目录1. mysql大表数据的分区1.1 什么是分区?1.2 分区的类型1.3 分区的优点1.4 分

一文详解如何从零构建Spring Boot Starter并实现整合

《一文详解如何从零构建SpringBootStarter并实现整合》SpringBoot是一个开源的Java基础框架,用于创建独立、生产级的基于Spring框架的应用程序,:本文主要介绍如何从... 目录一、Spring Boot Starter的核心价值二、Starter项目创建全流程2.1 项目初始化(

Mysql删除几亿条数据表中的部分数据的方法实现

《Mysql删除几亿条数据表中的部分数据的方法实现》在MySQL中删除一个大表中的数据时,需要特别注意操作的性能和对系统的影响,本文主要介绍了Mysql删除几亿条数据表中的部分数据的方法实现,具有一定... 目录1、需求2、方案1. 使用 DELETE 语句分批删除2. 使用 INPLACE ALTER T

MySQL INSERT语句实现当记录不存在时插入的几种方法

《MySQLINSERT语句实现当记录不存在时插入的几种方法》MySQL的INSERT语句是用于向数据库表中插入新记录的关键命令,下面:本文主要介绍MySQLINSERT语句实现当记录不存在时... 目录使用 INSERT IGNORE使用 ON DUPLICATE KEY UPDATE使用 REPLACE

mysql数据库重置表主键id的实现

《mysql数据库重置表主键id的实现》在我们的开发过程中,难免在做测试的时候会生成一些杂乱无章的SQL主键数据,本文主要介绍了mysql数据库重置表主键id的实现,具有一定的参考价值,感兴趣的可以了... 目录关键语法演示案例在我们的开发过程中,难免在做测试的时候会生成一些杂乱无章的SQL主键数据,当我们