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=rcosθsinsβ
    y=rsinθ
    y=r
    cosθ*cosβ

球体网格实现

cpp 复制代码
// 这个函数 `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度,采用了如下的计算方式:

cpp 复制代码
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度,采用了如下的计算方式:

cpp 复制代码
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度。
相关推荐
BBTSOH159015160447 天前
VR每日热点简报2026.2.25
人工智能·机器人·vr·具身智能·遥操作
程序员敲代码吗7 天前
A-Frame与WebXR:构建丰富的VR及AR体验
ar·vr
BBTSOH159015160448 天前
VR每日热点简报2026.2.24
人工智能·meta·vr·虚拟现实·热点
子辰ToT10 天前
LearnOpenGL——PBR(三)漫反射辐照度
笔记·图形渲染·opengl
爱看书的小沐11 天前
【小沐学CAD】基于OCCT读取和显示STEP模型文件(QT、MFC、glfw)
qt·mfc·opengl·stp·step·opencascade·occt
kUhzIPVBnE12 天前
基于VSG并网仿真的转动惯量和阻尼自适应控制探索
vr
mtouch33312 天前
三维沙盘系统配置管理数字沙盘模块
人工智能·ai·ar·vr·虚拟现实·电子沙盘·数字沙盘
子辰ToT14 天前
LearnOpenGL——高级光照(七)HDR
笔记·图形渲染·opengl
犽戾武18 天前
在 Quest 上用 OpenXR + MediaCodec + OES 外部纹理做一个“低延迟视频面板”(48小时的编码复盘)
linux·c++·嵌入式硬件·vr
犽戾武18 天前
准备工作:OpenXR Sample 示例工程“去掉 UI 渲染”& RK3588→Windows 低延迟 UDP 视频链路
linux·c++·ubuntu·vr