// Based on SphereWorld4.cpp
#include <GLTools.h>
#include <GLShaderManager.h>
#include <GLFrustum.h>
#include <GLBatch.h>
#include <GLFrame.h>
#include <GLMatrixStack.h>
#include <GLGeometryTransform.h>
#include <StopWatch.h>
#include <math.h>
#include <math3d.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <iostream>
#include <fstream>
#include <sstream>
#include <iostream>
#include "md2model.h"
using namespace std;
//#ifdef __APPLE__ // Don't need on Windows
//#include <glut/glut.h>
//#else
#define FREEGLUT_STATIC
#include <GL/glut.h>
//#endif
// ** GLOBALS **
GLShaderManager shaderManager; // Shader Manager
GLMatrixStack modelViewMatrix; // Modelview Matrix
GLMatrixStack projectionMatrix; // Projection Matrix
GLFrustum viewFrustum; // View Frustum
GLGeometryTransform transformPipeline; // Geometry Transform Pipeline
GLBatch floorBatch;
GLBatch backWallBatch;
GLBatch frontWallBatch;
GLBatch leftWallBatch;
GLBatch rightWallBatch;
GLFrame cameraFrame;
GLFrame modelFrame;
md2model model1;
GLuint currentAnim = MD2_STAND;
// Skybox objects
GLuint textures[10];
GLBatch boxPosX;
GLBatch boxNegX;
GLBatch boxPosY;
GLBatch boxNegY;
GLBatch boxPosZ;
GLBatch boxNegZ;
// ** FUNCTIONS **
//////////////////////////////////////////////////////////////////
// TEXTURE LOADER
GLuint loadTexture(char *fname, GLuint *texID)
{
// generate texture ID
glGenTextures(1, texID);
// texture dimensions and data buffer
GLint width, height, components;
GLenum format;
GLbyte *pBytes;
// load file - using GLTools library
pBytes = gltReadTGABits(fname, &width, &height, &components, &format);
if (pBytes == NULL)
{
cerr << "Something went wrong loading texture " << fname << endl;
}
// bind texture and set parameters
glBindTexture(GL_TEXTURE_2D, *texID);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT); // GL_REPEAT
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
glTexImage2D(GL_TEXTURE_2D,0,components,width, height, 0,
format, GL_UNSIGNED_BYTE, pBytes);
//glGenerateMipmap(GL_TEXTURE_2D);
// texture loaded, free the temporary buffer
free(pBytes);
return *texID; // return value of texure ID - redundant
}
//////////////////////////////////////////////////////////////////
// This function does any needed initialization on the rendering
// context.
void SetupRC()
{
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
// Initialze Shader Manager
shaderManager.InitializeStockShaders();
glShadeModel (GL_SMOOTH);
glEnable(GL_DEPTH_TEST);
glEnable (GL_TEXTURE_2D);
//////////////////////////////////////////////////////////////////
// LOAD TEXTURES
textures[0] = loadTexture("pos_x.tga", &textures[0]);
textures[1] = loadTexture("neg_x.tga", &textures[1]);
textures[2] = loadTexture("pos_y.tga", &textures[2]);
textures[3] = loadTexture("neg_y.tga", &textures[3]);
textures[4] = loadTexture("pos_z.tga", &textures[4]);
textures[5] = loadTexture("neg_z.tga", &textures[5]);
textures[6] = loadTexture("grass.tga", &textures[6]);
textures[7] = loadTexture("brick.tga", &textures[7]);
textures[8] = loadTexture("skeleton.tga", &textures[8]);
if (!model1.ReadMD2Model ("skeleton.md2"))
exit (EXIT_FAILURE);
// END
//////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
// SKYBOX
boxPosX.Begin(GL_TRIANGLE_FAN,4,1);
boxPosX.MultiTexCoord2f(0,0.0f,0.0f);
boxPosX.Vertex3f(10.0f,-10.0f,-10.0f);
boxPosX.MultiTexCoord2f(0,1.0f,0.0f);
boxPosX.Vertex3f(10.0f,-10.0f,10.0f);
boxPosX.MultiTexCoord2f(0,1.0f,1.0f);
boxPosX.Vertex3f(10.0f,10.0f,10.0f);
boxPosX.MultiTexCoord2f(0,0.0f,1.0f);
boxPosX.Vertex3f(10.0f,10.0f,-10.0f);
boxPosX.End();
boxNegX.Begin(GL_TRIANGLE_FAN,4,1);
boxNegX.MultiTexCoord2f(0,0.0f,0.0f);
boxNegX.Vertex3f(-10.0f,-10.0f,10.0f);
boxNegX.MultiTexCoord2f(0,1.0f,0.0f);
boxNegX.Vertex3f(-10.0f,-10.0f,-10.0f);
boxNegX.MultiTexCoord2f(0,1.0f,1.0f);
boxNegX.Vertex3f(-10.0f,10.0f,-10.0f);
boxNegX.MultiTexCoord2f(0,0.0f,1.0f);
boxNegX.Vertex3f(-10.0f,10.0f,10.0f);
boxNegX.End();
boxPosY.Begin(GL_TRIANGLE_FAN,4,1);
boxPosY.MultiTexCoord2f(0,0.0f,0.0f);
boxPosY.Vertex3f(-10.0f,10.0f,-10.0f);
boxPosY.MultiTexCoord2f(0,1.0f,0.0f);
boxPosY.Vertex3f(10.0f,10.0f,-10.0f);
boxPosY.MultiTexCoord2f(0,1.0f,1.0f);
boxPosY.Vertex3f(10.0f,10.0f,10.0f);
boxPosY.MultiTexCoord2f(0,0.0f,1.0f);
boxPosY.Vertex3f(-10.0f,10.0f,10.0f);
boxPosY.End();
boxNegY.Begin(GL_TRIANGLE_FAN,4,1);
boxNegY.MultiTexCoord2f(0,0.0f,0.0f);
boxNegY.Vertex3f(-10.0f,-10.0f,10.0f);
boxNegY.MultiTexCoord2f(0,1.0f,0.0f);
boxNegY.Vertex3f(10.0f,-10.0f,10.0f);
boxNegY.MultiTexCoord2f(0,1.0f,1.0f);
boxNegY.Vertex3f(10.0f,-10.0f,-10.0f);
boxNegY.MultiTexCoord2f(0,0.0f,1.0f);
boxNegY.Vertex3f(-10.0f,-10.0f,-10.0f);
boxNegY.End();
boxPosZ.Begin(GL_TRIANGLE_FAN,4,1);
boxPosZ.MultiTexCoord2f(0,0.0f,0.0f);
boxPosZ.Vertex3f(10.0f,-10.0f,10.0f);
boxPosZ.MultiTexCoord2f(0,1.0f,0.0f);
boxPosZ.Vertex3f(-10.0f,-10.0f,10.0f);
boxPosZ.MultiTexCoord2f(0,1.0f,1.0f);
boxPosZ.Vertex3f(-10.0f,10.0f,10.0f);
boxPosZ.MultiTexCoord2f(0,0.0f,1.0f);
boxPosZ.Vertex3f(10.0f,10.0f,10.0f);
boxPosZ.End();
boxNegZ.Begin(GL_TRIANGLE_FAN,4,1);
boxNegZ.MultiTexCoord2f(0,0.0f,0.0f);
boxNegZ.Vertex3f(-10.0f,-10.0f,-10.0f);
boxNegZ.MultiTexCoord2f(0,1.0f,0.0f);
boxNegZ.Vertex3f(10.0f,-10.0f,-10.0f);
boxNegZ.MultiTexCoord2f(0,1.0f,1.0f);
boxNegZ.Vertex3f(10.0f,10.0f,-10.0f);
boxNegZ.MultiTexCoord2f(0,0.0f,1.0f);
boxNegZ.Vertex3f(-10.0f,10.0f,-10.0f);
boxNegZ.End();
// END
//////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
// FLOOR
GLfloat texSize = 10.0f;
floorBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
floorBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
floorBatch.Vertex3f(-20.0f, -0.41f, 20.0f);
floorBatch.MultiTexCoord2f(0, texSize, 0.0f);
floorBatch.Vertex3f(20.0f, -0.41f, 20.0f);
floorBatch.MultiTexCoord2f(0, texSize, texSize);
floorBatch.Vertex3f(20.0f, -0.41f, -20.0f);
floorBatch.MultiTexCoord2f(0, 0.0f, texSize);
floorBatch.Vertex3f(-20.0f, -0.41f, -20.0f);
floorBatch.End();
// END
//////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
// WALLS
GLfloat wallTexSizeH = 0.5f; // Height of texture
GLfloat wallTexSizeW = 20.0f; // Width of texture
// BACK
backWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
backWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
backWallBatch.Vertex3f(-20.0f, -0.41f, 20.0f);
backWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
backWallBatch.Vertex3f(20.0f, -0.41f, 20.0f);
backWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
backWallBatch.Vertex3f(20.0f, 0.41f, 20.0f);
backWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
backWallBatch.Vertex3f(-20.0f, 0.41f, 20.0f);
backWallBatch.End();
// FRONT
frontWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
frontWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
frontWallBatch.Vertex3f(-20.0f, -0.41f, -20.0f);
frontWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
frontWallBatch.Vertex3f(20.0f, -0.41f, -20.0f);
frontWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
frontWallBatch.Vertex3f(20.0f, 0.41f, -20.0f);
frontWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
frontWallBatch.Vertex3f(-20.0f, 0.41f, -20.0f);
frontWallBatch.End();
// LEFT
leftWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
leftWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
leftWallBatch.Vertex3f(-20.0f, -0.41f, 20.0f);
leftWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
leftWallBatch.Vertex3f(-20.0f, -0.41f, -20.0f);
leftWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
leftWallBatch.Vertex3f(-20.0f, 0.41f, -20.0f);
leftWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
leftWallBatch.Vertex3f(-20.0f, 0.41f, 20.0f);
leftWallBatch.End();
// RIGHT
rightWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
rightWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
rightWallBatch.Vertex3f(20.0f, -0.41f, -20.0f);
rightWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
rightWallBatch.Vertex3f(20.0f, -0.41f, 20.0f);
rightWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
rightWallBatch.Vertex3f(20.0f, 0.41f, 20.0f);
rightWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
rightWallBatch.Vertex3f(20.0f, 0.41f, -20.0f);
rightWallBatch.End();
// END
//////////////////////////////////////////////////////////////////
}
//////////////////////////////////////////////////////////////////
// Clean up model data after exiting
void Cleanup()
{
model1.FreeModel();
}
//////////////////////////////////////////////////////////////////
// Screen changes size or is initialized
void ChangeSize(int nWidth, int nHeight)
{
glViewport(0, 0, nWidth, nHeight);
// Create the projection matrix, and load it on the projection matrix stack
viewFrustum.SetPerspective(35.0f, float(nWidth)/float(nHeight), 1.0f, 100.0f);
projectionMatrix.LoadMatrix(viewFrustum.GetProjectionMatrix());
// Set the transformation pipeline to use the two matrix stacks
transformPipeline.SetMatrixStacks(modelViewMatrix, projectionMatrix);
}
//////////////////////////////////////////////////////////////////
// Called to draw scene
void RenderScene(void)
{
//
static float interp = 0.0;
static double current_time = 0;
static double last_time = 0;
float dt;
GLfloat vColour[] = { 1.0f, 1.0f, 1.0f, 1.0f};
GLfloat vLightPos[] = {0.0f, 2.0f, -1.0f};
//
// Color values
static GLfloat vTexColour[] = { 1.0f, 1.0f, 1.0f, 1.0f}; // white colour for accurate walls
static GLfloat vCylColour[] = { 0.2f, 1.0f, 1.0f, 1.0f}; // Cylinder colour
// Clear the color and depth buffers
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
//
last_time = current_time;
current_time = (double)glutGet (GLUT_ELAPSED_TIME) / 100.0;
/* Animate model from frames 0 to num_frames-1 */
interp += (current_time - last_time);
dt = (current_time - last_time);
//
// Save the current modelview matrix (the identity matrix)
modelViewMatrix.PushMatrix();
M3DMatrix44f mCamera;
cameraFrame.GetCameraMatrix(mCamera);
modelViewMatrix.PushMatrix(mCamera);
//cameraFrame.MoveUp(0.0f);
// Transform the light position into eye coordinates
//M3DVector4f vLightPos = { 0.0f, 10.0f, 5.0f, 1.0f };
M3DVector4f vLightEyePos;
m3dTransformVector4(vLightEyePos, vLightPos, mCamera);
//////////////////////////////////////////////////////////////////
// SKYBOX
modelViewMatrix.PushMatrix();
// Set up the MVP matrix for drawing the skybox
// Get the current camera facing, but ignore its position
cameraFrame.GetCameraMatrix(mCamera,true);
modelViewMatrix.LoadMatrix(mCamera);
// Now set shader - this step is where the MVP matrix
// is set for any rendering
shaderManager.UseStockShader(GLT_SHADER_SHADED,
transformPipeline.GetModelViewProjectionMatrix());
// Disable Depth Testing then draw, so skybox renders in the distance
glDisable(GL_DEPTH_TEST);
glBindTexture(GL_TEXTURE_2D,textures[0]);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
boxPosX.Draw();
glBindTexture(GL_TEXTURE_2D,textures[1]);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
boxNegX.Draw();
glBindTexture(GL_TEXTURE_2D,textures[2]);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
boxPosY.Draw();
glBindTexture(GL_TEXTURE_2D,textures[3]);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
boxNegY.Draw();
glBindTexture(GL_TEXTURE_2D,textures[4]);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
boxPosZ.Draw();
glBindTexture(GL_TEXTURE_2D,textures[5]);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
boxNegZ.Draw();
// Re-enable Depth Testing, so other objects load correctly in environment
glEnable(GL_DEPTH_TEST);
modelViewMatrix.PopMatrix();
// END
//////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
// FLOOR
modelViewMatrix.PushMatrix();
glBindTexture(GL_TEXTURE_2D, textures[6]);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
floorBatch.Draw();
modelViewMatrix.PopMatrix();
//END
//////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
// WALLS
// BACK
modelViewMatrix.PushMatrix();
glBindTexture(GL_TEXTURE_2D, textures[7]);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
backWallBatch.Draw();
modelViewMatrix.PopMatrix();
// FRONT
modelViewMatrix.PushMatrix();
glBindTexture(GL_TEXTURE_2D, textures[7]);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
frontWallBatch.Draw();
modelViewMatrix.PopMatrix();
// LEFT
modelViewMatrix.PushMatrix();
glBindTexture(GL_TEXTURE_2D, textures[7]);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
leftWallBatch.Draw();
modelViewMatrix.PopMatrix();
// RIGHT
modelViewMatrix.PushMatrix();
glBindTexture(GL_TEXTURE_2D, textures[7]);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
rightWallBatch.Draw();
modelViewMatrix.PopMatrix();
//END
//////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
// MODEL
modelViewMatrix.PushMatrix();
modelViewMatrix.MultMatrix(modelFrame);
modelFrame.SetOrigin(0.0f, 0.0f, -1.0f);
modelViewMatrix.Rotate(-90,1.0f,0.0f,0.0f); // turn upright
modelViewMatrix.Rotate(-90,0.0f,0.0f,1.0f); // turn to face
modelViewMatrix.Scale(0.005f,0.005f,0.005f);
shaderManager.UseStockShader(GLT_SHADER_TEXTURE_POINT_LIGHT_DIFF,
modelViewMatrix.GetMatrix(), projectionMatrix.GetMatrix(),
vLightPos, vColour, 0);
glBindTexture(GL_TEXTURE_2D,textures[8]);
model1.Animate (currentAnim, dt);
model1.RenderFrameItp();
modelViewMatrix.PopMatrix();
// END
//////////////////////////////////////////////////////////////////
// Restore the previous modleview matrix (the identity matrix)
modelViewMatrix.PopMatrix();
modelViewMatrix.PopMatrix();
//cameraFrame.SetOrigin(0.0f, 0.0f, 1.0f);
if (interp >= 1.0f)
interp = 0.0f;
// Do the buffer Swap
glutSwapBuffers();
// Tell GLUT to do it again
glutPostRedisplay();
}
void Keyboard(unsigned char key, int x, int y)
{
/* Escape */
//if (key == 27)
//exit (0);
//if (currentAnim >19)
//currentAnim = 0;
float linear = 0.1f;
float angular = float(m3dDegToRad(5.0f));
float CAMx, CAMy, CAMz;
CAMx = cameraFrame.GetOriginX();
CAMy = cameraFrame.GetOriginY();
CAMz = cameraFrame.GetOriginZ();
switch(key)
{
case('w'):
{
cameraFrame.MoveForward(linear);
modelFrame.MoveForward(linear);
currentAnim = MD2_RUN;
break;
}
case('s'):
{
cameraFrame.MoveForward(-linear);
modelFrame.MoveForward(-linear);
currentAnim = MD2_RUN;
break;
}
case('a'):
{
cameraFrame.RotateWorld(angular, 0.0f, 1.0f, 0.0f);
modelFrame.RotateLocalY(angular);
break;
}
case('d'):
{
cameraFrame.RotateWorld(-angular, 0.0f, 1.0f, 0.0f);
modelFrame.RotateLocalY(-angular);
break;
}
default:
currentAnim = MD2_STAND;
}
}
void KeyboardUp (unsigned char key, int x, int y)
{
currentAnim = MD2_STAND;
}
int main(int argc, char* argv[])
{
//gltSetWorkingDirectory(argv[0]); // Don't need on Windows.
glutInit(&argc, argv);
glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGB | GLUT_DEPTH);
glutInitWindowSize(1280,720);
glutCreateWindow("Equilibrium Alpha");
glutKeyboardFunc(Keyboard);
glutKeyboardUpFunc(KeyboardUp);
glutReshapeFunc(ChangeSize);
glutDisplayFunc(RenderScene);
GLenum err = glewInit();
if (GLEW_OK != err)
{
fprintf(stderr, "GLEW Error: %s\n", glewGetErrorString(err));
return 1;
}
atexit(Cleanup);
SetupRC();
glutMainLoop();
return 0;
}
// Based on SphereWorld4.cpp

#include <GLTools.h>
#include <GLShaderManager.h>
#include <GLFrustum.h>
#include <GLBatch.h>
#include <GLFrame.h>
#include <GLMatrixStack.h>
#include <GLGeometryTransform.h>
#include <StopWatch.h>

#include <math.h>
#include <math3d.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <iostream>
#include <fstream>
#include <sstream>

#include <iostream>
#include "md2model.h"

using namespace std;

//#ifdef __APPLE__  // Don't need on Windows
//#include <glut/glut.h>
//#else
#define FREEGLUT_STATIC
#include <GL/glut.h>
//#endif


// ** GLOBALS **

GLShaderManager		shaderManager;			// Shader Manager
GLMatrixStack		modelViewMatrix;		// Modelview Matrix
GLMatrixStack		projectionMatrix;		// Projection Matrix
GLFrustum			viewFrustum;			// View Frustum
GLGeometryTransform	transformPipeline;		// Geometry Transform Pipeline

GLBatch				floorBatch;
GLBatch				backWallBatch;
GLBatch				frontWallBatch;
GLBatch				leftWallBatch;
GLBatch				rightWallBatch;
GLFrame             cameraFrame;
GLFrame				modelFrame;
md2model			model1;
GLuint				currentAnim = MD2_STAND;

// Skybox objects
GLuint textures[10];
GLBatch		boxPosX;
GLBatch		boxNegX;
GLBatch		boxPosY;
GLBatch		boxNegY;
GLBatch		boxPosZ;
GLBatch		boxNegZ;


// ** FUNCTIONS **

//////////////////////////////////////////////////////////////////
// TEXTURE LOADER
GLuint loadTexture(char *fname, GLuint *texID)
{
	// generate texture ID
	glGenTextures(1, texID);
	// texture dimensions and data buffer
	GLint width, height, components;
	GLenum format;
	GLbyte *pBytes;

	// load file - using GLTools library
    pBytes = gltReadTGABits(fname, &width, &height, &components, &format);
	if (pBytes == NULL)
	{
		cerr << "Something went wrong loading texture " << fname << endl;
	}

	// bind texture and set parameters
	glBindTexture(GL_TEXTURE_2D, *texID);
	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);  // GL_REPEAT
	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
	glTexImage2D(GL_TEXTURE_2D,0,components,width, height, 0,
		format, GL_UNSIGNED_BYTE, pBytes);
	//glGenerateMipmap(GL_TEXTURE_2D);

	// texture loaded, free the temporary buffer
	free(pBytes);

	return *texID;	// return value of texure ID - redundant

}       

//////////////////////////////////////////////////////////////////
// This function does any needed initialization on the rendering
// context. 
void SetupRC()
{
	glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
		
	// Initialze Shader Manager
	shaderManager.InitializeStockShaders();
	glShadeModel (GL_SMOOTH);
	glEnable(GL_DEPTH_TEST);
	glEnable (GL_TEXTURE_2D);

	//////////////////////////////////////////////////////////////////
	// LOAD TEXTURES
	textures[0] = loadTexture("pos_x.tga", &textures[0]);
	textures[1] = loadTexture("neg_x.tga", &textures[1]);
	textures[2] = loadTexture("pos_y.tga", &textures[2]);
	textures[3] = loadTexture("neg_y.tga", &textures[3]);
	textures[4] = loadTexture("pos_z.tga", &textures[4]);
	textures[5] = loadTexture("neg_z.tga", &textures[5]);
	textures[6] = loadTexture("grass.tga", &textures[6]);
	textures[7] = loadTexture("brick.tga", &textures[7]);
	textures[8] = loadTexture("skeleton.tga", &textures[8]);
	if (!model1.ReadMD2Model ("skeleton.md2"))
		exit (EXIT_FAILURE);
	// END
	//////////////////////////////////////////////////////////////////

	//////////////////////////////////////////////////////////////////
	// SKYBOX
	boxPosX.Begin(GL_TRIANGLE_FAN,4,1);
	boxPosX.MultiTexCoord2f(0,0.0f,0.0f);
	boxPosX.Vertex3f(10.0f,-10.0f,-10.0f);
	boxPosX.MultiTexCoord2f(0,1.0f,0.0f);
	boxPosX.Vertex3f(10.0f,-10.0f,10.0f);
	boxPosX.MultiTexCoord2f(0,1.0f,1.0f);
	boxPosX.Vertex3f(10.0f,10.0f,10.0f);
	boxPosX.MultiTexCoord2f(0,0.0f,1.0f);
	boxPosX.Vertex3f(10.0f,10.0f,-10.0f);
	boxPosX.End();

	boxNegX.Begin(GL_TRIANGLE_FAN,4,1);
	boxNegX.MultiTexCoord2f(0,0.0f,0.0f);
	boxNegX.Vertex3f(-10.0f,-10.0f,10.0f);
	boxNegX.MultiTexCoord2f(0,1.0f,0.0f);
	boxNegX.Vertex3f(-10.0f,-10.0f,-10.0f);
	boxNegX.MultiTexCoord2f(0,1.0f,1.0f);
	boxNegX.Vertex3f(-10.0f,10.0f,-10.0f);
	boxNegX.MultiTexCoord2f(0,0.0f,1.0f);
	boxNegX.Vertex3f(-10.0f,10.0f,10.0f);
	boxNegX.End();

	boxPosY.Begin(GL_TRIANGLE_FAN,4,1);
	boxPosY.MultiTexCoord2f(0,0.0f,0.0f);
	boxPosY.Vertex3f(-10.0f,10.0f,-10.0f);
	boxPosY.MultiTexCoord2f(0,1.0f,0.0f);
	boxPosY.Vertex3f(10.0f,10.0f,-10.0f);
	boxPosY.MultiTexCoord2f(0,1.0f,1.0f);
	boxPosY.Vertex3f(10.0f,10.0f,10.0f);
	boxPosY.MultiTexCoord2f(0,0.0f,1.0f);
	boxPosY.Vertex3f(-10.0f,10.0f,10.0f);
	boxPosY.End();

	boxNegY.Begin(GL_TRIANGLE_FAN,4,1);
	boxNegY.MultiTexCoord2f(0,0.0f,0.0f);
	boxNegY.Vertex3f(-10.0f,-10.0f,10.0f);
	boxNegY.MultiTexCoord2f(0,1.0f,0.0f);
	boxNegY.Vertex3f(10.0f,-10.0f,10.0f);
	boxNegY.MultiTexCoord2f(0,1.0f,1.0f);
	boxNegY.Vertex3f(10.0f,-10.0f,-10.0f);
	boxNegY.MultiTexCoord2f(0,0.0f,1.0f);
	boxNegY.Vertex3f(-10.0f,-10.0f,-10.0f);
	boxNegY.End();

	boxPosZ.Begin(GL_TRIANGLE_FAN,4,1);
	boxPosZ.MultiTexCoord2f(0,0.0f,0.0f);
	boxPosZ.Vertex3f(10.0f,-10.0f,10.0f);
	boxPosZ.MultiTexCoord2f(0,1.0f,0.0f);
	boxPosZ.Vertex3f(-10.0f,-10.0f,10.0f);
	boxPosZ.MultiTexCoord2f(0,1.0f,1.0f);
	boxPosZ.Vertex3f(-10.0f,10.0f,10.0f);
	boxPosZ.MultiTexCoord2f(0,0.0f,1.0f);
	boxPosZ.Vertex3f(10.0f,10.0f,10.0f);
	boxPosZ.End();

	boxNegZ.Begin(GL_TRIANGLE_FAN,4,1);
	boxNegZ.MultiTexCoord2f(0,0.0f,0.0f);
	boxNegZ.Vertex3f(-10.0f,-10.0f,-10.0f);
	boxNegZ.MultiTexCoord2f(0,1.0f,0.0f);
	boxNegZ.Vertex3f(10.0f,-10.0f,-10.0f);
	boxNegZ.MultiTexCoord2f(0,1.0f,1.0f);
	boxNegZ.Vertex3f(10.0f,10.0f,-10.0f);
	boxNegZ.MultiTexCoord2f(0,0.0f,1.0f);
	boxNegZ.Vertex3f(-10.0f,10.0f,-10.0f);
	boxNegZ.End();
	// END
	//////////////////////////////////////////////////////////////////

	//////////////////////////////////////////////////////////////////
	// FLOOR
	GLfloat texSize = 10.0f;
	floorBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
	floorBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
	floorBatch.Vertex3f(-20.0f, -0.41f, 20.0f);
	
	floorBatch.MultiTexCoord2f(0, texSize, 0.0f);
    floorBatch.Vertex3f(20.0f, -0.41f, 20.0f);
	
	floorBatch.MultiTexCoord2f(0, texSize, texSize);
	floorBatch.Vertex3f(20.0f, -0.41f, -20.0f);
	
	floorBatch.MultiTexCoord2f(0, 0.0f, texSize);
	floorBatch.Vertex3f(-20.0f, -0.41f, -20.0f);
	floorBatch.End();
	// END
	//////////////////////////////////////////////////////////////////

	//////////////////////////////////////////////////////////////////
	// WALLS

	GLfloat wallTexSizeH = 0.5f;  // Height of texture
	GLfloat wallTexSizeW = 20.0f;  // Width of texture

	// BACK
	backWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
	backWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
	backWallBatch.Vertex3f(-20.0f, -0.41f, 20.0f);
	
	backWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
    backWallBatch.Vertex3f(20.0f, -0.41f, 20.0f);
	
	backWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
	backWallBatch.Vertex3f(20.0f, 0.41f, 20.0f);
	
	backWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
	backWallBatch.Vertex3f(-20.0f, 0.41f, 20.0f);
	backWallBatch.End();

	// FRONT
	frontWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
	frontWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
	frontWallBatch.Vertex3f(-20.0f, -0.41f, -20.0f);
	
	frontWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
    frontWallBatch.Vertex3f(20.0f, -0.41f, -20.0f);
	
	frontWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
	frontWallBatch.Vertex3f(20.0f, 0.41f, -20.0f);
	
	frontWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
	frontWallBatch.Vertex3f(-20.0f, 0.41f, -20.0f);
	frontWallBatch.End();

	// LEFT
	leftWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
	leftWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
	leftWallBatch.Vertex3f(-20.0f, -0.41f, 20.0f);
	
	leftWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
    leftWallBatch.Vertex3f(-20.0f, -0.41f, -20.0f);
	
	leftWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
	leftWallBatch.Vertex3f(-20.0f, 0.41f, -20.0f);
	
	leftWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
	leftWallBatch.Vertex3f(-20.0f, 0.41f, 20.0f);
	leftWallBatch.End();

	// RIGHT
	rightWallBatch.Begin(GL_TRIANGLE_FAN, 4, 1);
	rightWallBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
	rightWallBatch.Vertex3f(20.0f, -0.41f, -20.0f);
	
	rightWallBatch.MultiTexCoord2f(0, wallTexSizeW, 0.0f);
    rightWallBatch.Vertex3f(20.0f, -0.41f, 20.0f);
	
	rightWallBatch.MultiTexCoord2f(0, wallTexSizeW, wallTexSizeH);
	rightWallBatch.Vertex3f(20.0f, 0.41f, 20.0f);
	
	rightWallBatch.MultiTexCoord2f(0, 0.0f, wallTexSizeH);
	rightWallBatch.Vertex3f(20.0f, 0.41f, -20.0f);
	rightWallBatch.End();
	// END
	//////////////////////////////////////////////////////////////////
}

//////////////////////////////////////////////////////////////////
// Clean up model data after exiting
void Cleanup()
{
  model1.FreeModel();
}


//////////////////////////////////////////////////////////////////
// Screen changes size or is initialized
void ChangeSize(int nWidth, int nHeight)
{
	glViewport(0, 0, nWidth, nHeight);
	
    // Create the projection matrix, and load it on the projection matrix stack
	viewFrustum.SetPerspective(35.0f, float(nWidth)/float(nHeight), 1.0f, 100.0f);
	projectionMatrix.LoadMatrix(viewFrustum.GetProjectionMatrix());
    
    // Set the transformation pipeline to use the two matrix stacks 
	transformPipeline.SetMatrixStacks(modelViewMatrix, projectionMatrix);
}

//////////////////////////////////////////////////////////////////
// Called to draw scene
void RenderScene(void)
{
    //
	static float interp = 0.0;
	static double current_time = 0;
	static double last_time = 0;
	float dt;
	GLfloat vColour[] = { 1.0f, 1.0f, 1.0f, 1.0f};
	GLfloat vLightPos[] = {0.0f, 2.0f, -1.0f};
	//
	
	// Color values
	static GLfloat vTexColour[] = { 1.0f, 1.0f, 1.0f, 1.0f};  // white colour for accurate walls
	static GLfloat vCylColour[] = { 0.2f, 1.0f, 1.0f, 1.0f};  // Cylinder colour
	
	// Clear the color and depth buffers
	glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

	//
	last_time = current_time;
	current_time = (double)glutGet (GLUT_ELAPSED_TIME) / 100.0;

	/* Animate model from frames 0 to num_frames-1 */
	interp += (current_time - last_time);
	dt = (current_time - last_time);
	//

    // Save the current modelview matrix (the identity matrix)
	modelViewMatrix.PushMatrix();
    
    M3DMatrix44f mCamera;
    cameraFrame.GetCameraMatrix(mCamera);
    modelViewMatrix.PushMatrix(mCamera);
	//cameraFrame.MoveUp(0.0f);

	// Transform the light position into eye coordinates
    //M3DVector4f vLightPos = { 0.0f, 10.0f, 5.0f, 1.0f };
    M3DVector4f vLightEyePos;
    m3dTransformVector4(vLightEyePos, vLightPos, mCamera);

	//////////////////////////////////////////////////////////////////
	// SKYBOX

	modelViewMatrix.PushMatrix();

	// Set up the MVP matrix for drawing the skybox
	// Get the current camera facing, but ignore its position
	cameraFrame.GetCameraMatrix(mCamera,true);
	modelViewMatrix.LoadMatrix(mCamera);

	// Now set shader - this step is where the MVP matrix
	// is set for any rendering
	shaderManager.UseStockShader(GLT_SHADER_SHADED,
		transformPipeline.GetModelViewProjectionMatrix());

	// Disable Depth Testing then draw, so skybox renders in the distance
	glDisable(GL_DEPTH_TEST);

	glBindTexture(GL_TEXTURE_2D,textures[0]);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
	boxPosX.Draw();

	glBindTexture(GL_TEXTURE_2D,textures[1]);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
	boxNegX.Draw();

	glBindTexture(GL_TEXTURE_2D,textures[2]);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
	boxPosY.Draw();

	glBindTexture(GL_TEXTURE_2D,textures[3]);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
	boxNegY.Draw();

	glBindTexture(GL_TEXTURE_2D,textures[4]);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
	boxPosZ.Draw();

	glBindTexture(GL_TEXTURE_2D,textures[5]);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
	boxNegZ.Draw();

	// Re-enable Depth Testing, so other objects load correctly in environment
	glEnable(GL_DEPTH_TEST);

	modelViewMatrix.PopMatrix();

	// END
	//////////////////////////////////////////////////////////////////

	//////////////////////////////////////////////////////////////////
	// FLOOR

	modelViewMatrix.PushMatrix();

	glBindTexture(GL_TEXTURE_2D, textures[6]);
	glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE, transformPipeline.GetModelViewProjectionMatrix(), 0);
	floorBatch.Draw();

	modelViewMatrix.PopMatrix();

	//END
	//////////////////////////////////////////////////////////////////

	//////////////////////////////////////////////////////////////////
	// WALLS

	// BACK

	modelViewMatrix.PushMatrix();

	glBindTexture(GL_TEXTURE_2D, textures[7]);
	glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
	backWallBatch.Draw();

	modelViewMatrix.PopMatrix();

	// FRONT

	modelViewMatrix.PushMatrix();

	glBindTexture(GL_TEXTURE_2D, textures[7]);
	glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
	frontWallBatch.Draw();

	modelViewMatrix.PopMatrix();

	// LEFT

	modelViewMatrix.PushMatrix();

	glBindTexture(GL_TEXTURE_2D, textures[7]);
	glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
	leftWallBatch.Draw();

	modelViewMatrix.PopMatrix();

	// RIGHT

	modelViewMatrix.PushMatrix();

	glBindTexture(GL_TEXTURE_2D, textures[7]);
	glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE, transformPipeline.GetModelViewProjectionMatrix(), vTexColour, 0);
	rightWallBatch.Draw();

	modelViewMatrix.PopMatrix();

	//END
	//////////////////////////////////////////////////////////////////

	//////////////////////////////////////////////////////////////////
	// MODEL
	modelViewMatrix.PushMatrix();
	modelViewMatrix.MultMatrix(modelFrame);
	modelFrame.SetOrigin(0.0f, 0.0f, -1.0f);
	modelViewMatrix.Rotate(-90,1.0f,0.0f,0.0f);   // turn upright
	modelViewMatrix.Rotate(-90,0.0f,0.0f,1.0f);   // turn to face
	modelViewMatrix.Scale(0.005f,0.005f,0.005f);
	shaderManager.UseStockShader(GLT_SHADER_TEXTURE_POINT_LIGHT_DIFF, 
		modelViewMatrix.GetMatrix(), projectionMatrix.GetMatrix(),
		vLightPos, vColour, 0);
	glBindTexture(GL_TEXTURE_2D,textures[8]);
	model1.Animate (currentAnim, dt);
	model1.RenderFrameItp();
	modelViewMatrix.PopMatrix();
	// END
	//////////////////////////////////////////////////////////////////

	// Restore the previous modleview matrix (the identity matrix)
	modelViewMatrix.PopMatrix();
    modelViewMatrix.PopMatrix();

	//cameraFrame.SetOrigin(0.0f, 0.0f, 1.0f);

	if (interp >= 1.0f)
		interp = 0.0f;

    // Do the buffer Swap
    glutSwapBuffers();
        
    // Tell GLUT to do it again
    glutPostRedisplay();
}

void Keyboard(unsigned char key, int x, int y)
{
  /* Escape */
  //if (key == 27)
    //exit (0);
  //if (currentAnim >19)
	  //currentAnim = 0;

  float linear = 0.1f;
	float angular = float(m3dDegToRad(5.0f));

	float CAMx, CAMy, CAMz;
	CAMx = cameraFrame.GetOriginX();
	CAMy = cameraFrame.GetOriginY();
	CAMz = cameraFrame.GetOriginZ();

	switch(key)
	{
	case('w'):
		{
			cameraFrame.MoveForward(linear);
			modelFrame.MoveForward(linear);
			currentAnim = MD2_RUN;
			break;
		}
	case('s'):
		{
			cameraFrame.MoveForward(-linear);
			modelFrame.MoveForward(-linear);
			currentAnim = MD2_RUN;
			break;
		}
	
	case('a'):
		{
			cameraFrame.RotateWorld(angular, 0.0f, 1.0f, 0.0f);
			modelFrame.RotateLocalY(angular);
			break;
		}
	
	case('d'):
		{
			cameraFrame.RotateWorld(-angular, 0.0f, 1.0f, 0.0f);
			modelFrame.RotateLocalY(-angular);
			break;
		}
	default:
		currentAnim = MD2_STAND;
	}
}

void KeyboardUp (unsigned char key, int x, int y)
{
	currentAnim = MD2_STAND;
}

int main(int argc, char* argv[])
{
	//gltSetWorkingDirectory(argv[0]);  // Don't need on Windows.
		
    glutInit(&argc, argv);
    glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGB | GLUT_DEPTH);
    glutInitWindowSize(1280,720);
  
    glutCreateWindow("Equilibrium Alpha");
 
    glutKeyboardFunc(Keyboard);
	glutKeyboardUpFunc(KeyboardUp);
    glutReshapeFunc(ChangeSize);
    glutDisplayFunc(RenderScene);
    
    GLenum err = glewInit();
    if (GLEW_OK != err) 
	{
		fprintf(stderr, "GLEW Error: %s\n", glewGetErrorString(err));
		return 1;
	}
        
	atexit(Cleanup);
    SetupRC();
    glutMainLoop();    
    return 0;
}