OpenGL判断一个点是否可见
关于OpenGL中判断一个点是否可见,可以分成两种情况讨论:点在2D空间中和3D空间中的时候。并且“在2D空间中”可以看作“在3D空间中”的特殊情况。
温馨提示:
1. 以下讨论以现代OpenGL为基础,如果您只会旧OpenGL,请先学习一些与本文相关的现代OpenGL知识。
2. 文中除vertex shader以外的代码可以认为是GLSL。
2D空间
如果是2D空间,就很好判断了。这里再分成几种情况。
1. 已知原始顶点坐标,没有矩阵变换
设有如下vertex shader:
#version 330 core
layout(location = 0) in vec4 position; // position.z为0.0,position.w为1.0
void main()
{
gl_Position = position;
}
将position记为v,由于NDC坐标系统只渲染x、y和z位于[-1,1]区间的点(这里z为0.0),因此判断方法如下:
bool visible(vec2 v)
{
return v.x >= -1.0 && v.x <= 1.0 && v.y >= -1.0 && v.y <= 1.0;
}
2. 已知顶点数据,有矩阵变换
设有如下vertex shader:
#version 330 core
layout(location = 0) in vec4 position; // position.z为0.0,position.w为1.0
uniform mat4 modelViewProjection;
void main()
{
gl_Position = modelViewProjection * position;
}
将position记为v,将modelViewProjection记为M,记v' = M * v,如果v'位于NDC坐标范围内就可见。但v'.w可能不为1,因此如果v'.x/v'.w和v'.y/v'.w位于[-1,1]就是可见的。判断方法如下:
bool visible(vec2 v, mat4 m)
{
vec4 result = m * vec4(v, 0.0, 1.0);
result.x /= result.w;
result.y /= result.w;
return result.x >= -1.0 && result.x <= 1.0 && result.y >= -1.0 && result.y <= 1.0;
}
或者合并一下:
bool visible(vec2 v, mat4 m)
{
vec4 result = m * vec4(v, 0.0, 1.0);
return result.x / result.w >= -1.0 && result.x / result.w <= 1.0 &&
result.y / result.w >= -1.0 && result.y / result.w <= 1.0;
}
3. 已知某点的像素坐标
设(x,y)为点v的像素坐标(不用管原点在左下角还是左上角),窗口大小为wsize(x和y分量分别表示宽和高),则判断方法如下:
bool visible(vec2 v, vec2 wsize)
{
return v.x >= && v.x < wsize.x && v.y >= && v.y < wsize.y;
}
到这里都很简单,因为是2D的。
那么3D呢?
3D空间
3D空间中这个事就要复杂许多了(也不多),我也很久都没想出来。最后发现是因为我对OpenGL的顶点变换不够熟悉。
1. 已知NDC坐标
首先,假设有一点p(x,y,z),已经在NDC坐标了,根据NDC坐标的知识,对它的判断应该是这样的:
bool visible(vec3 v)
{
return v.x >= -1.0 && v.x <= 1.0 && v.y >= -1.0 && v.y <= 1.0 && v.z >= -1.0 && v.z <= 1.0;
}
2. 直接判断
设有如下vertex shader:
#version 330 core
layout(location = 0) in vec4 position; // position.w为1.0
uniform mat4 modelViewProjection;
void main()
{
gl_Position = modelViewProjection * position;
}
将position记为v,将modelViewProjection记为M,将M * v记为v',v'在clip space(NDC)中。由于v'.w可能不为1,因此需要将v'.x、v'.y和v'.z除以v'.w,再判断v'.x、v'.y和v'.z是否在[-1,1]范围内。判断方法:
bool visible(vec3 v, mat4 m)
{
vec4 result = m * vec4(v, 1.0);
result.x /= result.w;
result.y /= result.w;
result.z /= result.w;
return result.x >= -1.0 && result.x <= 1.0 && result.y >= -1.0 && result.y <= 1.0 && result.z >= -1.0 && result.z <= 1.0;
}
或者合并一下:
bool visible(vec3 v, mat4 m)
{
vec4 result = m * vec4(v, 1.0);
return result.x / result.w >= -1.0 && result.x / result.w <= 1.0 &&
result.y / result.w >= -1.0 && result.y / result.w <= 1.0 &&
result.z / result.w >= -1.0 && result.z / result.w <= 1.0;
}
(现在想了想,这也不难,但是当时死活想不出来,大概是对OpenGL顶点变换了解地太少了吧……)
3. 根据frustum进行判断
发现这种方式就要感谢Google和Stack Overflow了,Google上搜opengl determine if object visible,第一个就是Stack Overflow。里面有人提供了一个链接(来自lighthouse3d,Index标题下方有一堆链接,有兴趣的可以自己看)。顺便再提供一个附加链接:http://www.lighthouse3d.com/tutorials/maths/。文章篇幅都比较长,这里就简单地翻译一部分,有部分更改。
(1) 平面的确定方法
首先要了解一些关于平面的知识。确定一个平面有多种方式,但最常用的方式是使用如下方程(A、B、C、D是常数):
Ax + By + Cz + D = 0 (1)
如果一个平面满足(1)式,为方便,将其称为平面Ax + By + Cz + D = 0。假设已知有3个点p0、p1和p2在同一平面上,A、B、C和D可以用下面方式确定:
1. 记u = p1 - p0,v = p2 - p0
2. 记n = u × v
3. 将n归一化(normalize)
4. 记n = (nx, ny, nz),则A = nx,B = ny,C = nz
5. 由(1)式可得D = -Ax - By - Cz (2)(此时A、B、C已算出)。设p(x,y,z)为平面上的任意一点,将坐标代入(2)式,即可算出D。如果将p用p0代替,则D = -n . p0
(2) 点到平面的距离
设有平面Ax + By + Cz + D = 0和点p(px,py,pz),记
d = Apx + Bpy + Cpz + D
或
d= n . p + D
则|d|为p到平面的距离。注意虽然|d|才是p到平面的距离,但d也是有用的——d的符号可以告诉我们点p在平面的哪一边。如果d > 0,则p在法线n所在的一边;如果d < 0,则p在另一边。当然如果d = 0,则说明p在平面上。这里暂且把d称为有向距离(signed distance)。
(3)
(TO BE CONTINUED)
本文可转载,转载请注明出处:http://www.cnblogs.com/collectionne/p/6750285.html
OpenGL判断一个点是否可见的更多相关文章
- Android 判断一个 View 是否可见 getLocalVisibleRect(rect) 与 getGlobalVisibleRect(rect)
Android 判断一个 View 是否可见 getLocalVisibleRect(rect) 与 getGlobalVisibleRect(rect) [TOC] 这两个方法的区别 View.ge ...
- [Modern OpenGL系列(三)]用OpenGL绘制一个三角形
本文已同步发表在CSDN:http://blog.csdn.net/wenxin2011/article/details/51347008 在上一篇文章中已经介绍了OpenGL窗口的创建.本文接着说如 ...
- excel批处理_判断一个名称是不是药品
把药品名称导入到sheet1的A字段 # -*- coding: utf-8 -*-"""Created on Fri Dec 9 09:38:58 2016判断一个名 ...
- 判断一个字符串str不为空的方法
1.str == null; 2."".equals(str); 3.str.length 4.str.isEmpty(); 注意:length是属性,一般集合类对象拥有的属性,取 ...
- Android 如何判断一个应用在运行(转)
Android 如何判断一个应用在运行 在一个应用中,或一个Service .Receiver中判断一个应用是否正在运行,以便进行一些相关的处理. 这个时候我们需要得到一个ActivityManag ...
- ZOJ Problem Set - 1331 Perfect Cubes 判断一个double是否为整数
zju对时间要求比较高,这就要求我们不能简单地暴力求解(三个循环搞定),就要换个思路:因为在循环时,已知a,确定b,c,d,在外重两层循环中已经给定了b和c,我们就不用遍历d,我们可以利用d^3=a^ ...
- PHP如何判断一个数组是一维数组或者是二维数组?用什么函数?
如题:如何判断一个数组是一维数组或者是二维数组?用什么函数? 判断数量即可 <?php if (count($array) == count($array, 1)) { echo '是一维数组' ...
- delphi 判断一个数组的长度用 Length 还是 SizeOf ?
判断一个数组的长度用 Length 还是 SizeOf ?最近发现一些代码, 甚至有一些专家代码, 在遍历数组时所用的数组长度竟然是 SizeOf(arr); 这不合适! 如果是一维数组.且元素大小是 ...
- 用jQuery判断一个元素的各种状态
用jQuery判断一个元素是否显示 用jQuery判断一个元素是否显示:$(element).is(":visible"); 类似的,判断一个元素是不是第一个子元素:$(ele ...
随机推荐
- golang kafka
golang kafka – hello world https://github.com/Shopify/sarama https://shopify.github.io/sarama/ con ...
- 从花式swap引出的pointer aliasing问题
上次,一个同学问我,你知不知道可以不用引入中间变量就可以实现swap? 我说,我知道,可以用加减法或者异或实现,像是这样 void mySwap(int &x,int &y) { x= ...
- 【正常向】CODEVS上分黄金
白银上分黄金失败=.= 在之前有很认真的写了一波排序,所以排序并不是很怂,还是那个理,现阶段学习的都是比较简单的排序,都是所谓的冒泡排序啊,桶排序这类,至于插排和选择排序,再往后又是什么快拍就很尬了. ...
- (原创)看我用各种姿势在手机和PC查看到连接到的wifi密码
今天一个女神来我家做客,她问我WiFi密码,然而我却奇迹般的忘记了(特么的当时心里一万个草泥马踏过去),让我在她面前尴尬求子的,所以为了防止你们也出现这种情况,我特地把各种方法整理了一下,那么感兴趣的 ...
- Java实现压缩文件与解压缩文件
由于工作需要,需要将zip的压缩文件进行解压,经过调查发现,存在两个开源的工具包,一个是Apache的ant工具包,另一个就是Java api自带的工具包:但是Java自带的工具包存在问题:如果压缩或 ...
- cas错误:org.jasig.cas.client.validation.TicketValidationException: No principal was found in the response from the CAS server.
这个问题困扰了我好几天,最终被这个哥们解决了,具体请参考:http://www.oschina.net/question/252484_149958?sort=time
- Attribute注解
class Program { static void Main(string[] args) { //Attribute注解,Attribute是附加到方法.属性.类等上面的特殊的标签,在类Type ...
- C++高精度模板
原文地址:http://blog.csdn.net/wall_f/article/details/8373395 原文只附代码,没有解析,本文增加了一些对代码的解释. 请注意:本模板不涉及实数运算与负 ...
- 简易版DES加密和解密详解
在DES密码里,是如何进行加密和解密的呢?这里采用DES的简易版来进行说明. 二进制数据的变换 由于不仅仅是DES密码,在其它的现代密码中也应用了二进制数据,所以无论是文章还是数字,都需要将明文变换为 ...
- SQL语法考核
--继上一篇MySQL的开发总结之后,适当的练习还是很有必要的-- SQL语法多变,不敢保证唯一,也不敢保证全对,如果错误欢迎指出,即刻修改. 一.现有表结构如下图 TABLENAME:afinfo ...