#include "Game.h" #include "Vertex.h" #include "Input.h" #include "SimpleShader.h" // Needed for a helper function to read compiled shader files from the hard drive #pragma comment(lib, "d3dcompiler.lib") #include #include #include "WICTextureLoader.h" // For the DirectX Math library using namespace DirectX; // -------------------------------------------------------- // Constructor // // DXCore (base class) constructor will set up underlying fields. // DirectX itself, and our window, are not ready yet! // // hInstance - the application's OS-level handle (unique ID) // -------------------------------------------------------- Game::Game(HINSTANCE hInstance) : DXCore( hInstance, // The application's handle "DirectX Game", // Text for the window's title bar 1280, // Width of the window's client area 720, // Height of the window's client area true), // Show extra stats (fps) in title bar? vsync(false) { #if defined(DEBUG) || defined(_DEBUG) // Do we want a console window? Probably only in debug mode CreateConsoleWindow(500, 120, 32, 120); printf("Console window created successfully. Feel free to printf() here.\n"); #endif camera = std::make_shared(0.0f, 0.0f, -20.0f, (float)width / height, 60, 0.01f, 1000.0f); } // -------------------------------------------------------- // Destructor - Clean up anything our game has created: // - Release all DirectX objects created here // - Delete any objects to prevent memory leaks // -------------------------------------------------------- Game::~Game() { // Note: Since we're using smart pointers (ComPtr), // we don't need to explicitly clean up those DirectX objects // - If we weren't using smart pointers, we'd need // to call Release() on each DirectX object created in Game } // -------------------------------------------------------- // Called once per program, after DirectX and the window // are initialized but before the game loop. // -------------------------------------------------------- void Game::Init() { // Helper methods for loading shaders, creating some basic // geometry to draw and some simple camera matrices. // - You'll be expanding and/or replacing these later LoadShaders(); LoadTextures(); LoadLighting(); CreateBasicGeometry(); // Tell the input assembler stage of the pipeline what kind of // geometric primitives (points, lines or triangles) we want to draw. // Essentially: "What kind of shape should the GPU draw with our data?" context->IASetPrimitiveTopology(D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST); } // -------------------------------------------------------- // Loads shaders from compiled shader object (.cso) files // and also created the Input Layout that describes our // vertex data to the rendering pipeline. // - Input Layout creation is done here because it must // be verified against vertex shader byte code // - We'll have that byte code already loaded below // -------------------------------------------------------- void Game::LoadShaders() { vertexShader = std::make_shared(device, context, GetFullPathTo_Wide(L"VertexShader.cso").c_str()); pixelShader = std::make_shared(device, context, GetFullPathTo_Wide(L"SimplePixelShader.cso").c_str()); XMFLOAT3 white = XMFLOAT3(1.0f, 1.0f, 1.0f); materials = { std::make_shared(white, 0, vertexShader, pixelShader), std::make_shared(white, 0, vertexShader, pixelShader), }; } void Game::LoadTextures() { Microsoft::WRL::ComPtr sampler; D3D11_SAMPLER_DESC sampDesc = {}; sampDesc.AddressU = D3D11_TEXTURE_ADDRESS_WRAP; sampDesc.AddressV = D3D11_TEXTURE_ADDRESS_WRAP; sampDesc.AddressW = D3D11_TEXTURE_ADDRESS_WRAP; sampDesc.Filter = D3D11_FILTER_ANISOTROPIC; sampDesc.MaxAnisotropy = 16; sampDesc.MaxLOD = D3D11_FLOAT32_MAX; device->CreateSamplerState(&sampDesc, sampler.GetAddressOf()); Microsoft::WRL::ComPtr deepFloorAlbedo, floorAlbedo; // taking the preprocessor macro from the demo because I don't like typing #define GetTex(pathToTexture, shaderResourceView) CreateWICTextureFromFile(device.Get(), context.Get(), GetFullPathTo_Wide(pathToTexture).c_str(), 0, shaderResourceView.GetAddressOf()); GetTex(L"Assets/Textures/HQGame/structure-endgame-deepfloor_albedo.png", deepFloorAlbedo); GetTex(L"Assets/Textures/HQGame/structure-endgame-floor_albedo.png", floorAlbedo); materials[0]->PushSampler("BasicSampler", sampler); materials[0]->PushTexture("Albedo", deepFloorAlbedo); materials[1]->PushSampler("BasicSampler", sampler); materials[1]->PushTexture("Albedo", floorAlbedo); } void Game::LoadLighting() { ambient = XMFLOAT3(0.1f, 0.1f, 0.25f); Light directionalLight0 = {}; directionalLight0.Type = LIGHT_TYPE_DIRECTIONAL; directionalLight0.Direction = XMFLOAT3(1, 0.5f, 0.5f); directionalLight0.Color = XMFLOAT3(1, 1, 1); directionalLight0.Intensity = 1.0f; Light directionalLight1 = {}; directionalLight1.Type = LIGHT_TYPE_DIRECTIONAL; directionalLight1.Direction = XMFLOAT3(-0.25f, -1, 0.75f); directionalLight1.Color = XMFLOAT3(1, 1, 1); directionalLight1.Intensity = 1.0f; Light directionalLight2 = {}; directionalLight2.Type = LIGHT_TYPE_DIRECTIONAL; directionalLight2.Direction = XMFLOAT3(-1, 1, -0.5f); directionalLight2.Color = XMFLOAT3(1, 1, 1); directionalLight2.Intensity = 1.0f; Light pointLight0 = {}; pointLight0.Type = LIGHT_TYPE_POINT; pointLight0.Position = XMFLOAT3(-1.5f, 0, 0); pointLight0.Color = XMFLOAT3(1, 1, 1); pointLight0.Intensity = 1.0f; pointLight0.Range = 10; Light pointLight1 = {}; pointLight1.Type = LIGHT_TYPE_POINT; pointLight1.Position = XMFLOAT3(1.5f, 0, 0); pointLight1.Color = XMFLOAT3(1, 1, 1); pointLight1.Intensity = 0.5f; pointLight1.Range = 10; lights = { directionalLight0, directionalLight1, directionalLight2, pointLight0, pointLight1, }; } // -------------------------------------------------------- // Creates the geometry we're going to draw - a single triangle for now // -------------------------------------------------------- void Game::CreateBasicGeometry() { shapes = { std::make_shared( GetFullPathTo("Assets/Models/cube.obj").c_str(), device, context), std::make_shared( GetFullPathTo("Assets/Models/cylinder.obj").c_str(), device, context), std::make_shared( GetFullPathTo("Assets/Models/helix.obj").c_str(), device, context), std::make_shared( GetFullPathTo("Assets/Models/sphere.obj").c_str(), device, context), std::make_shared( GetFullPathTo("Assets/Models/torus.obj").c_str(), device, context), std::make_shared( GetFullPathTo("Assets/Models/quad.obj").c_str(), device, context), std::make_shared( GetFullPathTo("Assets/Models/quad_double_sided.obj").c_str(), device, context), }; entities = { std::make_shared(materials[0], shapes[0]), std::make_shared(materials[0], shapes[1]), std::make_shared(materials[0], shapes[2]), std::make_shared(materials[1], shapes[3]), std::make_shared(materials[1], shapes[4]), std::make_shared(materials[1], shapes[5]), std::make_shared(materials[1], shapes[6]), }; for (int i = 0; i < entities.size(); ++i) { entities[i]->GetTransform()->SetPosition((-(int)(entities.size() / 2) + i) * 5, 0, 0); } } // -------------------------------------------------------- // Handle resizing DirectX "stuff" to match the new window size. // For instance, updating our projection matrix's aspect ratio. // -------------------------------------------------------- void Game::OnResize() { // Handle base-level DX resize stuff DXCore::OnResize(); // Ensure camera has its projection matrix updated when window size changes camera->SetAspect((float)width / height); } // -------------------------------------------------------- // Update your game here - user input, move objects, AI, etc. // -------------------------------------------------------- void Game::Update(float deltaTime, float totalTime) { // Example input checking: Quit if the escape key is pressed if (Input::GetInstance().KeyDown(VK_ESCAPE)) Quit(); camera->Update(deltaTime); for (int i = 0; i < entities.size(); ++i) { entities[i]->GetTransform()->SetRotation(sin(totalTime / 360) * 360, 0, 0); entities[i]->GetMaterial()->SetRoughness(sin(totalTime) * 0.5f + 0.49f); } } // -------------------------------------------------------- // Clear the screen, redraw everything, present to the user // -------------------------------------------------------- void Game::Draw(float deltaTime, float totalTime) { // Background color for clearing static const float color[4] = { 0.1f, 0.1f, 0.1f, 0.0f }; // Clear the render target and depth buffer (erases what's on the screen) // - Do this ONCE PER FRAME // - At the beginning of Draw (before drawing *anything*) context->ClearRenderTargetView(backBufferRTV.Get(), color); context->ClearDepthStencilView( depthStencilView.Get(), D3D11_CLEAR_DEPTH | D3D11_CLEAR_STENCIL, 1.0f, 0); for (auto entity : entities) { entity->Draw(camera, ambient, lights); } // Present the back buffer to the user // - Puts the final frame we're drawing into the window so the user can see it // - Do this exactly ONCE PER FRAME (always at the very end of the frame) swapChain->Present(vsync ? 1 : 0, 0); // Due to the usage of a more sophisticated swap chain, // the render target must be re-bound after every call to Present() context->OMSetRenderTargets(1, backBufferRTV.GetAddressOf(), depthStencilView.Get()); }