VR 全景模式OpenGL原理

VR 全景模式OpenGL原理

VR 全景模式原理

VR 全景模式原理将画面渲染到球面上,相当于从球心去观察内部球面,观察到的画面 360 度无死角,与普通播平面渲染的本质区别在渲染图像部分,画面渲染到一个矩形平面上,而全景需要将画面渲染到球面,利用 OpenGL 构建一个球体。OpenGL ES 中所有 3D 物体均是由三角形构成的,构建一个球体只需要利用球坐标系中的经度角、维度角以及半径计算出球面点的三维坐标,最后这些坐标点构成一个个小矩形,每个矩形就可以分成 2 个三角形。
在这里插入图片描述

纬度和经度的含义:

1、首先,纬度是地球表面上某一点与赤道之间的角度,取值范围为-90度到+90度。经度是地球表面上某一点与本初子午线之间的角度,取值范围为-180度到+180度。
2、将纬度和经度转换为弧度表示。OpenGL中的数学函数通常使用弧度作为单位,因此需要将纬度和经度从角度转换为弧度。可以使用以下公式进行转换:
弧度 = 角度 * π / 180
3、根据转换后的纬度和经度计算右手世界坐标。右手世界坐标系是OpenGL中常用的坐标系,其中x轴指向右侧,y轴指向上方,z轴指向观察者的反方向。
首先,根据纬度和经度计算球面上的点的坐标。可以使用以下公式:

  • x = cos(纬度) cos(经度)
  • y = sin(纬度)
  • z = cos(纬度) sin(经度)
x=r*cosθ*sinsβ
y=r*sinθ
y=r*cosθ*cosβ

在这里插入图片描述

球体网格实现

// 这个函数 `createSphere` 用于根据指定的参数在3D空间中生成一个球体网格。void VR_FullViewSphere3D::createSphere(float radius, int rings, int sectors, void *tag)
{// 定义必要的变量和常数float PI            = M3D_PI;float nowradius     = radius;int longtitude      = rings;int latitude        = sectors;float longtiRatio   = 1.0f;float latiRatio     = 2.0f;// 计算所需的顶点、纹理坐标和索引的总数int numPoints      = longtitude * (latitude + 1) * 3;int numTexcoords   = longtitude * (latitude + 1) * 2;int numIndices     = (longtitude - 1) * latitude * 6;// 为顶点数据、纹理坐标和索引分配内存m_vertexs1   = (float *)malloc(sizeof(float) * numPoints);m_texcoords1 = (float *)malloc(sizeof(float) * numTexcoords);m_indices1   = (short *)malloc(sizeof(short) * numIndices);int t = 0, v = 0, counter = 0;float theta = 0.0f, phi = 0.0f;// 生成顶点和纹理坐标for (int i = 0; i < longtitude; i++) {phi = (PI / 2 - i / (longtitude - 1 + 0.0) * PI) * longtiRatio;for (int j = 0; j < latitude + 1; j++) {theta = (j / (latitude + 0.0) * PI - PI / 2) * latiRatio;float r = -nowradius * (float)cosf(phi);float x = r * (float)sinf(theta);float y = nowradius * (float)sinf(phi);float z = r * (float)cosf(theta);// 分配顶点坐标m_vertexs1[v++] = x;  // X轴m_vertexs1[v++] = y;  // Y轴m_vertexs1[v++] = z;  // Z轴// 分配纹理坐标m_texcoords1[t++] = 1.0f - (float)((j + 0.0) / (latitude + 0.0)); // X轴m_texcoords1[t++] = (float)((i + 0.0) / (longtitude - 1.0));       // Y轴}}// 生成三角形的索引for (int i = 0; i < longtitude - 1; i++) {for (int j = 0; j < latitude; j++) {// 第一个三角形m_indices1[counter++] = (short)(i * (latitude + 1) + j);    // 上顶点m_indices1[counter++] = (short)(i * (latitude + 1) + j + 1); // 右上顶点m_indices1[counter++] = (short)((i + 1) * (latitude + 1) + j); // 下顶点// 第二个三角形m_indices1[counter++] = (short)((i + 1) * (latitude + 1) + j);m_indices1[counter++] = (short)(i * (latitude + 1) + j + 1);m_indices1[counter++] = (short)((i + 1) * (latitude + 1) + j + 1); // 右下顶点}}m_NumIndices = numIndices; // 存储生成的索引总数
}

这个方法根据给定的半径、环数和扇区数,在3D空间中创建一个球体网格。它计算用于渲染球体的顶点、纹理坐标和索引。

在计算球体的纬度角度(phi) 落在 [-π/2, π/2] 的范围时,也就是-90度到+90度,采用了如下的计算方式:

phi = (PI / 2 - i / (longtitude - 1 + 0.0) * PI) * longtiRatio;

这里的目的是为了在球体上均匀生成经线(经度)并控制其分布。详细解释如下:

  • (longtitude - 1 + 0.0):这里将(longtitude - 1)是类似计算机语言数组下标0开始,目的是为了避免整数相除后得到的结果被截断成整数。
  • i / (longtitude - 1 + 0.0):这一部分将当前经线编号i映射到一个范围在[0, 1]之间的值。当i=0时,表示顶端纬度,i=longtitude-1时表示底端纬度。
  • (PI / 2 - i / (longtitude - 1 + 0.0) * PI):根据上述得到的比例值,乘以π并减去π/2,可以将范围从[0, 1]映射到[π/2, -π/2]之间,即从顶端到底端的纬度范围。
  • longtiRatio:这个参数用于调节经度方向上的角度变化,影响经线所在的位置。

综上所述,这种计算方式能够确保在球体表面上沿着经线均匀生成点,并且通过调节longtiRatio参数,可以控制经线的分布密度或者改变球面形状。

在计算球体的经度角度(theta)落在 [-π, π] 的范围时,也就是-180度到+180度,采用了如下的计算方式:

theta = (j / (latitude + 0.0) * PI - PI / 2) * latiRatio;

这里的目的是为了在球体上均匀生成纬线(纬度)并控制其分布。详细解释如下:

  • (latitude + 0.0):将latitude转换为浮点数,以避免整数相除后结果被截断成整数。
  • j / (latitude + 0.0):将当前纬线编号j映射到一个范围在[0, 1]之间的值。当j=0时,表示经线从左侧开始,j=latitude时表示经线到达右侧。
  • (j / (latitude + 0.0) * PI - PI / 2):将上述比例乘以π并减去π/2,将范围从[0, 1]映射到[-π/2, π/2]之间,即从左侧到右侧的经线范围。
  • latiRatio:此参数用于调节纬度方向上的角度变化,影响纬线所在的位置,latiRatio=2实现-180度到+180度。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【代码随想录算法训练营Day34】860.柠檬水找零;406.根据身高重建队列;452.用最少数量的箭引爆气球

❇️Day 34 第八章 贪心算法 part04 ✴️今日任务 860.柠檬水找零406.根据身高重建队列452.用最少数量的箭引爆气球 ❇️860.柠檬水找零 本题看上好像挺难&#xff0c;其实挺简单的&#xff0c;大家先尝试自己做一做。题目链接&#xff1a;https://leetcode.cn/problems/lem…

即时设计-高效易用的界面工具

界面工具是设计师的得力助手&#xff0c;为设计师快速创建精美易用的用户界面提供了丰富的功能和直观的界面。在众多的界面工具中&#xff0c;有的支持预设模板、图标库和样式库&#xff0c;有的更注重原型和互动。如何选择优秀的界面工具&#xff1f;这里有一个高效易用的界面…

python并发 map函数的妙用

1.map是什么&#xff1f; map函数是Python中的一个内置函数&#xff0c;用于将一个函数应用到一个或多个可迭代对象的每个元素上&#xff0c;生成一个新的可迭代对象。它的一般形式是&#xff1a; map(function, iterable1, iterable2, ...)其中&#xff0c;function是一个函…

JS逆向进阶篇【去哪儿旅行登录】【下篇-逆向Bella参数JS加密逻辑Python生成】

目录&#xff1a; 每篇前言&#xff1a;引子——本篇目的1、 代码混淆和还原&#xff08;1&#xff09;单独替换&#xff1a;&#xff08;2&#xff09;整个js文件替换&#xff1a; 2、算法入口分析3、 深入分析&#xff08;0&#xff09;整体分析&#xff1a;&#xff08;1&am…

STM32启动过程及反汇编

STM32从Flash启动的过程&#xff0c;主要是从上电复位到main函数的过程&#xff0c;主要有以下步骤&#xff1a; 1.初始化堆栈指针 SP_initial_sp&#xff0c;初始化 PC 指针Reset_Handler 2.初始化中断向量表 3.配置系统时钟 4.调用 C 库函数_main 初始化用户堆栈&#xf…

华为配置WLAN高密业务示例

配置WLAN高密业务示例 组网图形 图1 配置高密WLAN环境网络部署组网图 业务需求组网需求数据规划配置思路配置注意事项操作步骤配置文件 业务需求 体育场由于需要接入用户数量很大&#xff0c;AP间部署距离较小&#xff0c;因此AP间的干扰较大&#xff0c;可能导致用户上网网…

Qt 中Qwidget相关属性

文章目录 1. QWidget 核心属性1.1 enabled1.2 geometry1.2.1 window frame 的影响 1.3 windowTitle1.4 windowIcon1.4.1 qrc的使用 1.5 windowOpacity1.6 cursor1.7 focusPolicy1.8 styleSheet 1. QWidget 核心属性 在 Qt 中, 使⽤ QWidget 类表⽰ “控件”. 像按钮, 视图, 输…

Shell管道和过滤器

一、Shell管道 Shell 还有一种功能&#xff0c;就是可以将两个或者多个命令&#xff08;程序或者进程&#xff09;连接到一起&#xff0c;把一个命令的输出作为下一个命令的输入&#xff0c;以这种方式连接的两个或者多个命令就形成了管道&#xff08;pipe&#xff09;。 重定…

应用方案 | D722 9MHz,轨对轨I/O CMOS运放,低噪声、低电压、低功耗运放,应用广泛

D722是低噪声、低电压、低功耗运放&#xff0c;应用广泛。D722具有9MHz的高增益带宽积&#xff0c;转换速率为8.5V/μs&#xff0c;静态电流为1.7mA&#xff08;5V电源电压&#xff09;。 D722具有低电压、低噪声的特点&#xff0c;并提供轨到轨输出能力&#xff0c;D722的最大…

Shell编程——条件测试(五)

在shell编程中&#xff0c;if语句本身不执行任何判断&#xff0c;它实际上接受一个程序作为参数&#xff0c;然后执行这个程序&#xff0c;并依据这个程序的返回值来判断是否执行相应的语句。 程序的返回值是0&#xff0c;则为真&#xff0c;反之则为假。 目录 test命令&…

SpringCloud搭建微服务之Consul服务注册与发现

1. Consul介绍 Consul是由HashiCorp公司使用Go语言开发的一款开源工具&#xff0c;主要用于实现分布式系统的服务发现和服务配置&#xff0c;其内置了服务注册与发现框架、分布式一致性协议实现、健康检查、Key-Value存储、多数据中心方案。Consul具有高可移植性&#xff0c;支…

《Redis 设计与实现》读书概要

注&#xff1a; 《Redis 设计与实现》一书基于 Redis 2.9 版本编写&#xff0c;部分内容已过时&#xff0c;过时之处本文会有所说明。本文为读书笔记&#xff0c;部分简单和日常使用较少的知识点未记录。原书网页版地址 https://redisbook.com/ 一、底层数据结构 SDS(Simple Dy…

654.最大二叉树

这段Java代码实现了一个名为Solution的类&#xff0c;其中包含两个方法&#xff1a;constructMaximumBinaryTree()和constructMaximumBinaryTree1()&#xff0c;目的是从给定的整数数组nums中构建出一个最大二叉树。以下是详细的注释说明&#xff1a; class Solution {// 主方…

某品零食交易平台设计与实现|基于springboot+ Mysql+Java的某品交易平台设计与实现(源码+数据库+文档+PPT)

目录 基于springboot MysqlJava的某品交易平台设计与实现 摘 要 系统详细设计 数据库设计 论文参考 源码获取 文末获取源码联系 基于springboot MysqlJava的某品交易平台设计与实现 摘 要 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的…

利用excel文件增量同步一个库的数据并自动校正两端数据库条数不一致

利用excel文件增量同步一个库的数据并自动校正两端数据库条数不一致 现在有sqlserver和mysql两个库上的表在进行同步&#xff0c;sqlserver上的是源表&#xff0c;mysql上是目标表。 我们就把sqlserver上的数据同步到mysql上 mysql 是没有数据的。 sqlserver的三个表只是创建了…

240多道!Go开发岗位面试题合集(含答案)

随着今年大环境的影响&#xff0c;找工作的人在今年达到顶峰&#xff0c;马上金九银十面试黄金季&#xff0c;很多人都想在这次面试中拿到心仪Offer。 数以百计的面试&#xff0c;为何迟迟无法顺利入职&#xff1f;能力无疑是至关重要的&#xff0c;可却有不少能力不比已入职的…

未来已来!AI大模型引领科技革命

未来已来&#xff01;AI大模型正以惊人的速度引领着科技革命。随着科技的发展&#xff0c;人工智能在各个领域展现出了非凡的能力和潜力&#xff0c;大模型更是成为了科技领域的明星。从自然语言处理到图像识别&#xff0c;从智能推荐到语音识别&#xff0c;大模型的应用正在改…

Linux-信号3_sigaction、volatile与SIGCHLD

文章目录 前言一、sigaction__sighandler_t sa_handler;__sigset_t sa_mask; 二、volatile关键字三、SIGCHLD方法一方法二 前言 本章内容主要对之前的内容做一些补充。 一、sigaction #include <signal.h> int sigaction(int signum, const struct sigaction *act,struc…

阿里云2核4G服务器支持多少人在线?多少钱?

阿里云2核4G服务器多少钱一年&#xff1f;2核4G5M带宽优惠价格199元一年&#xff0c;轻量应用服务器2核4G4M带宽165元一年&#xff0c;2核4G服务器30元3个月&#xff0c;可以在阿里云官方活动查看2核4G配置详细报价 https://t.aliyun.com/U/bLynLC 阿里云2核4G服务器价格 2核4G…

07. Nginx进阶-Nginx负载均衡

简介 负载均衡 什么是负载均衡&#xff1f; 负载均衡&#xff0c;英文名称为Load Balance&#xff0c;其含义就是指将负载&#xff08;工作任务&#xff09;进行平衡、分摊到多个操作单元上进行运行。 Nginx负载均衡 什么是Nginx负载均衡&#xff1f; Nginx负载均衡可以大…