#include "graphics.hh" #include #include #include using namespace ACGL::OpenGL; Graphics::Graphics(glm::uvec2 windowSize, float nearPlane, float farPlane) { this->windowSize = windowSize; this->nearPlane = nearPlane; this->farPlane = farPlane; } Graphics::Graphics() { } void Graphics::init() { // construct VAO to give shader correct Attribute locations SharedArrayBuffer ab = SharedArrayBuffer(new ArrayBuffer()); ab->defineAttribute("aPosition", GL_FLOAT, 3); ab->defineAttribute("aTexCoord", GL_FLOAT, 2); ab->defineAttribute("aNormal", GL_FLOAT, 3); SharedVertexArrayObject vao = SharedVertexArrayObject(new VertexArrayObject()); vao->attachAllAttributes(ab); // look up all shader files starting with 'phong' and build a ShaderProgram from it: lightingShader = ShaderProgramCreator("phong").attributeLocations( vao->getAttributeLocations()).create(); depthShader = ShaderProgramCreator("depth") .attributeLocations(vao->getAttributeLocations()).create(); depthTexture_near = SharedTexture2D( new Texture2D(windowSize, GL_DEPTH24_STENCIL8)); depthTexture_near->setMinFilter(GL_NEAREST); depthTexture_near->setMagFilter(GL_NEAREST); depthTexture_near->setWrapS(GL_CLAMP_TO_EDGE); depthTexture_near->setWrapT(GL_CLAMP_TO_EDGE); depthTexture_near->setCompareMode(GL_COMPARE_REF_TO_TEXTURE); framebuffer_near = SharedFrameBufferObject(new FrameBufferObject()); framebuffer_near->setDepthTexture(depthTexture_near); framebuffer_near->validate(); depthTexture_middle = SharedTexture2D( new Texture2D(windowSize, GL_DEPTH24_STENCIL8)); depthTexture_middle->setMinFilter(GL_NEAREST); depthTexture_middle->setMagFilter(GL_NEAREST); depthTexture_middle->setWrapS(GL_CLAMP_TO_EDGE); depthTexture_middle->setWrapT(GL_CLAMP_TO_EDGE); depthTexture_middle->setCompareMode(GL_COMPARE_REF_TO_TEXTURE); framebuffer_middle = SharedFrameBufferObject(new FrameBufferObject()); framebuffer_middle->setDepthTexture(depthTexture_middle); framebuffer_middle->validate(); depthTexture_far = SharedTexture2D( new Texture2D(windowSize, GL_DEPTH24_STENCIL8)); depthTexture_far->setMinFilter(GL_NEAREST); depthTexture_far->setMagFilter(GL_NEAREST); depthTexture_far->setWrapS(GL_CLAMP_TO_EDGE); depthTexture_far->setWrapT(GL_CLAMP_TO_EDGE); depthTexture_far->setCompareMode(GL_COMPARE_REF_TO_TEXTURE); framebuffer_far = SharedFrameBufferObject(new FrameBufferObject()); framebuffer_far->setDepthTexture(depthTexture_far); framebuffer_far->validate(); } GLFWwindow* Graphics::getWindow() { return window; } glm::uvec2 Graphics::getWindowSize() { return windowSize; } void Graphics::render(Level* level) { // render depth texture for sun // near pass framebuffer_near->bind(); glClear(GL_DEPTH_BUFFER_BIT); depthShader->use(); glm::vec3 sunVector = (level->getCameraCenter()->getPosition() + level->getDirectionalLight()->getPosition()); glm::mat4 depthViewProjectionMatrix_near = glm::ortho(-5, 5, -5, 5, -5, 5) * glm::lookAt(sunVector, level->getCameraCenter()->getPosition(), glm::vec3(0,1,0)); depthShader->setUniform("viewProjectionMatrix", depthViewProjectionMatrix_near); level->render(depthShader, false); if (!framebuffer_near->isFrameBufferObjectComplete()) { printf("Framebuffer incomplete, unknown error occured during shadow generation!\n"); } // middle pass framebuffer_middle->bind(); glClear(GL_DEPTH_BUFFER_BIT); glm::mat4 depthViewProjectionMatrix_middle = glm::ortho(-20, 20, -20, 20, -20, 20) * glm::lookAt(sunVector, level->getCameraCenter()->getPosition(), glm::vec3(0,1,0)); depthShader->setUniform("viewProjectionMatrix", depthViewProjectionMatrix_middle); level->render(depthShader, false); if (!framebuffer_middle->isFrameBufferObjectComplete()) { printf("Framebuffer incomplete, unknown error occured during shadow generation!\n"); } // far pass framebuffer_far->bind(); glClear(GL_DEPTH_BUFFER_BIT); glm::mat4 depthViewProjectionMatrix_far = glm::ortho(-farPlane/2.0f, farPlane/2.0f, -farPlane/2.0f, farPlane/2.0f, -farPlane/2.0f, farPlane/2.0f) * glm::lookAt(sunVector, level->getCameraCenter()->getPosition(), glm::vec3(0,1,0)); depthShader->setUniform("viewProjectionMatrix", depthViewProjectionMatrix_far); level->render(depthShader, false); if (!framebuffer_far->isFrameBufferObjectComplete()) { printf("Framebuffer incomplete, unknown error occured during shadow generation!\n"); } // final render pass glBindFramebuffer(GL_FRAMEBUFFER, 0); glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); lightingShader->use(); //set view and projection matrix glm::mat4 lightingViewProjectionMatrix = buildFrustum(75.0f, 0.1f, farPlane, (float)windowSize.x/(float)windowSize.y) * buildViewMatrix(level); lightingShader->setUniform("lightingViewProjectionMatrix", lightingViewProjectionMatrix); // convert texture to homogenouse coordinates glm::mat4 biasMatrix( 0.5, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.0, 0.0, 0.0, 0.5, 0.0, 0.5, 0.5, 0.5, 1.0 ); glm::mat4 depthBiasMVP_near = biasMatrix*depthViewProjectionMatrix_near; glm::mat4 depthBiasMVP_middle = biasMatrix*depthViewProjectionMatrix_middle; glm::mat4 depthBiasMVP_far = biasMatrix*depthViewProjectionMatrix_far; lightingShader->setUniform("shadowMVP_near", depthBiasMVP_near); lightingShader->setTexture("shadowMap_near", depthTexture_near, 1); lightingShader->setUniform("shadowMVP_middle", depthBiasMVP_middle); lightingShader->setTexture("shadowMap_middle", depthTexture_middle, 2); lightingShader->setUniform("shadowMVP_far", depthBiasMVP_far); lightingShader->setTexture("shadowMap_far", depthTexture_far, 3); //set lighting parameters if (level->getLights().size() > 0) { lightingShader->setUniform("lightCount", (int) level->getLights().size()); // TODO look into doing this less often // Build light position array glm::vec3 lightSources[level->getLights().size()]; for(unsigned int i = 0; igetLights().size(); i++) { lightSources[i] = level->getLights()[i].getPosition(); } glUniform3fv(lightingShader->getUniformLocation("lightSources"), sizeof(lightSources), (GLfloat*) lightSources); // Build light colour array glm::vec3 lightColours[level->getLights().size()]; for(unsigned int i = 0; igetLights().size(); i++) { lightColours[i] = level->getLights()[i].getColour(); } glUniform3fv(lightingShader->getUniformLocation("lightColors"), sizeof(lightColours), (GLfloat*) lightColours); // Build light attenuation array float lightIntensities[level->getLights().size()]; for(unsigned int i = 0; igetLights().size(); i++) { lightIntensities[i] = level->getLights()[i].getIntensity(); } glUniform1fv(lightingShader->getUniformLocation("lightIntensities"), sizeof(lightIntensities), (GLfloat*) lightIntensities); } // set directional Light if(level->getDirectionalLight()) { lightingShader->setUniform("directionalLightVector", level->getDirectionalLight()->getPosition()); lightingShader->setUniform("directionalColor", level->getDirectionalLight()->getColour()); lightingShader->setUniform("directionalIntensity", level->getDirectionalLight()->getIntensity()); } // set fog Parameters lightingShader->setUniform("fogEnd", (farPlane)-35.0f); lightingShader->setUniform("fogColor", level->getFogColor()); lightingShader->setUniform("cameraCenter", level->getCameraCenter()->getPosition()); // set Material Parameters lightingShader->setUniform("ambientColor", level->getAmbientLight()); lightingShader->setUniform("camera", level->getCameraPosition()); // render the level level->render(lightingShader, true); } void Graphics::resize(glm::uvec2 windowSize) { this->windowSize = windowSize; depthTexture_near->resize(glm::vec2(windowSize.x, windowSize.y)); depthTexture_middle->resize(glm::vec2(windowSize.x, windowSize.y)); depthTexture_far->resize(glm::vec2(windowSize.x, windowSize.y)); } glm::mat4 Graphics::buildFrustum( float phiInDegree, float _near, float _far, float aspectRatio) { float phiHalfInRadians = 0.5*phiInDegree * (M_PI/180.0); float top = _near * tan( phiHalfInRadians ); float bottom = -top; float left = bottom * aspectRatio; float right = -left; return glm::frustum(left, right, bottom, top, _near, _far); } glm::mat4 Graphics::buildViewMatrix(Level* level) { //construct lookAt (cameraPosition = cameraCenter + cameraVector return glm::lookAt((level->getCameraCenter()->getPosition() + level->getCamera()->getVector()), level->getCameraCenter()->getPosition(), glm::vec3(0.0f, 1.0f, 0.0f)); } float Graphics::getFarPlane() { return farPlane; } void Graphics::setGLFWHintsForOpenGLVersion( unsigned int _version ) { #ifdef __APPLE__ #if (ACGL_OPENGL_VERSION >= 30) // request OpenGL 3.2, will return a 4.1 context on Mavericks glfwWindowHint( GLFW_CONTEXT_VERSION_MAJOR, 3 ); glfwWindowHint( GLFW_CONTEXT_VERSION_MINOR, 2 ); glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE); glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE); #endif #else // non-apple glfwWindowHint( GLFW_CONTEXT_VERSION_MAJOR, _version / 10 ); glfwWindowHint( GLFW_CONTEXT_VERSION_MINOR, _version % 10 ); #ifdef ACGL_OPENGL_PROFILE_CORE glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE); glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE); #endif #endif } bool Graphics::createWindow() { // Initialise GLFW if ( !glfwInit() ) { ACGL::Utils::error() << "Failed to initialize GLFW" << std::endl; exit( -1 ); } // Configure OpenGL context setGLFWHintsForOpenGLVersion( ACGL_OPENGL_VERSION ); // activate multisampling (second parameter is the number of samples): //glfwWindowHint( GLFW_SAMPLES, 8 ); // request an OpenGL debug context: glfwWindowHint( GLFW_OPENGL_DEBUG_CONTEXT, true ); // define whether the window can get resized: glfwWindowHint(GLFW_RESIZABLE, true); // try to create an OpenGL context in a window and check the supported OpenGL version: // R,G,B,A, Depth,Stencil window = glfwCreateWindow(windowSize.x, windowSize.y, "SWP MarbleGame Group C", NULL, NULL); if (!getWindow()) { ACGL::Utils::error() << "Failed to open a GLFW window - requested OpenGL: " << ACGL_OPENGL_VERSION << std::endl; return false; } glfwMakeContextCurrent(window); ACGL::init(); return true; }