www.pudn.com > simple_fragment_program2.zip > glh_convenience.h


/* 
    glh - is a platform-indepenedent C++ OpenGL helper library  
 
 
    Copyright (c) 2000 Cass Everitt 
	Copyright (c) 2000 NVIDIA Corporation 
    All rights reserved. 
 
    Redistribution and use in source and binary forms, with or 
	without modification, are permitted provided that the following 
	conditions are met: 
 
     * Redistributions of source code must retain the above 
	   copyright notice, this list of conditions and the following 
	   disclaimer. 
 
     * Redistributions in binary form must reproduce the above 
	   copyright notice, this list of conditions and the following 
	   disclaimer in the documentation and/or other materials 
	   provided with the distribution. 
 
     * The names of contributors to this software may not be used 
	   to endorse or promote products derived from this software 
	   without specific prior written permission.  
 
       THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 
	   ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 
	   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 
	   FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 
	   REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 
	   INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 
	   BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 
	   LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 
	   CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 
	   LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN 
	   ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE  
	   POSSIBILITY OF SUCH DAMAGE.  
 
 
    Cass Everitt - cass@r3.nu 
*/ 
 
#ifndef GLH_CONVENIENCE_H 
#define GLH_CONVENIENCE_H 
 
// Convenience methods for using glh_linear objects 
// with opengl... 
 
 
 
// debugging hack... 
#include  
 
using namespace std; 
 
#ifdef MACOS 
#include  
#else 
#include  
#endif 
 
#include  
#include  
 
 
namespace glh 
{ 
 
  // matrix helpers 
 
  inline matrix4f get_matrix(GLenum matrix)  
  { 
	GLfloat m[16]; 
	glGetFloatv(matrix, m); 
	return matrix4f(m); 
  } 
 
  // transform helpers 
 
  inline void glh_rotate(const quaternionf & r) 
  { 
	float angle; 
	vec3f axis; 
	r.get_value(axis, angle); 
	glRotatef(to_degrees(angle), axis.v[0], axis.v[1], axis.v[2]); 
  } 
 
  // inverse of camera_lookat 
  inline matrix4f object_lookat(const vec3f & from, const vec3f & to, const vec3f & Up) 
  { 
	  vec3f look = to - from; 
	  look.normalize(); 
	  vec3f up(Up); 
	  up -= look * look.dot(up); 
	  up.normalize(); 
	   
	  quaternionf r(vec3f(0,0,-1), vec3f(0,1,0), look, up); 
	  matrix4f m; 
	  r.get_value(m); 
	  m.set_translate(from); 
	  return m; 
  } 
 
 
  // inverse of object_lookat 
  inline matrix4f camera_lookat(const vec3f & eye, const vec3f & lookpoint, const vec3f & Up) 
  { 
	  vec3f look = lookpoint - eye; 
	  look.normalize(); 
	  vec3f up(Up); 
	  up -= look * look.dot(up); 
	  up.normalize(); 
 
	  matrix4f t; 
	  t.set_translate(-eye); 
 
	  quaternionf r(vec3f(0,0,-1), vec3f(0,1,0), look, up); 
	  r.invert(); 
	  matrix4f rm; 
	  r.get_value(rm); 
	  return rm*t;	   
  } 
 
 
  inline matrix4f frustum(float left, float right, 
				   float bottom, float top, 
				   float zNear, float zFar) 
  { 
	matrix4f m; 
	m.make_identity(); 
 
	m(0,0) = (2*zNear) / (right - left); 
	m(0,2) = (right + left) / (right - left); 
	 
	m(1,1) = (2*zNear) / (top - bottom); 
	m(1,2) = (top + bottom) / (top - bottom); 
	 
	m(2,2) = -(zFar + zNear) / (zFar - zNear); 
	m(2,3) = -2*zFar*zNear / (zFar - zNear); 
    
	m(3,2) = -1; 
	m(3,3) = 0; 
 
	return m; 
  } 
 
  inline matrix4f frustum_inverse(float left, float right, 
						   float bottom, float top, 
						   float zNear, float zFar) 
  { 
	matrix4f m; 
	m.make_identity(); 
 
	m(0,0) = (right - left) / (2 * zNear); 
	m(0,3) = (right + left) / (2 * zNear); 
	 
	m(1,1) = (top - bottom) / (2 * zNear); 
	m(1,3) = (top + bottom) / (2 * zNear); 
 
	m(2,2) = 0; 
	m(2,3) = -1; 
	 
	m(3,2) = -(zFar - zNear) / (2 * zFar * zNear); 
	m(3,3) = (zFar + zNear) / (2 * zFar * zNear); 
 
	return m; 
  } 
 
  inline matrix4f perspective(float fovy, float aspect, float zNear, float zFar) 
  { 
	double tangent = tan(to_radians(fovy/2.0f)); 
	float y = (float)tangent * zNear; 
	float x = aspect * y; 
	return frustum(-x, x, -y, y, zNear, zFar); 
  } 
 
  inline matrix4f perspective_inverse(float fovy, float aspect, float zNear, float zFar) 
  { 
	double tangent = tan(to_radians(fovy/2.0f)); 
	float y = (float)tangent * zNear; 
	float x = aspect * y; 
	return frustum_inverse(-x, x, -y, y, zNear, zFar); 
  } 
 
 
 
  // are these names ok? 
 
  inline void set_texgen_planes(GLenum plane_type, const matrix4f & m) 
  { 
	  GLenum coord[] = {GL_S, GL_T, GL_R, GL_Q }; 
	  for(int i = 0; i < 4; i++) 
          { 
          vec4f row; 
          m.get_row(i,row); 
		  glTexGenfv(coord[i], plane_type, row.v); 
          } 
  } 
 
  // handy for register combiners 
  inline vec3f range_compress(const vec3f & v) 
  { vec3f vret(v); vret *= .5f; vret += .5f; return vret; } 
   
  inline vec3f range_uncompress(const vec3f & v) 
  { vec3f vret(v); vret -= .5f; vret *= 2.f; return vret; } 
 
} // namespace glh 
 
#endif