AR/VR中使用Overlay提升清晰度

2024-05-30 04:18

本文主要是介绍AR/VR中使用Overlay提升清晰度,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在AR/VR应用中,清晰度是影响用户体验一个至关重要的因素,虽然目前提高清晰度的方案有很多:提高物理屏幕的分辨率,使用畸变网格进行畸变上屏等。但是Overlay感觉是在软件层面可以增加清晰度的一种很好的方式。

一,为什么要引入Overlay的实现。

考虑一个简单的场景,我现在要在AR/VR设备中呈现一个简单的场景(在正前方一块电影屏幕,并在上面播放电影):

正常的流程是新建一个三维场景,在正前方添加一个电影屏幕的网格,从内存中解析mp4格式的视频内容,并贴图到电影屏幕的三维网格上,然后通过渲染引擎渲染出左右眼看到的场景内容到缓存Buffer中(假设正好看向电影屏幕方向,此时电影屏幕也在视野中),然后我们会将缓存Buffer中的内容通过畸变上屏并呈现给用户(为什么不直接渲染出畸变后的场景内容并呈现到屏幕上?这涉及到目前AR/VR中的另一项技术ATW,俗称异步时间扭曲的一种插帧的技术,在帧率不足Vsync时使用缓存的视频帧进行插帧,所以需要将渲染和上屏进行分隔开来)。

但是这种实现会有两个弊端,一个是电影屏幕会受到场景中光照的影响,导致看到的电影内容比mp4中实际的内容看起来发白;第二个弊端是会涉及到两次像素采样,一次是将MP4中的电影内容采样到缓存Buffer中,第二次是将缓存Buffer中的内容畸变采样到屏幕上,这样直接导致了清晰度的下降。

Overlay的实现方式:并不会将电影屏幕的内容渲染到缓存Buffer中,缓存Buffer中只包含三维场景的背景内容,后续在畸变上屏的过程中,首先操作的是缓存Buffer,然后开启Blend功能后,直接将MP4的纹理也畸变上屏,此过程MP4的纹理只会产生一次像素采样,保持了高清晰度,效果图如下所示:
在这里插入图片描述
场景中左侧的面板是采用的Overlay技术,在清晰度上有明显的提升;而且没有受到场景光照的影响,右侧的面板明显颜色已经失真(发白)。

二, Overlay的实现原理

在这里主要以平面Overlay为例,对相关原理进行简单的记录,其他类型的Overlay原理基本相同。
Overlay的实现难点主要在如何直接将纹理畸变到屏幕的准确位置上:
正常的AR/VR上屏流程,畸变处理的缓存Buffer都是经过Model,View,Perspective矩阵渲染出来的场景内容,但是当电影屏幕的内容不在缓存Buffer上,而是直接从纹理到屏幕,并且跳过了Model,View,Perspective矩阵的处理时,这个过程该如何处理呢?
第一步我们需要计算屏幕上畸变网格的顶点,在采样时会不会落入电影屏幕的纹理上,如果落在电影屏幕的纹理范围内,这个顶点对应到纹理的那个UV坐标,作为这一步的计算输入我们需要电影屏幕在实际场景中的位置,大小,也就是Model矩阵,以及当前相机的姿态View矩阵。
第二步,需要计算Model View矩阵的逆:
MV_inverse = View_inverse * Model_inverse
使用MV_inverse 将畸变网格转换到模型坐标系中,然后使用转化到模型坐标系中的畸变网格顶点对电影屏幕纹理进行采样处理。
整个流程还是相对比较简单清晰的,下面是MV_inverse 矩阵的计算过程,以及Shader的处理代码:

// If a simple quad defined as a -1 to 1 XY unit square is transformed to
// the camera view with the given modelView matrix, it can alternately be
// drawn as a TimeWarp overlay image to take advantage of the full window
// resolution, which is usually higher than the eye buffer textures, and
// avoid resampling both into the eye buffer, and again to the screen.
// This is used for high quality movie screens and user interface planes.
//
// Note that this is NOT an MVP matrix -- the "projection" is handled
// by the distortion process.
//
// The exact composition of the overlay image and the base image is
// determined by the warpProgram, you may still need to draw the geometry
// into the eye buffer to punch a hole in the alpha channel to let the
// overlay/underlay show through.
//
// This utility functions converts a model-view matrix that would normally
// draw a -1 to 1 unit square to the view into a TanAngle matrix for an
// overlay surface.
//
// The resulting z value should be straight ahead distance to the plane.
// The x and y values will be pre-multiplied by z for projective texturing.
inline ovrMatrix4f TanAngleMatrixFromUnitSquare( const ovrMatrix4f * modelView )
{const ovrMatrix4f inv = ovrMatrix4f_Inverse( modelView );ovrMatrix4f m;m.M[0][0] = 0.5f * inv.M[2][0] - 0.5f * ( inv.M[0][0] * inv.M[2][3] - inv.M[0][3] * inv.M[2][0] );m.M[0][1] = 0.5f * inv.M[2][1] - 0.5f * ( inv.M[0][1] * inv.M[2][3] - inv.M[0][3] * inv.M[2][1] );m.M[0][2] = 0.5f * inv.M[2][2] - 0.5f * ( inv.M[0][2] * inv.M[2][3] - inv.M[0][3] * inv.M[2][2] );m.M[0][3] = 0.0f;m.M[1][0] = 0.5f * inv.M[2][0] + 0.5f * ( inv.M[1][0] * inv.M[2][3] - inv.M[1][3] * inv.M[2][0] );m.M[1][1] = 0.5f * inv.M[2][1] + 0.5f * ( inv.M[1][1] * inv.M[2][3] - inv.M[1][3] * inv.M[2][1] );m.M[1][2] = 0.5f * inv.M[2][2] + 0.5f * ( inv.M[1][2] * inv.M[2][3] - inv.M[1][3] * inv.M[2][2] );m.M[1][3] = 0.0f;m.M[2][0] = m.M[3][0] = inv.M[2][0];m.M[2][1] = m.M[3][1] = inv.M[2][1];m.M[2][2] = m.M[3][2] = inv.M[2][2];m.M[2][3] = m.M[3][3] = 0.0f;return m;
}

Shader相关代码:

//vertex shader"uniform mediump mat4 Mvpm;\n""uniform mediump mat4 Texm;\n""uniform mediump mat4 Texm2;\n""uniform mediump mat4 Texm3;\n""uniform mediump mat4 Texm4;\n""attribute vec4 Position;\n""attribute vec2 TexCoord;\n""attribute vec2 TexCoord1;\n""varying  vec2 oTexCoord;\n""varying  vec3 oTexCoord2;\n"	// Must do the proj in fragment shader or you get wiggles when you view the plane at even modest angles."void main()\n""{\n""   gl_Position = Mvpm * Position;\n""	vec3 proj;\n""	float projIZ;\n""""   proj = mix( vec3( Texm * vec4(TexCoord,-1,1) ), vec3( Texm2 * vec4(TexCoord,-1,1) ), TexCoord1.x );\n""	projIZ = 1.0 / max( proj.z, 0.00001 );\n""	oTexCoord = vec2( proj.x * projIZ, proj.y * projIZ );\n""""   oTexCoord2 = mix( vec3( Texm3 * vec4(TexCoord,-1,1) ), vec3( Texm4 * vec4(TexCoord,-1,1) ), TexCoord1.x );\n""""}\n"
//fragment shader"uniform sampler2D Texture0;\n""uniform sampler2D Texture1;\n""varying highp vec2 oTexCoord;\n""varying highp vec3 oTexCoord2;\n""void main()\n""{\n""	lowp vec4 color0 = texture2D(Texture0, oTexCoord);\n""	{\n""		lowp vec4 color1 = vec4( texture2DProj(Texture1, oTexCoord2).xyz, 1.0 );\n""		gl_FragColor = mix( color1, color0, color0.w );\n"	// pass through destination alpha"	}\n""}\n"

三, Overlay有哪些类型

目前Overlay支持的类型主要有四种Quad(平面面板),Equirect(360球形),Cylinder(柱面),Cube(天空盒),目前这四种Overlay都可以从技术层面进行相关的实现。

四, Overlay应用场景

Overlay主要应用在对场景清晰度要求较高的情形下,比如用户操作UI界面(可以极大提高说明文字的清晰度),360度图片展示,360视频播放,虚拟电影院中的荧幕部分。

这篇关于AR/VR中使用Overlay提升清晰度的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

使用Python开发一个带EPUB转换功能的Markdown编辑器

《使用Python开发一个带EPUB转换功能的Markdown编辑器》Markdown因其简单易用和强大的格式支持,成为了写作者、开发者及内容创作者的首选格式,本文将通过Python开发一个Markd... 目录应用概览代码结构与核心组件1. 初始化与布局 (__init__)2. 工具栏 (setup_t

Python虚拟环境终极(含PyCharm的使用教程)

《Python虚拟环境终极(含PyCharm的使用教程)》:本文主要介绍Python虚拟环境终极(含PyCharm的使用教程),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录一、为什么需要虚拟环境?二、虚拟环境创建方式对比三、命令行创建虚拟环境(venv)3.1 基础命令3

Python Transformer 库安装配置及使用方法

《PythonTransformer库安装配置及使用方法》HuggingFaceTransformers是自然语言处理(NLP)领域最流行的开源库之一,支持基于Transformer架构的预训练模... 目录python 中的 Transformer 库及使用方法一、库的概述二、安装与配置三、基础使用:Pi

关于pandas的read_csv方法使用解读

《关于pandas的read_csv方法使用解读》:本文主要介绍关于pandas的read_csv方法使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录pandas的read_csv方法解读read_csv中的参数基本参数通用解析参数空值处理相关参数时间处理相关

使用Node.js制作图片上传服务的详细教程

《使用Node.js制作图片上传服务的详细教程》在现代Web应用开发中,图片上传是一项常见且重要的功能,借助Node.js强大的生态系统,我们可以轻松搭建高效的图片上传服务,本文将深入探讨如何使用No... 目录准备工作搭建 Express 服务器配置 multer 进行图片上传处理图片上传请求完整代码示例

SpringBoot条件注解核心作用与使用场景详解

《SpringBoot条件注解核心作用与使用场景详解》SpringBoot的条件注解为开发者提供了强大的动态配置能力,理解其原理和适用场景是构建灵活、可扩展应用的关键,本文将系统梳理所有常用的条件注... 目录引言一、条件注解的核心机制二、SpringBoot内置条件注解详解1、@ConditionalOn

Python中使用正则表达式精准匹配IP地址的案例

《Python中使用正则表达式精准匹配IP地址的案例》Python的正则表达式(re模块)是完成这个任务的利器,但你知道怎么写才能准确匹配各种合法的IP地址吗,今天我们就来详细探讨这个问题,感兴趣的朋... 目录为什么需要IP正则表达式?IP地址的基本结构基础正则表达式写法精确匹配0-255的数字验证IP地

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Spring LDAP目录服务的使用示例

《SpringLDAP目录服务的使用示例》本文主要介绍了SpringLDAP目录服务的使用示例... 目录引言一、Spring LDAP基础二、LdapTemplate详解三、LDAP对象映射四、基本LDAP操作4.1 查询操作4.2 添加操作4.3 修改操作4.4 删除操作五、认证与授权六、高级特性与最佳