This tutorial is part of a Collection: 03. DirectX 11 - Braynzar Soft Tutorials
16473
views
35. Render To Texture
We will be making a sort of map in this lesson, by rendering the terrain onto a texture, then drawing that texture in the bottom right corner of our backbuffer. It's actually a very easy thing to do, but I decided to make a lesson on it since we will be using this in the next two lessons.
##Introduction## In this lesson, we will be learning how to create a texture that we can render to, then bind it as a shader resource to be rendered as a texture onto a primitive, such as a square. We will be making a map in this lesson. The map in this lesson is not a good example of how to create a map, for two reasons. First, If you want a map, you will most definitely want have a preprocessed map, wich is loaded from a texture for example. It is bad for performance if you are redrawing your entire scene for the map, which we do in this lesson basically. It is only an example of how to render to a texture, not a good example of how to create a map. Also, The map doesn't follow the player exactly as it should, you will notice that the map reaches the end of the terrain before the player. ##Render To Texture## We cannot bind a render target as a shader resource directly, so how do we render to a texture, then use that texture as a shader resource to fill in a primitive such as a quad? We will have to create three things. First, we will create the 2d texture. This is the texture that stores all the color information. We will then create a render target, which points to this texture. When we bind the render target to be drawn onto, we will actually be drawing on the texture that the render target points to. Then we create a shader resource view, which points to the texture. When we bind the shader resource view, the data we get from the shader resource view is actually the data in the texture that the shader resource view points to. It's really simple actually, so i hope this makes sense ;) ##New Globals## We start by declaring three interface objects. The first one is the texture that the maps render target and shader resource view point to. The next two are pointers to the texture, the maps render target and the maps shader resource view. After these, we have two new matrices. They are the for the map camera. The view matrix will hold the map cameras position, and orientation and stuff in world space, while the projection will define it's frustum and stuff. We will be creating an orthonal projection, so that things do not get smaller as they get further away from the camera. ID3D11Texture2D* renderTargetTextureMap; ID3D11RenderTargetView* renderTargetViewMap; ID3D11ShaderResourceView* shaderResourceViewMap; // Our map camera's view and projection matrices XMMATRIX mapView; XMMATRIX mapProjection; ##Create the Map Texture## We start by declaring three descriptions, one for the texture, one for the render target, and one for the shader resource view. Then we create the maps texture. This texture will take up one fourth of the screen, so i have set it's dimensions to be half the screens width and height. Also in the description, it is important that we set the bind flags for a render target AND a shader resource, so that we are able to bind this texture as both. D3D11_TEXTURE2D_DESC textureDesc; D3D11_RENDER_TARGET_VIEW_DESC renderTargetViewDesc; D3D11_SHADER_RESOURCE_VIEW_DESC shaderResourceViewDesc; ///////////////////////// Map's Texture // Initialize the texture description. ZeroMemory(&textureDesc, sizeof(textureDesc)); // Setup the texture description. // We will have our map be a square // We will need to have this texture bound as a render target AND a shader resource textureDesc.Width = Width/2; textureDesc.Height = Height/2; textureDesc.MipLevels = 1; textureDesc.ArraySize = 1; textureDesc.Format = DXGI_FORMAT_R32G32B32A32_FLOAT; textureDesc.SampleDesc.Count = 1; textureDesc.Usage = D3D11_USAGE_DEFAULT; textureDesc.BindFlags = D3D11_BIND_RENDER_TARGET | D3D11_BIND_SHADER_RESOURCE; textureDesc.CPUAccessFlags = 0; textureDesc.MiscFlags = 0; // Create the texture d3d11Device->CreateTexture2D(&textureDesc, NULL, &renderTargetTextureMap); ##Create the Map's Render Target## Now we will create a render target, which is a pointer to the texture we just created above. We have to make sure that the format of all three of these objects (texture, render target, and shader resource) are the exact same. We can do that by setting the format member of the description to be the textures format member. /////////////////////// Map's Render Target // Setup the description of the render target view. renderTargetViewDesc.Format = textureDesc.Format; renderTargetViewDesc.ViewDimension = D3D11_RTV_DIMENSION_TEXTURE2D; renderTargetViewDesc.Texture2D.MipSlice = 0; // Create the render target view. d3d11Device->CreateRenderTargetView(renderTargetTextureMap, &renderTargetViewDesc, &renderTargetViewMap); ##Create the Map's Shader Resource View## Here we create the shader resource view. The format again has to be the same as the other two. /////////////////////// Map's Shader Resource View // Setup the description of the shader resource view. shaderResourceViewDesc.Format = textureDesc.Format; shaderResourceViewDesc.ViewDimension = D3D11_SRV_DIMENSION_TEXTURE2D; shaderResourceViewDesc.Texture2D.MostDetailedMip = 0; shaderResourceViewDesc.Texture2D.MipLevels = 1; // Create the shader resource view. d3d11Device->CreateShaderResourceView(renderTargetTextureMap, &shaderResourceViewDesc, &shaderResourceViewMap); ##Create Map Camera's View and Projection Matrices## We want the camera to follow the players position, so we set the position of the camera to be 100 units directly above the players position, and the target to be the players position, so that the camera is looking straight down. We set the camera's up vector to be the positive z axis. Look at how we create the projection matrix this time. We are using a different function now. This function creates an orthographical projection, so that no matter how far away an object is from the camera, it will always be the same size. We can look at the projection as a stretched cube, so that anything inside the stretched cube is rendered, and anything outside is culled. Our projection will take on an area of 512x512, and the length is from 1 to 1000. //////////////////////// Map's camera information // We will have the camera follow the player XMVECTOR mapCamPosition = XMVectorSetY(camPosition, XMVectorGetY(camPosition) + 100.0f); XMVECTOR mapCamTarget = camPosition; XMVECTOR mapCamUp = XMVectorSet( 0.0f, 0.0f, 1.0f, 0.0f ); //Set the View matrix mapView = XMMatrixLookAtLH( mapCamPosition, mapCamTarget, mapCamUp ); // Build an orthographic projection matrix mapProjection = XMMatrixOrthographicLH( 512, 512, 1.0f, 1000.0f); ##UpdateScene() Function## Here is our new update scene function. We want the camera to follow the player, so we just update the map's camera here every frame. void UpdateScene(double time) { /************************************New Stuff****************************************************/ // Update the map's camera XMVECTOR mapCamPosition = XMVectorSetY(camPosition, XMVectorGetY(camPosition) + 100.0f); XMVECTOR mapCamTarget = camPosition; XMVECTOR mapCamUp = XMVectorSet( 0.0f, 0.0f, 1.0f, 0.0f ); //Set the View matrix mapView = XMMatrixLookAtLH( mapCamPosition, mapCamTarget, mapCamUp ); /*************************************************************************************************/ } ##Draw Terrain From Map's Point of View## Now, We draw the terrain onto the new map's texture (by binding the maps render target which points to the maps texture). We start by binding the maps render target to the output merger pipeline stage (OM). Then we render our terrain from the map camera's perspective (mapView * mapProjection). ////////////////////////// Draw Terrain Onto Map // Here we will draw our map, which is just the terrain from the mapCam's view // Set our maps Render Target d3d11DevCon->OMSetRenderTargets( 1, &renderTargetViewMap, depthStencilView ); // Now clear the render target d3d11DevCon->ClearRenderTargetView(renderTargetViewMap, bgColor); // Since we just drew the terrain, and all the states are already set the way we want them // (besides the render target) we just need to provide the shaders with the new WVP and draw the terrain again WVP = groundWorld * mapView * mapProjection; cbPerObj.WVP = XMMatrixTranspose(WVP); cbPerObj.World = XMMatrixTranspose(groundWorld); d3d11DevCon->UpdateSubresource( cbPerObjectBuffer, 0, NULL, &cbPerObj, 0, 0 ); d3d11DevCon->DrawIndexed( NumFaces * 3, 0, 0 ); ##Draw the Map onto the Backbuffer## Now we have our terrain drawn to the map's texture, We will need to render that texture onto a quad that sits in the bottom right corner of the screen. We start by setting the render target back to the default one, which is our backbuffer. We do not want lighting to effect the map's quad, so we cheat and set the D2D's pixel shader, which we created in an earlier lesson that does not impliment lighting. We are also going to use the quad from the d2d lesson to draw our map onto. Now, to draw things directly to the screen (not in world space), all we have to do is set it's position to be in screen space. Screen space is defined like this: -1 < x < 1 // Where -1 is the left side of the screen -1 < y < 1 // Where -1 is the bottom of the screen 0 < z < 1 // Where 1 is the furthest from the screen\ We will have our map take up the bottom right side of the screen, which is one fourth of the screen. We do this by scaling the quad to be half the size on the x and y axis (ultimately 1/4 the size) since the quad was created as -1 to 1 for the x axis and -1 to 1 for the y axis. The we translate the cube to be in the bottom right side of the screen. //////////////////////////// Draw the Map // Make sure to set the render target back d3d11DevCon->OMSetRenderTargets( 1, &renderTargetView, depthStencilView ); // Now lets actually draw the map. We only need a square to put the texture on, so we'll just // use the d2d's square we made in an earlier lesson. We will be drawing this square directly // in screen space so we don't need to use the view or projection matrix. also, the square is // set as -1 to 1 for both the x and y axis's, which will cover the entire screen. We only want // to cover the bottom right corner of the screen, so we will scale the square down and translate // it to the bottom right corner of the screen. // Set it to the D2D_PS so that we do not impliment lighting d3d11DevCon->PSSetShader(D2D_PS, 0, 0); //Set the d2d square's Index buffer d3d11DevCon->IASetIndexBuffer( d2dIndexBuffer, DXGI_FORMAT_R32_UINT, 0); //Set the d2d square's vertex buffer d3d11DevCon->IASetVertexBuffers( 0, 1, &d2dVertBuffer, &stride, &offset ); // Just set the WVP to a scale and translate, which will put the square into the bottom right corner of the screen WVP = XMMatrixScaling( 0.5f, 0.5f, 0.0f ) * XMMatrixTranslation( 0.5f, -0.5f, 0.0f ); cbPerObj.WVP = XMMatrixTranspose(WVP); d3d11DevCon->UpdateSubresource( cbPerObjectBuffer, 0, NULL, &cbPerObj, 0, 0 ); d3d11DevCon->VSSetConstantBuffers( 0, 1, &cbPerObjectBuffer ); d3d11DevCon->PSSetShaderResources( 0, 1, &shaderResourceViewMap ); // Draw the map to the square d3d11DevCon->PSSetSamplers( 0, 1, &CubesTexSamplerState ); d3d11DevCon->RSSetState(CWcullMode); //Draw the second cube d3d11DevCon->DrawIndexed( 6, 0, 0 ); This is a terrible way to impliment a map, but like i said, it was only to show you how to render to a texture ;) ##Exercise:## 1. make a marker on the map to show the players position 2. Try making a video camera, and a tv that shows what the video camera sees Here's the final code: main.cpp //Include and link appropriate libraries and headers// #pragma comment(lib, "d3d11.lib") #pragma comment(lib, "d3dx11.lib") #pragma comment(lib, "d3dx10.lib") #pragma comment (lib, "D3D10_1.lib") #pragma comment (lib, "DXGI.lib") #pragma comment (lib, "D2D1.lib") #pragma comment (lib, "dwrite.lib") #pragma comment (lib, "dinput8.lib") #pragma comment (lib, "dxguid.lib") #include <windows.h> #include <d3d11.h> #include <d3dx11.h> #include <d3dx10.h> #include <xnamath.h> #include <D3D10_1.h> #include <DXGI.h> #include <D2D1.h> #include <sstream> #include <dwrite.h> #include <dinput.h> #include <vector> //Global Declarations - Interfaces// IDXGISwapChain* SwapChain; ID3D11Device* d3d11Device; ID3D11DeviceContext* d3d11DevCon; ID3D11RenderTargetView* renderTargetView; ID3D11Buffer* squareIndexBuffer; ID3D11DepthStencilView* depthStencilView; ID3D11Texture2D* depthStencilBuffer; ID3D11Buffer* squareVertBuffer; ID3D11VertexShader* VS; ID3D11PixelShader* PS; ID3D11PixelShader* D2D_PS; ID3D10Blob* D2D_PS_Buffer; ID3D10Blob* VS_Buffer; ID3D10Blob* PS_Buffer; ID3D11InputLayout* vertLayout; ID3D11Buffer* cbPerObjectBuffer; ID3D11BlendState* Transparency; ID3D11RasterizerState* CCWcullMode; ID3D11RasterizerState* CWcullMode; ID3D11ShaderResourceView* CubesTexture; ID3D11SamplerState* CubesTexSamplerState; ID3D11Buffer* cbPerFrameBuffer; ID3D10Device1 *d3d101Device; IDXGIKeyedMutex *keyedMutex11; IDXGIKeyedMutex *keyedMutex10; ID2D1RenderTarget *D2DRenderTarget; ID2D1SolidColorBrush *Brush; ID3D11Texture2D *BackBuffer11; ID3D11Texture2D *sharedTex11; ID3D11Buffer *d2dVertBuffer; ID3D11Buffer *d2dIndexBuffer; ID3D11ShaderResourceView *d2dTexture; IDWriteFactory *DWriteFactory; IDWriteTextFormat *TextFormat; IDirectInputDevice8* DIKeyboard; IDirectInputDevice8* DIMouse; /************************************New Stuff****************************************************/ ID3D11Texture2D* renderTargetTextureMap; ID3D11RenderTargetView* renderTargetViewMap; ID3D11ShaderResourceView* shaderResourceViewMap; // Our map camera's view and projection matrices XMMATRIX mapView; XMMATRIX mapProjection; /*************************************************************************************************/ std::wstring printText; //Global Declarations - Others// LPCTSTR WndClassName = L"firstwindow"; HWND hwnd = NULL; HRESULT hr; int Width = 800; int Height = 600; DIMOUSESTATE mouseLastState; LPDIRECTINPUT8 DirectInput; float rotx = 0; float rotz = 0; float scaleX = 1.0f; float scaleY = 1.0f; XMMATRIX Rotationx; XMMATRIX Rotationz; XMMATRIX WVP; XMMATRIX cube1World; XMMATRIX cube2World; XMMATRIX camView; XMMATRIX camProjection; XMMATRIX d2dWorld; XMVECTOR camPosition; XMVECTOR camTarget; XMVECTOR camUp; XMVECTOR DefaultForward = XMVectorSet(0.0f,0.0f,1.0f, 0.0f); XMVECTOR DefaultRight = XMVectorSet(1.0f,0.0f,0.0f, 0.0f); XMVECTOR camForward = XMVectorSet(0.0f,0.0f,1.0f, 0.0f); XMVECTOR camRight = XMVectorSet(1.0f,0.0f,0.0f, 0.0f); XMMATRIX camRotationMatrix; XMMATRIX groundWorld; float moveLeftRight = 0.0f; float moveBackForward = 0.0f; float camYaw = 0.0f; float camPitch = 0.0f; XMMATRIX Rotation; XMMATRIX Scale; XMMATRIX Translation; float rot = 0.01f; double countsPerSecond = 0.0; __int64 CounterStart = 0; int frameCount = 0; int fps = 0; __int64 frameTimeOld = 0; double frameTime; //Function Prototypes// bool InitializeDirect3d11App(HINSTANCE hInstance); void CleanUp(); bool InitScene(); void DrawScene(); bool InitD2D_D3D101_DWrite(IDXGIAdapter1 *Adapter); void InitD2DScreenTexture(); void UpdateScene(double time); void UpdateCamera(); void RenderText(std::wstring text, int inInt); void StartTimer(); double GetTime(); double GetFrameTime(); bool InitializeWindow(HINSTANCE hInstance, int ShowWnd, int width, int height, bool windowed); int messageloop(); bool InitDirectInput(HINSTANCE hInstance); void DetectInput(double time); // This is the scale of our scene, which we will use to find "a very small distance" // If this is set too small, you will notice the camera "stick" to the geometry once in a while // Just play with it and use whatever works best for your application const float unitsPerMeter = 100.0f; // The gravity's velocity vector XMVECTOR gravity = XMVectorSet(0.0f, -0.2f, 0.0f, 0.0f); // Polygon Soup std::vector<XMFLOAT3> collidableGeometryPositions; std::vector<DWORD> collidableGeometryIndices; struct CollisionPacket{ // Information about ellipsoid (in world space) XMVECTOR ellipsoidSpace; XMVECTOR w_Position; XMVECTOR w_Velocity; // Information about ellipsoid (in ellipsoid space) XMVECTOR e_Position; XMVECTOR e_Velocity; XMVECTOR e_normalizedVelocity; // Collision Information bool foundCollision; float nearestDistance; XMVECTOR intersectionPoint; int collisionRecursionDepth; }; // Collision Detection and Response Function Prototypes XMVECTOR CollisionSlide(CollisionPacket& cP, // Pointer to a CollisionPacket object (expects ellipsoidSpace, w_Position and w_Velocity to be filled) std::vector<XMFLOAT3>& vertPos, // An array holding the polygon soup vertex positions std::vector<DWORD>& indices); // An array holding the polygon soup indices (triangles) XMVECTOR CollideWithWorld(CollisionPacket& cP, // Same arguments as the above function std::vector<XMFLOAT3>& vertPos, std::vector<DWORD>& indices); bool SphereCollidingWithTriangle(CollisionPacket& cP, // Pointer to a CollisionPacket object XMVECTOR &p0, // First vertex position of triangle XMVECTOR &p1, // Second vertex position of triangle XMVECTOR &p2, // Third vertex position of triangle XMVECTOR &triNormal); // Triangle's Normal // Checks if a point (inside the triangle's plane) is inside the triangle bool checkPointInTriangle(const XMVECTOR& point, const XMVECTOR& triV1,const XMVECTOR& triV2, const XMVECTOR& triV3); // Solves the quadratic eqation, and returns the lowest root if equation is solvable, returns false if not solvable bool getLowestRoot(float a, float b, float c, float maxR, float* root); int NumFaces = 0; int NumVertices = 0; struct HeightMapInfo{ // Heightmap structure int terrainWidth; // Width of heightmap int terrainHeight; // Height (Length) of heightmap XMFLOAT3 *heightMap; // Array to store terrain's vertex positions }; bool HeightMapLoad(char* filename, HeightMapInfo &hminfo); LRESULT CALLBACK WndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam); //Create effects constant buffer's structure// struct cbPerObject { XMMATRIX WVP; XMMATRIX World; }; cbPerObject cbPerObj; struct Light { Light() { ZeroMemory(this, sizeof(Light)); } XMFLOAT3 dir; float pad; XMFLOAT4 ambient; XMFLOAT4 diffuse; }; Light light; struct cbPerFrame { Light light; }; cbPerFrame constbuffPerFrame; struct Vertex //Overloaded Vertex Structure { Vertex(){} Vertex(float x, float y, float z, float u, float v, float nx, float ny, float nz) : pos(x,y,z), texCoord(u, v), normal(nx, ny, nz){} XMFLOAT3 pos; XMFLOAT2 texCoord; XMFLOAT3 normal; }; D3D11_INPUT_ELEMENT_DESC layout[] = { { "POSITION", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 0, D3D11_INPUT_PER_VERTEX_DATA, 0 }, { "TEXCOORD", 0, DXGI_FORMAT_R32G32_FLOAT, 0, 12, D3D11_INPUT_PER_VERTEX_DATA, 0 }, { "NORMAL", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 20, D3D11_INPUT_PER_VERTEX_DATA, 0} }; UINT numElements = ARRAYSIZE(layout); int WINAPI WinMain(HINSTANCE hInstance, //Main windows function HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd) { if(!InitializeWindow(hInstance, nShowCmd, Width, Height, true)) { MessageBox(0, L"Window Initialization - Failed", L"Error", MB_OK); return 0; } if(!InitializeDirect3d11App(hInstance)) //Initialize Direct3D { MessageBox(0, L"Direct3D Initialization - Failed", L"Error", MB_OK); return 0; } if(!InitScene()) //Initialize our scene { MessageBox(0, L"Scene Initialization - Failed", L"Error", MB_OK); return 0; } if(!InitDirectInput(hInstance)) { MessageBox(0, L"Direct Input Initialization - Failed", L"Error", MB_OK); return 0; } messageloop(); CleanUp(); return 0; } bool InitializeWindow(HINSTANCE hInstance, int ShowWnd, int width, int height, bool windowed) { typedef struct _WNDCLASS { UINT cbSize; UINT style; WNDPROC lpfnWndProc; int cbClsExtra; int cbWndExtra; HANDLE hInstance; HICON hIcon; HCURSOR hCursor; HBRUSH hbrBackground; LPCTSTR lpszMenuName; LPCTSTR lpszClassName; } WNDCLASS; WNDCLASSEX wc; wc.cbSize = sizeof(WNDCLASSEX); wc.style = CS_HREDRAW | CS_VREDRAW; wc.lpfnWndProc = WndProc; wc.cbClsExtra = NULL; wc.cbWndExtra = NULL; wc.hInstance = hInstance; wc.hIcon = LoadIcon(NULL, IDI_APPLICATION); wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wc.lpszMenuName = NULL; wc.lpszClassName = WndClassName; wc.hIconSm = LoadIcon(NULL, IDI_APPLICATION); if (!RegisterClassEx(&wc)) { MessageBox(NULL, L"Error registering class", L"Error", MB_OK | MB_ICONERROR); return 1; } hwnd = CreateWindowEx( NULL, WndClassName, L"Lesson 30 - Sliding Camera Collision Detection", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, width, height, NULL, NULL, hInstance, NULL ); if (!hwnd) { MessageBox(NULL, L"Error creating window", L"Error", MB_OK | MB_ICONERROR); return 1; } ShowWindow(hwnd, ShowWnd); UpdateWindow(hwnd); return true; } bool InitializeDirect3d11App(HINSTANCE hInstance) { //Describe our SwapChain Buffer DXGI_MODE_DESC bufferDesc; ZeroMemory(&bufferDesc, sizeof(DXGI_MODE_DESC)); bufferDesc.Width = Width; bufferDesc.Height = Height; bufferDesc.RefreshRate.Numerator = 60; bufferDesc.RefreshRate.Denominator = 1; bufferDesc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; bufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; bufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; //Describe our SwapChain DXGI_SWAP_CHAIN_DESC swapChainDesc; ZeroMemory(&swapChainDesc, sizeof(DXGI_SWAP_CHAIN_DESC)); swapChainDesc.BufferDesc = bufferDesc; swapChainDesc.SampleDesc.Count = 1; swapChainDesc.SampleDesc.Quality = 0; swapChainDesc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; swapChainDesc.BufferCount = 1; swapChainDesc.OutputWindow = hwnd; ///////////////**************new**************//////////////////// swapChainDesc.Windowed = true; ///////////////**************new**************//////////////////// swapChainDesc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; // Create DXGI factory to enumerate adapters/////////////////////////////////////////////////////////////////////////// IDXGIFactory1 *DXGIFactory; HRESULT hr = CreateDXGIFactory1(__uuidof(IDXGIFactory1), (void**)&DXGIFactory); // Use the first adapter IDXGIAdapter1 *Adapter; hr = DXGIFactory->EnumAdapters1(0, &Adapter); DXGIFactory->Release(); //Create our Direct3D 11 Device and SwapChain////////////////////////////////////////////////////////////////////////// hr = D3D11CreateDeviceAndSwapChain(Adapter, D3D_DRIVER_TYPE_UNKNOWN, NULL, D3D11_CREATE_DEVICE_BGRA_SUPPORT, NULL, NULL, D3D11_SDK_VERSION, &swapChainDesc, &SwapChain, &d3d11Device, NULL, &d3d11DevCon); //Initialize Direct2D, Direct3D 10.1, DirectWrite InitD2D_D3D101_DWrite(Adapter); //Release the Adapter interface Adapter->Release(); //Create our BackBuffer and Render Target hr = SwapChain->GetBuffer( 0, __uuidof( ID3D11Texture2D ), (void**)&BackBuffer11 ); hr = d3d11Device->CreateRenderTargetView( BackBuffer11, NULL, &renderTargetView ); //Describe our Depth/Stencil Buffer D3D11_TEXTURE2D_DESC depthStencilDesc; depthStencilDesc.Width = Width; depthStencilDesc.Height = Height; depthStencilDesc.MipLevels = 1; depthStencilDesc.ArraySize = 1; depthStencilDesc.Format = DXGI_FORMAT_D24_UNORM_S8_UINT; depthStencilDesc.SampleDesc.Count = 1; depthStencilDesc.SampleDesc.Quality = 0; depthStencilDesc.Usage = D3D11_USAGE_DEFAULT; depthStencilDesc.BindFlags = D3D11_BIND_DEPTH_STENCIL; depthStencilDesc.CPUAccessFlags = 0; depthStencilDesc.MiscFlags = 0; //Create the Depth/Stencil View d3d11Device->CreateTexture2D(&depthStencilDesc, NULL, &depthStencilBuffer); d3d11Device->CreateDepthStencilView(depthStencilBuffer, NULL, &depthStencilView); return true; } bool InitD2D_D3D101_DWrite(IDXGIAdapter1 *Adapter) { //Create our Direc3D 10.1 Device/////////////////////////////////////////////////////////////////////////////////////// hr = D3D10CreateDevice1(Adapter, D3D10_DRIVER_TYPE_HARDWARE, NULL,D3D10_CREATE_DEVICE_BGRA_SUPPORT, D3D10_FEATURE_LEVEL_9_3, D3D10_1_SDK_VERSION, &d3d101Device ); //Create Shared Texture that Direct3D 10.1 will render on////////////////////////////////////////////////////////////// D3D11_TEXTURE2D_DESC sharedTexDesc; ZeroMemory(&sharedTexDesc, sizeof(sharedTexDesc)); sharedTexDesc.Width = Width; sharedTexDesc.Height = Height; sharedTexDesc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; sharedTexDesc.MipLevels = 1; sharedTexDesc.ArraySize = 1; sharedTexDesc.SampleDesc.Count = 1; sharedTexDesc.Usage = D3D11_USAGE_DEFAULT; sharedTexDesc.BindFlags = D3D11_BIND_SHADER_RESOURCE | D3D11_BIND_RENDER_TARGET; sharedTexDesc.MiscFlags = D3D11_RESOURCE_MISC_SHARED_KEYEDMUTEX; hr = d3d11Device->CreateTexture2D(&sharedTexDesc, NULL, &sharedTex11); // Get the keyed mutex for the shared texture (for D3D11)/////////////////////////////////////////////////////////////// hr = sharedTex11->QueryInterface(__uuidof(IDXGIKeyedMutex), (void**)&keyedMutex11); // Get the shared handle needed to open the shared texture in D3D10.1/////////////////////////////////////////////////// IDXGIResource *sharedResource10; HANDLE sharedHandle10; hr = sharedTex11->QueryInterface(__uuidof(IDXGIResource), (void**)&sharedResource10); hr = sharedResource10->GetSharedHandle(&sharedHandle10); sharedResource10->Release(); // Open the surface for the shared texture in D3D10.1/////////////////////////////////////////////////////////////////// IDXGISurface1 *sharedSurface10; hr = d3d101Device->OpenSharedResource(sharedHandle10, __uuidof(IDXGISurface1), (void**)(&sharedSurface10)); hr = sharedSurface10->QueryInterface(__uuidof(IDXGIKeyedMutex), (void**)&keyedMutex10); // Create D2D factory/////////////////////////////////////////////////////////////////////////////////////////////////// ID2D1Factory *D2DFactory; hr = D2D1CreateFactory(D2D1_FACTORY_TYPE_SINGLE_THREADED, __uuidof(ID2D1Factory), (void**)&D2DFactory); D2D1_RENDER_TARGET_PROPERTIES renderTargetProperties; ZeroMemory(&renderTargetProperties, sizeof(renderTargetProperties)); renderTargetProperties.type = D2D1_RENDER_TARGET_TYPE_HARDWARE; renderTargetProperties.pixelFormat = D2D1::PixelFormat(DXGI_FORMAT_UNKNOWN, D2D1_ALPHA_MODE_PREMULTIPLIED); hr = D2DFactory->CreateDxgiSurfaceRenderTarget(sharedSurface10, &renderTargetProperties, &D2DRenderTarget); sharedSurface10->Release(); D2DFactory->Release(); // Create a solid color brush to draw something with hr = D2DRenderTarget->CreateSolidColorBrush(D2D1::ColorF(1.0f, 1.0f, 1.0f, 1.0f), &Brush); //DirectWrite/////////////////////////////////////////////////////////////////////////////////////////////////////////// hr = DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory), reinterpret_cast<IUnknown**>(&DWriteFactory)); hr = DWriteFactory->CreateTextFormat( L"Script", NULL, DWRITE_FONT_WEIGHT_REGULAR, DWRITE_FONT_STYLE_NORMAL, DWRITE_FONT_STRETCH_NORMAL, 24.0f, L"en-us", &TextFormat ); hr = TextFormat->SetTextAlignment(DWRITE_TEXT_ALIGNMENT_LEADING); hr = TextFormat->SetParagraphAlignment(DWRITE_PARAGRAPH_ALIGNMENT_NEAR); d3d101Device->IASetPrimitiveTopology(D3D10_PRIMITIVE_TOPOLOGY_POINTLIST); return true; } bool HeightMapLoad(char* filename, HeightMapInfo &hminfo) { FILE *filePtr; // Point to the current position in the file BITMAPFILEHEADER bitmapFileHeader; // Structure which stores information about file BITMAPINFOHEADER bitmapInfoHeader; // Structure which stores information about image int imageSize, index; unsigned char height; // Open the file filePtr = fopen(filename,"rb"); if (filePtr == NULL) return 0; // Read bitmaps header fread(&bitmapFileHeader, sizeof(BITMAPFILEHEADER), 1,filePtr); // Read the info header fread(&bitmapInfoHeader, sizeof(BITMAPINFOHEADER), 1, filePtr); // Get the width and height (width and length) of the image hminfo.terrainWidth = bitmapInfoHeader.biWidth; hminfo.terrainHeight = bitmapInfoHeader.biHeight; // Size of the image in bytes. the 3 represents RBG (byte, byte, byte) for each pixel imageSize = hminfo.terrainWidth * hminfo.terrainHeight * 3; // Initialize the array which stores the image data unsigned char* bitmapImage = new unsigned char[imageSize]; // Set the file pointer to the beginning of the image data fseek(filePtr, bitmapFileHeader.bfOffBits, SEEK_SET); // Store image data in bitmapImage fread(bitmapImage, 1, imageSize, filePtr); // Close file fclose(filePtr); // Initialize the heightMap array (stores the vertices of our terrain) hminfo.heightMap = new XMFLOAT3[hminfo.terrainWidth * hminfo.terrainHeight]; // We use a greyscale image, so all 3 rgb values are the same, but we only need one for the height // So we use this counter to skip the next two components in the image data (we read R, then skip BG) int k=0; // We divide the height by this number to "water down" the terrains height, otherwise the terrain will // appear to be "spikey" and not so smooth. float heightFactor = 10.0f; // Read the image data into our heightMap array for(int j=0; j< hminfo.terrainHeight; j++) { for(int i=0; i< hminfo.terrainWidth; i++) { height = bitmapImage[k]; index = ( hminfo.terrainHeight * j) + i; hminfo.heightMap[index].x = (float)i; hminfo.heightMap[index].y = (float)height / heightFactor; hminfo.heightMap[index].z = (float)j; k+=3; } } delete [] bitmapImage; bitmapImage = 0; return true; } // This is the function we will call when we want to find if an ellipsoid will collide with // the world (polygon soup) while traveling along it's velocity vector, and also impliment // gravity by doing the collision detection and response algorithm with the gravity's // velocity vector. It's kind of like the collision detection and response package XMVECTOR CollisionSlide(CollisionPacket& cP, std::vector<XMFLOAT3>& vertPos, std::vector<DWORD>& indices) { // Transform velocity vector to the ellipsoid space (e_ denotes ellipsoid space) cP.e_Velocity = cP.w_Velocity/cP.ellipsoidSpace; // Transform position vector to the ellipsoid space cP.e_Position = cP.w_Position/cP.ellipsoidSpace; // Now we check for a collision with our world, this function will // call itself 5 times at most, or until the velocity vector is // used up (very small (near zero to zero length)) cP.collisionRecursionDepth = 0; XMVECTOR finalPosition = CollideWithWorld(cP, vertPos, indices); // Add gravity pull: // This is simply adding a new velocity vector in the downward // direction (defined globaly) to pull the ellipsoid down, then doing the // collision check against all the geometry again. The way it is now, the // ellipsoid will "slide" down even the most slightest slope. Consider this // an exercise: only impliment gravity when standing on a very steep slope, // or if you are not standing on anything at all (free fall) // To remove gravity uncomment from here ..... cP.e_Velocity = gravity / cP.ellipsoidSpace; // We defined gravity in world space, so now we have // to convert it to ellipsoid space cP.e_Position = finalPosition; cP.collisionRecursionDepth = 0; finalPosition = CollideWithWorld(cP, vertPos, indices); // ... to here // Convert our final position from ellipsoid space to world space finalPosition = finalPosition * cP.ellipsoidSpace; // Return our final position! return finalPosition; } // This function impliments the collision detection and collision response XMVECTOR CollideWithWorld(CollisionPacket& cP, std::vector<XMFLOAT3>& vertPos, std::vector<DWORD>& indices) { // These are based off the unitsPerMeter from above float unitScale = unitsPerMeter / 100.0f; float veryCloseDistance = 0.005f * unitScale; // This is used to keep the sphere from actually "touching" // the triangle, as that would cause problems since // each loop it would ALWAYS find a collision instead // of just sliding along the triangle // This will stop us from entering an infinite loop, or a very long loop. For example, there are times when the sphere // might actually be pushed slightly into the triangles center, where the recursion will keep repeating and finding a collision // even though the velocity vector does not change (I had serious problems with this part for a couple days... I couldn't // figure out why the ellipsoid would LAUNCH at certain times, but if i set the ellipsoid space to (1,1,1) (a sphere), it would // act normal. Stupid me made a mistake and was returning w_Position here instead of e_Position, so that the world space position // was being multiplied by the ellipsoid space and "launching" it whenever it accidently got pushed into a triangle) if (cP.collisionRecursionDepth > 5) return cP.e_Position; // Normalize velocity vector cP.e_normalizedVelocity = XMVector3Normalize(cP.e_Velocity); // Initialize collision packet stuff cP.foundCollision = false; cP.nearestDistance = 0.0f; // Loop through each triangle in mesh and check for a collision for(int triCounter = 0; triCounter < indices.size() / 3; triCounter++) { // Get triangle XMVECTOR p0, p1, p2, tempVec; p0 = XMLoadFloat3(&vertPos[indices[3*triCounter]]); p1 = XMLoadFloat3(&vertPos[indices[3*triCounter+1]]); p2 = XMLoadFloat3(&vertPos[indices[3*triCounter+2]]); // Put triangle into ellipsoid space p0 = p0/cP.ellipsoidSpace; p1 = p1/cP.ellipsoidSpace; p2 = p2/cP.ellipsoidSpace; // Calculate the normal for this triangle XMVECTOR triNormal; triNormal = XMVector3Normalize(XMVector3Cross((p1 - p0),(p2 - p0))); // Now we check to see if the sphere is colliding with the current triangle SphereCollidingWithTriangle(cP, p0, p1, p2, triNormal); } // If there was no collision, return the position + velocity if (cP.foundCollision == false) { return cP.e_Position + cP.e_Velocity; } // If we've made it here, a collision occured // destinationPoint is where the sphere would travel if there was // no collisions, however, at this point, there has a been a collision // detected. We will use this vector to find the new "sliding" vector // based off the plane created from the sphere and collision point XMVECTOR destinationPoint = cP.e_Position + cP.e_Velocity; XMVECTOR newPosition = cP.e_Position; // Just initialize newPosition // If the position is further than "veryCloseDistance" from the point // of collision, we will move the sphere along the velocity path until // it "almost" touches the triangle, or point of collision. We do this so // that the next recursion (if there is one) does not detect a collision // with the triangle we just collided with. We don't need to find a collision // with the triangle we just collided with because we will be moving parallel // to it now, and if we were finding the collision with it every recursion // (since it's the closest triangle we would collide with), we would // finish our 5 recursions (checked above) without ever moving after // touching a triangle, because before the triangle has a chance to move // down the new velocity path (even though it's about parallel with the triangle) // it would find the collision with the triangle, and simply recompute the same // velocity vector it computed the first time. This would happen because of // floating point innacuracy. theoretically, we would not have to worry about this // because after the new velocity vector is created, it SHOULD be perfectly parallel // to the triangle, and we detect that in our code and basically skip triangles // who are perfectly parallel with the velocity vector. But like i said, because // of innacuracy, the new velocity vector might be VERY SLIGHTLY pointed down towards // the triangles plane, which would make us waste a recursion just to recompute the same // velocity vector. Basically, the whole sliding thing works without this, but it's a lot // more "choppy" and "sticky", where you get stuck in random places. if (cP.nearestDistance >= veryCloseDistance) { // Move the new position down velocity vector to ALMOST touch the collision point XMVECTOR V = cP.e_Velocity; V = XMVector3Normalize(V); V = V * (cP.nearestDistance - veryCloseDistance); newPosition = cP.e_Position + V; // Adjust polygon intersection point (so sliding // plane will be unaffected by the fact that we // move slightly less than collision tells us) V = XMVector3Normalize(V); cP.intersectionPoint -= veryCloseDistance * V; } // This is our sliding plane (point in the plane and plane normal) XMVECTOR slidePlaneOrigin = cP.intersectionPoint; XMVECTOR slidePlaneNormal = newPosition - cP.intersectionPoint; slidePlaneNormal = XMVector3Normalize(slidePlaneNormal); // We will use the sliding plane to compute our new "destination" point // and new velocity vector. To do this, we will need to solve another quadratic // equation (Ax + By + Cz + D = 0), where D is what we call the plane constant, // which we use to find the distance between the sliding plane and our original // destination point (original as up until now, since it's likely that this is // not the first recursion of this function, and the original original destination // has been changed up until now). // First the point in the plane float x = XMVectorGetX(slidePlaneOrigin); float y = XMVectorGetY(slidePlaneOrigin); float z = XMVectorGetZ(slidePlaneOrigin); // Next the planes normal float A = XMVectorGetX(slidePlaneNormal); float B = XMVectorGetY(slidePlaneNormal); float C = XMVectorGetZ(slidePlaneNormal); float D = -((A*x) + (B*y) + (C*z)); // To keep the variable names clear, we will rename D to planeConstant float planeConstant = D; // Get the distance between sliding plane and destination point float signedDistFromDestPointToSlidingPlane = XMVectorGetX(XMVector3Dot(destinationPoint, slidePlaneNormal)) + planeConstant; // Now we calculate the new destination point. To get the new destination point, we will subtract // the distance from the plane to the original destination point (down the planes normal) from the // original destination point. It's easier to picture this in your head than explain, so let me try // to give you a very simple picture. Pretend you are this equation, standing on the plane, where UP // (your head) is pointing the same direction as the plane's normal. directly below you is the "destination" // point of the sphere. Your job as this equation is to "pull" the destination point up (towards the planes // normal) until it is resting "in" the plane. If you can picture this the way i'm trying to get you to, you // can see that the new velocity vector (from the point of collision between sphere and plane) to the new // destination is "shorter" and parallel to the plane, so that now when the sphere follows this new velocity // vector, it will be traveling parallel (sliding) across the triangle, at the same time, it does not travel // as far as it would have if there was no collision. This is exactly what we want, because when you think about // it, we do not run up mountains as fast as we run on flat ground, and if we run straight into a wall in our // game, we will just stop moving, or if we run ALMOST straight into the wall, we will not go cruising sideways, // but instead slowly move to either side. In my lesson on braynzarsoft.net, This is explained in pictures XMVECTOR newDestinationPoint = destinationPoint - signedDistFromDestPointToSlidingPlane * slidePlaneNormal; // I believe this line was covered briefly in the above explanation XMVECTOR newVelocityVector = newDestinationPoint - cP.intersectionPoint; // After this check, we will recurse. This check makes sure that we have not // come to the end of our velocity vector (or very close to it, because if the velocity // vector is very small, there is no reason to lose performance by doing an extra recurse // when we won't even notice the distance "thrown away" by this check anyway) before // we recurse if (XMVectorGetX(XMVector3Length(newVelocityVector)) < veryCloseDistance) { return newPosition; } // We are going to recurse now since a collision was found and the velocity // changed directions. we need to check if the new velocity vector will // cause the sphere to collide with other geometry. cP.collisionRecursionDepth++; cP.e_Position = newPosition; cP.e_Velocity = newVelocityVector; return CollideWithWorld(cP, vertPos, indices); } // This function checks if the swept sphere collides with a single triangle bool SphereCollidingWithTriangle(CollisionPacket& cP, XMVECTOR &p0, XMVECTOR &p1, XMVECTOR &p2, XMVECTOR &triNormal) { // This function assumes p0, p1, p2, and the triangle normal are in ellipsoid space // and that e_Position e_Velocity, and e_normalizedVelocity are defined in ellipsoid space // In other words, this function checks for a collision between a SPHERE and a triangle, // not an ellipsoid and a triangle. Because of this, the results from this function // (specifically cP.nearestDistance and cP.intersectionPoint) are in ellipsoid space // Check to see if triangle is facing velocity vector // We will not triangle facing away from the velocity vector to speed this up // since we assume that we will never run into the back face of triangles float facing = XMVectorGetX(XMVector3Dot(triNormal, cP.e_normalizedVelocity)); if(facing <= 0) { // Create these because cP.e_Velocity and cP.e_Position add slightly to the difficulty // of reading the equations XMVECTOR velocity = cP.e_Velocity; XMVECTOR position = cP.e_Position; // t0 and t1 hold the time it takes along the velocity vector that the sphere (called a swept sphere) // will "collide" (resting on or touching), once on the front side of the triangle (t0), and once on the // backside after it goes "through" the triangle (t1) (or vertex or edge). float t0, t1; // If sphere is in the plane, it will not intersect with the center of the triangle // but instead possibly intersect with one of the vertices or edges first bool sphereInPlane = false; // Find the plane equation in which the triangle lies in (Ax + By + Cz + D = 0) // A, B, and C are the planes normal, x, y, and z respectively // We can find D (a.k.a the plane constant) using some simple algebra, which we will do below // x, y, and z in the equation defines a point in the plane. Any point in the plane // will do, so we will just use p0 // First the point in the plane float x = XMVectorGetX(p0); float y = XMVectorGetY(p0); float z = XMVectorGetZ(p0); // Next the planes normal float A = XMVectorGetX(triNormal); float B = XMVectorGetY(triNormal); float C = XMVectorGetZ(triNormal); // Lets solve for D // step 1: 0 = Ax + By + Cz + D // step 2: subtract D from both sides // -D = Ax + By + Cz // setp 3: multiply both sides by -1 // -D*-1 = -1 * (Ax + By + Cz) // final answer: D = -(Ax + By + Cz) float D = -((A*x) + (B*y) + (C*z)); // To keep the variable names clear, we will rename D to planeConstant float planeConstant = D; // Get the signed distance from the cameras position (or object if you are using an object) // We can get the signed distance between a point and plane with the equation: // SignedDistance = PlaneNormal * Point + PlaneConstant // I've mentioned this before, but i'll do it again. When using xna math library vector function // that return a scalar value (like a float) such as "XMVector3Dot", an XMVECTOR is returned, with // all elements (x,y,z,w) containing that scalar value. We need to extract one, and any will do since // they are all the same, so we extract the x component using "XMVectorGetX" float signedDistFromPositionToTriPlane = XMVectorGetX(XMVector3Dot(position, triNormal)) + planeConstant; // This will be used a couple times below, so we'll just calculate and store it now float planeNormalDotVelocity = XMVectorGetX(XMVector3Dot(triNormal, velocity)); /////////////////////////////////////Sphere Plane Collision Test//////////////////////////////////////////// // Check to see if the velocity vector is parallel with the plane if (planeNormalDotVelocity == 0.0f) { if (fabs(signedDistFromPositionToTriPlane) >= 1.0f) { // sphere not in plane, and velocity is // parallel to plane, no collision possible return false; } else { // sphere is in the plane, so we will now only test for a collision // with the triangle's vertices and edges // Set sphereInPlane to true so we do not do the operation // which will divide by zero if the velocity and plane are parallel sphereInPlane = true; } } else { // We know the velocity vector at some point intersects with the plane, we just // need to find how far down the velocity vector the sphere will "touch" or rest // on the plane. t0 is when it first touches the plane (front side of sphere touches) // and t1 is when the back side of the sphere touches. // To find when (the time or how far down the velocity vector) the "velocity vector" itself //intersects with the plane, we use the equation: (* stands for a dot product) // t = (PlaneNormal * Point + PlaneConstant) / (PlaneNormal * Velocity); // We have already calculated both sides of the divide sign "/", so: // t = signedDistance / normalDotVelocity; // Now remember we are working with a unit sphere (since everything has been moved from // the usual space to our ellipsoid space). The unit sphere means that the distance from // the center of the sphere to ANYWHERE on it's surface is "1". We are not interested in // finding when the actual velocity vector intersects with the plane, but instead when // the surface of the sphere "touches" the surface of the plane. We know that the distance // from the center of the sphere is "1", so all we have to do to find when the sphere touches // the plane is subtract and subtract 1 from the signed distance to get when both sides of the // sphere touch the plane (t0, and t1) t0 = ( 1.0f - signedDistFromPositionToTriPlane) / planeNormalDotVelocity; t1 = (-1.0f - signedDistFromPositionToTriPlane) / planeNormalDotVelocity; // We will make sure that t0 is smaller than t1, which means that t0 is when the sphere FIRST // touches the planes surface if(t0 > t1) { float temp = t0; t0 = t1; t1 = temp; } // If the swept sphere touches the plane outside of the 0 to 1 "timeframe", we know that // the sphere is not going to intersect with the plane (and of course triangle) this frame if (t0 > 1.0f || t1 < 0.0f) { return false; } // If t0 is smaller than 0 then we will make it 0 // and if t1 is greater than 1 we will make it 1 if (t0 < 0.0) t0 = 0.0; if (t1 > 1.0) t1 = 1.0; } ////////////////////////////////Sphere-(Inside Triangle) Collision Test/////////////////////////////////////// // If we've made it this far, we know that the sphere will intersect with the triangles plane // This frame, so now we will check to see if the collision happened INSIDE the triangle XMVECTOR collisionPoint; // Point on plane where collision occured bool collidingWithTri = false; // This is set so we know if there was a collision with the CURRENT triangle float t = 1.0; // Time // If the sphere is not IN the triangles plane, we continue the sphere to inside of triangle test if (!sphereInPlane) { // We get the point on the triangles plane where the sphere "touches" the plane // using the equation: planeIntersectionPoint = (Position - Normal) + t0 * Velocity // Where t0 is the distance down the velocity vector that the sphere first makes // contact with the plane XMVECTOR planeIntersectionPoint = (position + t0 * velocity - triNormal); // Now we call the function that checks if a point on a triangle's plane is inside the triangle if (checkPointInTriangle(planeIntersectionPoint,p0,p1,p2)) { // If the point on the plane IS inside the triangle, we know that the sphere is colliding // with the triangle now, so we set collidingWithTri to true so we don't do all the extra // calculations on the triangle. We set t to t0, which is the time (or distance) down // the velocity vector that the sphere first makes contact, then we set the point of // collision, which will be used later for our collision response collidingWithTri = true; t = t0; collisionPoint = planeIntersectionPoint; } } /////////////////////////////////////Sphere-Vertex Collision Test////////////////////////////////////////////// // If the sphere is not colliding with the triangles INSIDE, we check to see if it will collide with one of // the vertices of the triangle using the sweep test we did above, but this time check for each vertex instead // of the triangles plane if (collidingWithTri == false) { // We will be working with the quadratic function "At^2 + Bt + C = 0" to find when (t) the "swept sphere"s center // is 1 unit (spheres radius) away from the vertex position. Remember the swept spheres position is actually a line defined // by the spheres position and velocity. t represents it's position along the velocity vector. // a = sphereVelocityLength * sphereVelocityLength // b = 2(sphereVelocity . (spherePosition - vertexPosition)) // . denotes dot product // c = (vertexPosition - spherePosition)^2 - 1 // This equation allows for two solutions. One is when the sphere "first" touches the vertex, and the other is when // the "other" side of the sphere touches the vertex on it's way past the vertex. We need the first "touch" float a, b, c; // Equation Parameters // We can use the squared velocities length below when checking for collisions with the edges of the triangles // to, so to keep things clear, we won't set a directly float velocityLengthSquared = XMVectorGetX(XMVector3Length(velocity)); velocityLengthSquared *= velocityLengthSquared; // We'll start by setting 'a', since all 3 point equations use this 'a' a = velocityLengthSquared; // This is a temporary variable to hold the distance down the velocity vector that // the sphere will touch the vertex. float newT; // P0 - Collision test with sphere and p0 b = 2.0f * ( XMVectorGetX( XMVector3Dot( velocity, position - p0 ))); c = XMVectorGetX(XMVector3Length((p0 - position))); c = (c*c) - 1.0f; if (getLowestRoot(a,b,c, t, &newT)) { // Check if the equation can be solved // If the equation was solved, we can set a couple things. First we set t (distance // down velocity vector the sphere first collides with vertex) to the temporary newT, // Then we set collidingWithTri to be true so we know there was for sure a collision // with the triangle, then we set the exact point the sphere collides with the triangle, // which is the position of the vertex it collides with t = newT; collidingWithTri = true; collisionPoint = p0; } // P1 - Collision test with sphere and p1 b = 2.0*(XMVectorGetX(XMVector3Dot(velocity, position - p1))); c = XMVectorGetX(XMVector3Length((p1 - position))); c = (c*c) - 1.0; if (getLowestRoot(a,b,c, t, &newT)) { t = newT; collidingWithTri = true; collisionPoint = p1; } // P2 - Collision test with sphere and p2 b = 2.0*(XMVectorGetX(XMVector3Dot(velocity, position - p2))); c = XMVectorGetX(XMVector3Length((p2 - position))); c = (c*c) - 1.0; if (getLowestRoot(a,b,c, t, &newT)) { t = newT; collidingWithTri = true; collisionPoint = p2; } //////////////////////////////////////////////Sphere-Edge Collision Test////////////////////////////////////////////// // Even though there might have been a collision with a vertex, we will still check for a collision with an edge of the // triangle in case an edge was hit before the vertex. Again we will solve a quadratic equation to find where (and if) // the swept sphere's position is 1 unit away from the edge of the triangle. The equation parameters this time are a // bit more complex: (still "Ax^2 + Bx + C = 0") // a = edgeLength^2 * -velocityLength^2 + (edge . velocity)^2 // b = edgeLength^2 * 2(velocity . spherePositionToVertex) - 2((edge . velocity)(edge . spherePositionToVertex)) // c = edgeLength^2 * (1 - spherePositionToVertexLength^2) + (edge . spherePositionToVertex)^2 // . denotes dot product // Edge (p0, p1): XMVECTOR edge = p1 - p0; XMVECTOR spherePositionToVertex = p0 - position; float edgeLengthSquared = XMVectorGetX(XMVector3Length(edge)); edgeLengthSquared *= edgeLengthSquared; float edgeDotVelocity = XMVectorGetX(XMVector3Dot(edge, velocity)); float edgeDotSpherePositionToVertex = XMVectorGetX(XMVector3Dot(edge, spherePositionToVertex)); float spherePositionToVertexLengthSquared = XMVectorGetX(XMVector3Length(spherePositionToVertex)); spherePositionToVertexLengthSquared = spherePositionToVertexLengthSquared * spherePositionToVertexLengthSquared; // Equation parameters a = edgeLengthSquared * -velocityLengthSquared + (edgeDotVelocity * edgeDotVelocity); b = edgeLengthSquared * (2.0f * XMVectorGetX(XMVector3Dot(velocity, spherePositionToVertex))) - (2.0f * edgeDotVelocity * edgeDotSpherePositionToVertex); c = edgeLengthSquared * (1.0f - spherePositionToVertexLengthSquared) + (edgeDotSpherePositionToVertex * edgeDotSpherePositionToVertex); // We start by finding if the swept sphere collides with the edges "infinite line" if (getLowestRoot(a,b,c, t, &newT)) { // Now we check to see if the collision happened between the two vertices that make up this edge // We can calculate where on the line the collision happens by doing this: // f = (edge . velocity)newT - (edge . spherePositionToVertex) / edgeLength^2 // if f is between 0 and 1, then we know the collision happened between p0 and p1 // If the collision happened at p0, the f = 0, if the collision happened at p1 then f = 1 float f = (edgeDotVelocity * newT - edgeDotSpherePositionToVertex) / edgeLengthSquared; if (f >= 0.0f && f <= 1.0f) { // If the collision with the edge happened, we set the results t = newT; collidingWithTri = true; collisionPoint = p0 + f * edge; } } // Edge (p1, p2): edge = p2 - p1; spherePositionToVertex = p1 - position; edgeLengthSquared = XMVectorGetX(XMVector3Length(edge)); edgeLengthSquared = edgeLengthSquared * edgeLengthSquared; edgeDotVelocity = XMVectorGetX(XMVector3Dot(edge, cP.e_Velocity)); edgeDotSpherePositionToVertex = XMVectorGetX(XMVector3Dot(edge, spherePositionToVertex)); spherePositionToVertexLengthSquared = XMVectorGetX(XMVector3Length(spherePositionToVertex)); spherePositionToVertexLengthSquared = spherePositionToVertexLengthSquared * spherePositionToVertexLengthSquared; a = edgeLengthSquared * -velocityLengthSquared + (edgeDotVelocity * edgeDotVelocity); b = edgeLengthSquared * (2.0f * XMVectorGetX(XMVector3Dot(velocity, spherePositionToVertex))) - (2.0f * edgeDotVelocity * edgeDotSpherePositionToVertex); c = edgeLengthSquared * (1.0f - spherePositionToVertexLengthSquared) + (edgeDotSpherePositionToVertex * edgeDotSpherePositionToVertex); if (getLowestRoot(a,b,c, t, &newT)) { float f = (edgeDotVelocity * newT - edgeDotSpherePositionToVertex) / edgeLengthSquared; if (f >= 0.0f && f <= 1.0f) { t = newT; collidingWithTri = true; collisionPoint = p1 + f * edge; } } // Edge (p2, p0): edge = p0 - p2; spherePositionToVertex = p2 - position; edgeLengthSquared = XMVectorGetX(XMVector3Length(edge)); edgeLengthSquared = edgeLengthSquared * edgeLengthSquared; edgeDotVelocity = XMVectorGetX(XMVector3Dot(edge, velocity)); edgeDotSpherePositionToVertex = XMVectorGetX(XMVector3Dot(edge, spherePositionToVertex)); spherePositionToVertexLengthSquared = XMVectorGetX(XMVector3Length(spherePositionToVertex)); spherePositionToVertexLengthSquared = spherePositionToVertexLengthSquared * spherePositionToVertexLengthSquared; a = edgeLengthSquared * -velocityLengthSquared + (edgeDotVelocity * edgeDotVelocity); b = edgeLengthSquared * (2.0f * XMVectorGetX(XMVector3Dot(velocity, spherePositionToVertex))) - (2.0f * edgeDotVelocity * edgeDotSpherePositionToVertex); c = edgeLengthSquared * (1.0f - spherePositionToVertexLengthSquared) + (edgeDotSpherePositionToVertex * edgeDotSpherePositionToVertex); if (getLowestRoot(a,b,c, t, &newT)) { float f = (edgeDotVelocity * newT - edgeDotSpherePositionToVertex) / edgeLengthSquared; if (f >= 0.0f && f <= 1.0f) { t = newT; collidingWithTri = true; collisionPoint = p2 + f * edge; } } } // If we have found a collision, we will set the results of the collision here if (collidingWithTri == true) { // We find the distance to the collision using the time variable (t) times the length of the velocity vector float distToCollision = t * XMVectorGetX(XMVector3Length(velocity)); // Now we check if this is the first triangle that has been collided with OR it is // the closest triangle yet that was collided with if (cP.foundCollision == false || distToCollision < cP.nearestDistance) { // Collision response information (used for "sliding") cP.nearestDistance = distToCollision; cP.intersectionPoint = collisionPoint; // Make sure this is set to true if we've made it this far cP.foundCollision = true; return true; } } } return false; } // These are the "helper" functions // This function is found in my lesson on picking in direct3d 11 // This function finds if a point (in the triangle plane) is INSIDE the triangle bool checkPointInTriangle(const XMVECTOR& point, const XMVECTOR& triV1,const XMVECTOR& triV2, const XMVECTOR& triV3) { XMVECTOR cp1 = XMVector3Cross((triV3 - triV2), (point - triV2)); XMVECTOR cp2 = XMVector3Cross((triV3 - triV2), (triV1 - triV2)); if(XMVectorGetX(XMVector3Dot(cp1, cp2)) >= 0) { cp1 = XMVector3Cross((triV3 - triV1), (point - triV1)); cp2 = XMVector3Cross((triV3 - triV1), (triV2 - triV1)); if(XMVectorGetX(XMVector3Dot(cp1, cp2)) >= 0) { cp1 = XMVector3Cross((triV2 - triV1), (point - triV1)); cp2 = XMVector3Cross((triV2 - triV1), (triV3 - triV1)); if(XMVectorGetX(XMVector3Dot(cp1, cp2)) >= 0) { return true; } } } return false; } // This function solves the quadratic eqation "At^2 + Bt + C = 0" and is found in Kasper Fauerby's paper on collision detection and response bool getLowestRoot(float a, float b, float c, float maxR, float* root) { // Check if a solution exists float determinant = b*b - 4.0f*a*c; // If determinant is negative it means no solutions. if (determinant < 0.0f) return false; // calculate the two roots: (if determinant == 0 then // x1==x2 but lets disregard that slight optimization) float sqrtD = sqrt(determinant); float r1 = (-b - sqrtD) / (2*a); float r2 = (-b + sqrtD) / (2*a); // Sort so x1 <= x2 if (r1 > r2) { float temp = r2; r2 = r1; r1 = temp; } // Get lowest root: if (r1 > 0 && r1 < maxR) { *root = r1; return true; } // It is possible that we want x2 - this can happen // if x1 < 0 if (r2 > 0 && r2 < maxR) { *root = r2; return true; } // No (valid) solutions return false; } bool InitDirectInput(HINSTANCE hInstance) { hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, IID_IDirectInput8, (void**)&DirectInput, NULL); hr = DirectInput->CreateDevice(GUID_SysKeyboard, &DIKeyboard, NULL); hr = DirectInput->CreateDevice(GUID_SysMouse, &DIMouse, NULL); hr = DIKeyboard->SetDataFormat(&c_dfDIKeyboard); hr = DIKeyboard->SetCooperativeLevel(hwnd, DISCL_FOREGROUND | DISCL_NONEXCLUSIVE); hr = DIMouse->SetDataFormat(&c_dfDIMouse); hr = DIMouse->SetCooperativeLevel(hwnd, DISCL_EXCLUSIVE | DISCL_NOWINKEY | DISCL_FOREGROUND); return true; } void UpdateCamera() { camRotationMatrix = XMMatrixRotationRollPitchYaw(camPitch, camYaw, 0); camTarget = XMVector3TransformCoord(DefaultForward, camRotationMatrix ); camTarget = XMVector3Normalize(camTarget); // First-Person Camera XMMATRIX RotateYTempMatrix; RotateYTempMatrix = XMMatrixRotationY(camYaw); camRight = XMVector3TransformCoord(DefaultRight, RotateYTempMatrix); camUp = XMVector3TransformCoord(camUp, RotateYTempMatrix); camForward = XMVector3TransformCoord(DefaultForward, RotateYTempMatrix); /* // Free-Look Camera camRight = XMVector3TransformCoord(DefaultRight, camRotationMatrix); camForward = XMVector3TransformCoord(DefaultForward, camRotationMatrix); camUp = XMVector3Cross(camForward, camRight); */ CollisionPacket cameraCP; cameraCP.ellipsoidSpace = XMVectorSet(1.0f, 3.0f, 1.0f, 0.0f); cameraCP.w_Position = camPosition; cameraCP.w_Velocity = (moveLeftRight*camRight)+(moveBackForward*camForward); camPosition = CollisionSlide(cameraCP, collidableGeometryPositions, collidableGeometryIndices); /*camPosition += moveLeftRight*camRight; camPosition += moveBackForward*camForward;*/ moveLeftRight = 0.0f; moveBackForward = 0.0f; camTarget = camPosition + camTarget; camView = XMMatrixLookAtLH( camPosition, camTarget, camUp ); } void DetectInput(double time) { DIMOUSESTATE mouseCurrState; BYTE keyboardState[256]; DIKeyboard->Acquire(); DIMouse->Acquire(); DIMouse->GetDeviceState(sizeof(DIMOUSESTATE), &mouseCurrState); DIKeyboard->GetDeviceState(sizeof(keyboardState),(LPVOID)&keyboardState); if(keyboardState[DIK_ESCAPE] & 0x80) PostMessage(hwnd, WM_DESTROY, 0, 0); float speed = 15.0f * time; if(keyboardState[DIK_A] & 0x80) { moveLeftRight -= speed; } if(keyboardState[DIK_D] & 0x80) { moveLeftRight += speed; } if(keyboardState[DIK_W] & 0x80) { moveBackForward += speed; } if(keyboardState[DIK_S] & 0x80) { moveBackForward -= speed; } if((mouseCurrState.lX != mouseLastState.lX) || (mouseCurrState.lY != mouseLastState.lY)) { camYaw += mouseLastState.lX * 0.001f; camPitch += mouseCurrState.lY * 0.001f; mouseLastState = mouseCurrState; } UpdateCamera(); return; } void CleanUp() { SwapChain->SetFullscreenState(false, NULL); PostMessage(hwnd, WM_DESTROY, 0, 0); //Release the COM Objects we created SwapChain->Release(); d3d11Device->Release(); d3d11DevCon->Release(); renderTargetView->Release(); squareVertBuffer->Release(); squareIndexBuffer->Release(); VS->Release(); PS->Release(); VS_Buffer->Release(); PS_Buffer->Release(); vertLayout->Release(); depthStencilView->Release(); depthStencilBuffer->Release(); cbPerObjectBuffer->Release(); Transparency->Release(); CCWcullMode->Release(); CWcullMode->Release(); d3d101Device->Release(); keyedMutex11->Release(); keyedMutex10->Release(); D2DRenderTarget->Release(); Brush->Release(); BackBuffer11->Release(); sharedTex11->Release(); DWriteFactory->Release(); TextFormat->Release(); d2dTexture->Release(); cbPerFrameBuffer->Release(); DIKeyboard->Unacquire(); DIMouse->Unacquire(); DirectInput->Release(); } void InitD2DScreenTexture() { //Create the vertex buffer Vertex v[] = { // Front Face Vertex(-1.0f, -1.0f, -1.0f, 0.0f, 1.0f,-1.0f, -1.0f, -1.0f), Vertex(-1.0f, 1.0f, -1.0f, 0.0f, 0.0f,-1.0f, 1.0f, -1.0f), Vertex( 1.0f, 1.0f, -1.0f, 1.0f, 0.0f, 1.0f, 1.0f, -1.0f), Vertex( 1.0f, -1.0f, -1.0f, 1.0f, 1.0f, 1.0f, -1.0f, -1.0f), }; DWORD indices[] = { // Front Face 0, 1, 2, 0, 2, 3, }; D3D11_BUFFER_DESC indexBufferDesc; ZeroMemory( &indexBufferDesc, sizeof(indexBufferDesc) ); indexBufferDesc.Usage = D3D11_USAGE_DEFAULT; indexBufferDesc.ByteWidth = sizeof(DWORD) * 2 * 3; indexBufferDesc.BindFlags = D3D11_BIND_INDEX_BUFFER; indexBufferDesc.CPUAccessFlags = 0; indexBufferDesc.MiscFlags = 0; D3D11_SUBRESOURCE_DATA iinitData; iinitData.pSysMem = indices; d3d11Device->CreateBuffer(&indexBufferDesc, &iinitData, &d2dIndexBuffer); D3D11_BUFFER_DESC vertexBufferDesc; ZeroMemory( &vertexBufferDesc, sizeof(vertexBufferDesc) ); vertexBufferDesc.Usage = D3D11_USAGE_DEFAULT; vertexBufferDesc.ByteWidth = sizeof( Vertex ) * 4; vertexBufferDesc.BindFlags = D3D11_BIND_VERTEX_BUFFER; vertexBufferDesc.CPUAccessFlags = 0; vertexBufferDesc.MiscFlags = 0; D3D11_SUBRESOURCE_DATA vertexBufferData; ZeroMemory( &vertexBufferData, sizeof(vertexBufferData) ); vertexBufferData.pSysMem = v; hr = d3d11Device->CreateBuffer( &vertexBufferDesc, &vertexBufferData, &d2dVertBuffer); //Create A shader resource view from the texture D2D will render to, //So we can use it to texture a square which overlays our scene d3d11Device->CreateShaderResourceView(sharedTex11, NULL, &d2dTexture); } bool InitScene() { InitD2DScreenTexture(); //Compile Shaders from shader file hr = D3DX11CompileFromFile(L"Effects.fx", 0, 0, "VS", "vs_4_0", 0, 0, 0, &VS_Buffer, 0, 0); hr = D3DX11CompileFromFile(L"Effects.fx", 0, 0, "PS", "ps_4_0", 0, 0, 0, &PS_Buffer, 0, 0); hr = D3DX11CompileFromFile(L"Effects.fx", 0, 0, "D2D_PS", "ps_4_0", 0, 0, 0, &D2D_PS_Buffer, 0, 0); //Create the Shader Objects hr = d3d11Device->CreateVertexShader(VS_Buffer->GetBufferPointer(), VS_Buffer->GetBufferSize(), NULL, &VS); hr = d3d11Device->CreatePixelShader(PS_Buffer->GetBufferPointer(), PS_Buffer->GetBufferSize(), NULL, &PS); hr = d3d11Device->CreatePixelShader(D2D_PS_Buffer->GetBufferPointer(), D2D_PS_Buffer->GetBufferSize(), NULL, &D2D_PS); //Set Vertex and Pixel Shaders d3d11DevCon->VSSetShader(VS, 0, 0); d3d11DevCon->PSSetShader(PS, 0, 0); light.dir = XMFLOAT3(0.0f, -1.0f, 0.0f); light.ambient = XMFLOAT4(0.2f, 0.2f, 0.2f, 1.0f); light.diffuse = XMFLOAT4(1.0f, 1.0f, 1.0f, 1.0f); HeightMapInfo hmInfo; HeightMapLoad("heightmap.bmp", hmInfo); // Load the heightmap and store it into hmInfo int cols = hmInfo.terrainWidth; int rows = hmInfo.terrainHeight; //Create the grid NumVertices = rows * cols; NumFaces = (rows-1)*(cols-1)*2; std::vector<Vertex> v(NumVertices); for(DWORD i = 0; i < rows; ++i) { for(DWORD j = 0; j < cols; ++j) { v[i*cols+j].pos = hmInfo.heightMap[i*cols+j]; v[i*cols+j].normal = XMFLOAT3(0.0f, -1.0f, 0.0f); } } std::vector<DWORD> indices(NumFaces * 3); int k = 0; int texUIndex = 0; int texVIndex = 0; for(DWORD i = 0; i < rows-1; i++) { for(DWORD j = 0; j < cols-1; j++) { indices[k] = i*cols+j; // Bottom left of quad v[i*cols+j].texCoord = XMFLOAT2(texUIndex + 0.0f, texVIndex + 1.0f); indices[k+1] = (i+1)*cols+j; // Top left of quad v[(i+1)*cols+j].texCoord = XMFLOAT2(texUIndex + 0.0f, texVIndex + 0.0f); indices[k+2] = i*cols+j+1; // Bottom right of quad v[i*cols+j+1].texCoord = XMFLOAT2(texUIndex + 1.0f, texVIndex + 1.0f); indices[k+3] = (i+1)*cols+j; // Top left of quad v[(i+1)*cols+j].texCoord = XMFLOAT2(texUIndex + 0.0f, texVIndex + 0.0f); indices[k+4] = (i+1)*cols+j+1; // Top right of quad v[(i+1)*cols+j+1].texCoord = XMFLOAT2(texUIndex + 1.0f, texVIndex + 0.0f); indices[k+5] = i*cols+j+1; // Bottom right of quad v[i*cols+j+1].texCoord = XMFLOAT2(texUIndex + 1.0f, texVIndex + 1.0f); k += 6; // next quad texUIndex++; } texUIndex = 0; texVIndex++; } // Since our terrain will not be transformed throughout our scene, we will set the our groundWorlds // world matrix here so that when we put the terrains positions in the "polygon soup", they will // already be transformed to world space groundWorld = XMMatrixIdentity(); Scale = XMMatrixScaling( 10.0f, 10.0f, 10.0f ); Translation = XMMatrixTranslation( -520.0f, -10.0f, -1020.0f ); groundWorld = Scale * Translation; // Store the terrains vertex positions and indices in the // polygon soup that we will check for collisions with // We can store ALL static (non-changing) geometry in here that we want to check for collisions with int vertexOffset = collidableGeometryPositions.size(); // Vertex offset (each "mesh" will be added to the end of the positions array) // Temp arrays because we need to store the geometry in world space XMVECTOR tempVertexPosVec; XMFLOAT3 tempVertF3; // Push back vertex positions to the polygon soup for(int i = 0; i < v.size(); i++) { tempVertexPosVec = XMLoadFloat3(&v[i].pos); tempVertexPosVec = XMVector3TransformCoord(tempVertexPosVec, groundWorld); XMStoreFloat3(&tempVertF3, tempVertexPosVec); collidableGeometryPositions.push_back(tempVertF3); } // Push back indices for polygon soup. We need to make sure we are // pushing back the indices "on top" of the previous pushed back // objects vertex positions, hence "+ vertexOffset" (This is the // first object we are putting in here, so it really doesn't // matter right now, but I just wanted to show you how to do it for(int i = 0; i < indices.size(); i++) { collidableGeometryIndices.push_back(indices[i] + vertexOffset); } //////////////////////Compute Normals/////////////////////////// //Now we will compute the normals for each vertex using normal averaging std::vector<XMFLOAT3> tempNormal; //normalized and unnormalized normals XMFLOAT3 unnormalized = XMFLOAT3(0.0f, 0.0f, 0.0f); //Used to get vectors (sides) from the position of the verts float vecX, vecY, vecZ; //Two edges of our triangle XMVECTOR edge1 = XMVectorSet(0.0f, 0.0f, 0.0f, 0.0f); XMVECTOR edge2 = XMVectorSet(0.0f, 0.0f, 0.0f, 0.0f); //Compute face normals for(int i = 0; i < NumFaces; ++i) { //Get the vector describing one edge of our triangle (edge 0,2) vecX = v[indices[(i*3)]].pos.x - v[indices[(i*3)+2]].pos.x; vecY = v[indices[(i*3)]].pos.y - v[indices[(i*3)+2]].pos.y; vecZ = v[indices[(i*3)]].pos.z - v[indices[(i*3)+2]].pos.z; edge1 = XMVectorSet(vecX, vecY, vecZ, 0.0f); //Create our first edge //Get the vector describing another edge of our triangle (edge 2,1) vecX = v[indices[(i*3)+2]].pos.x - v[indices[(i*3)+1]].pos.x; vecY = v[indices[(i*3)+2]].pos.y - v[indices[(i*3)+1]].pos.y; vecZ = v[indices[(i*3)+2]].pos.z - v[indices[(i*3)+1]].pos.z; edge2 = XMVectorSet(vecX, vecY, vecZ, 0.0f); //Create our second edge //Cross multiply the two edge vectors to get the un-normalized face normal XMStoreFloat3(&unnormalized, XMVector3Cross(edge1, edge2)); tempNormal.push_back(unnormalized); //Save unormalized normal (for normal averaging) } //Compute vertex normals (normal Averaging) XMVECTOR normalSum = XMVectorSet(0.0f, 0.0f, 0.0f, 0.0f); int facesUsing = 0; float tX; float tY; float tZ; //Go through each vertex for(int i = 0; i < NumVertices; ++i) { //Check which triangles use this vertex for(int j = 0; j < NumFaces; ++j) { if(indices[j*3] == i || indices[(j*3)+1] == i || indices[(j*3)+2] == i) { tX = XMVectorGetX(normalSum) + tempNormal[j].x; tY = XMVectorGetY(normalSum) + tempNormal[j].y; tZ = XMVectorGetZ(normalSum) + tempNormal[j].z; normalSum = XMVectorSet(tX, tY, tZ, 0.0f); //If a face is using the vertex, add the unormalized face normal to the normalSum facesUsing++; } } //Get the actual normal by dividing the normalSum by the number of faces sharing the vertex normalSum = normalSum / facesUsing; //Normalize the normalSum vector normalSum = XMVector3Normalize(normalSum); //Store the normal in our current vertex v[i].normal.x = XMVectorGetX(normalSum); v[i].normal.y = XMVectorGetY(normalSum); v[i].normal.z = XMVectorGetZ(normalSum); //Clear normalSum and facesUsing for next vertex normalSum = XMVectorSet(0.0f, 0.0f, 0.0f, 0.0f); facesUsing = 0; } D3D11_BUFFER_DESC indexBufferDesc; ZeroMemory( &indexBufferDesc, sizeof(indexBufferDesc) ); indexBufferDesc.Usage = D3D11_USAGE_DEFAULT; indexBufferDesc.ByteWidth = sizeof(DWORD) * NumFaces * 3; indexBufferDesc.BindFlags = D3D11_BIND_INDEX_BUFFER; indexBufferDesc.CPUAccessFlags = 0; indexBufferDesc.MiscFlags = 0; D3D11_SUBRESOURCE_DATA iinitData; iinitData.pSysMem = &indices[0]; d3d11Device->CreateBuffer(&indexBufferDesc, &iinitData, &squareIndexBuffer); D3D11_BUFFER_DESC vertexBufferDesc; ZeroMemory( &vertexBufferDesc, sizeof(vertexBufferDesc) ); vertexBufferDesc.Usage = D3D11_USAGE_DEFAULT; vertexBufferDesc.ByteWidth = sizeof( Vertex ) * NumVertices; vertexBufferDesc.BindFlags = D3D11_BIND_VERTEX_BUFFER; vertexBufferDesc.CPUAccessFlags = 0; vertexBufferDesc.MiscFlags = 0; D3D11_SUBRESOURCE_DATA vertexBufferData; ZeroMemory( &vertexBufferData, sizeof(vertexBufferData) ); vertexBufferData.pSysMem = &v[0]; hr = d3d11Device->CreateBuffer( &vertexBufferDesc, &vertexBufferData, &squareVertBuffer); //Create the Input Layout hr = d3d11Device->CreateInputLayout( layout, numElements, VS_Buffer->GetBufferPointer(), VS_Buffer->GetBufferSize(), &vertLayout ); //Set the Input Layout d3d11DevCon->IASetInputLayout( vertLayout ); //Set Primitive Topology d3d11DevCon->IASetPrimitiveTopology( D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST ); //Create the Viewport D3D11_VIEWPORT viewport; ZeroMemory(&viewport, sizeof(D3D11_VIEWPORT)); viewport.TopLeftX = 0; viewport.TopLeftY = 0; viewport.Width = Width; viewport.Height = Height; viewport.MinDepth = 0.0f; viewport.MaxDepth = 1.0f; //Set the Viewport d3d11DevCon->RSSetViewports(1, &viewport); //Create the buffer to send to the cbuffer in effect file D3D11_BUFFER_DESC cbbd; ZeroMemory(&cbbd, sizeof(D3D11_BUFFER_DESC)); cbbd.Usage = D3D11_USAGE_DEFAULT; cbbd.ByteWidth = sizeof(cbPerObject); cbbd.BindFlags = D3D11_BIND_CONSTANT_BUFFER; cbbd.CPUAccessFlags = 0; cbbd.MiscFlags = 0; hr = d3d11Device->CreateBuffer(&cbbd, NULL, &cbPerObjectBuffer); //Create the buffer to send to the cbuffer per frame in effect file ZeroMemory(&cbbd, sizeof(D3D11_BUFFER_DESC)); cbbd.Usage = D3D11_USAGE_DEFAULT; cbbd.ByteWidth = sizeof(cbPerFrame); cbbd.BindFlags = D3D11_BIND_CONSTANT_BUFFER; cbbd.CPUAccessFlags = 0; cbbd.MiscFlags = 0; hr = d3d11Device->CreateBuffer(&cbbd, NULL, &cbPerFrameBuffer); //Camera information camPosition = XMVectorSet( 0.0f, 5.0f, -8.0f, 0.0f ); camTarget = XMVectorSet( 0.0f, 0.0f, 0.0f, 0.0f ); camUp = XMVectorSet( 0.0f, 1.0f, 0.0f, 0.0f ); //Set the View matrix camView = XMMatrixLookAtLH( camPosition, camTarget, camUp ); //Set the Projection matrix camProjection = XMMatrixPerspectiveFovLH( 0.4f*3.14f, (float)Width/Height, 1.0f, 1000.0f); D3D11_BLEND_DESC blendDesc; ZeroMemory( &blendDesc, sizeof(blendDesc) ); D3D11_RENDER_TARGET_BLEND_DESC rtbd; ZeroMemory( &rtbd, sizeof(rtbd) ); rtbd.BlendEnable = true; rtbd.SrcBlend = D3D11_BLEND_SRC_COLOR; rtbd.DestBlend = D3D11_BLEND_INV_SRC_ALPHA; rtbd.BlendOp = D3D11_BLEND_OP_ADD; rtbd.SrcBlendAlpha = D3D11_BLEND_ONE; rtbd.DestBlendAlpha = D3D11_BLEND_ZERO; rtbd.BlendOpAlpha = D3D11_BLEND_OP_ADD; rtbd.RenderTargetWriteMask = D3D10_COLOR_WRITE_ENABLE_ALL; blendDesc.AlphaToCoverageEnable = false; blendDesc.RenderTarget[0] = rtbd; hr = D3DX11CreateShaderResourceViewFromFile( d3d11Device, L"grass.jpg", NULL, NULL, &CubesTexture, NULL ); // Describe the Sample State D3D11_SAMPLER_DESC sampDesc; ZeroMemory( &sampDesc, sizeof(sampDesc) ); sampDesc.Filter = D3D11_FILTER_MIN_MAG_MIP_LINEAR; sampDesc.AddressU = D3D11_TEXTURE_ADDRESS_WRAP; sampDesc.AddressV = D3D11_TEXTURE_ADDRESS_WRAP; sampDesc.AddressW = D3D11_TEXTURE_ADDRESS_WRAP; sampDesc.ComparisonFunc = D3D11_COMPARISON_NEVER; sampDesc.MinLOD = 0; sampDesc.MaxLOD = D3D11_FLOAT32_MAX; //Create the Sample State hr = d3d11Device->CreateSamplerState( &sampDesc, &CubesTexSamplerState ); d3d11Device->CreateBlendState(&blendDesc, &Transparency); D3D11_RASTERIZER_DESC cmdesc; ZeroMemory(&cmdesc, sizeof(D3D11_RASTERIZER_DESC)); cmdesc.FillMode = D3D11_FILL_SOLID; cmdesc.CullMode = D3D11_CULL_BACK; cmdesc.FrontCounterClockwise = true; hr = d3d11Device->CreateRasterizerState(&cmdesc, &CCWcullMode); cmdesc.FrontCounterClockwise = false; hr = d3d11Device->CreateRasterizerState(&cmdesc, &CWcullMode); /************************************New Stuff****************************************************/ // What we will be doing, is first create a 2d texture. We will then use this texture as a render // target AND a shader resource. We cannot render to a shader resource directly, so to do this, // we will create a render target and shader resource separately as pointers to this texture. Then // when we want to render to the texture, we will render to the render target, which is actually // a pointer to the texture, so we will be rendering to the texture. When using the shader resource, // we are actually getting the data from the texture that the shader resource points to D3D11_TEXTURE2D_DESC textureDesc; D3D11_RENDER_TARGET_VIEW_DESC renderTargetViewDesc; D3D11_SHADER_RESOURCE_VIEW_DESC shaderResourceViewDesc; ///////////////////////// Map's Texture // Initialize the texture description. ZeroMemory(&textureDesc, sizeof(textureDesc)); // Setup the texture description. // We will have our map be a square // We will need to have this texture bound as a render target AND a shader resource textureDesc.Width = Width/2; textureDesc.Height = Height/2; textureDesc.MipLevels = 1; textureDesc.ArraySize = 1; textureDesc.Format = DXGI_FORMAT_R32G32B32A32_FLOAT; textureDesc.SampleDesc.Count = 1; textureDesc.Usage = D3D11_USAGE_DEFAULT; textureDesc.BindFlags = D3D11_BIND_RENDER_TARGET | D3D11_BIND_SHADER_RESOURCE; textureDesc.CPUAccessFlags = 0; textureDesc.MiscFlags = 0; // Create the texture d3d11Device->CreateTexture2D(&textureDesc, NULL, &renderTargetTextureMap); /////////////////////// Map's Render Target // Setup the description of the render target view. renderTargetViewDesc.Format = textureDesc.Format; renderTargetViewDesc.ViewDimension = D3D11_RTV_DIMENSION_TEXTURE2D; renderTargetViewDesc.Texture2D.MipSlice = 0; // Create the render target view. d3d11Device->CreateRenderTargetView(renderTargetTextureMap, &renderTargetViewDesc, &renderTargetViewMap); // Setup the description of the shader resource view. shaderResourceViewDesc.Format = textureDesc.Format; shaderResourceViewDesc.ViewDimension = D3D11_SRV_DIMENSION_TEXTURE2D; shaderResourceViewDesc.Texture2D.MostDetailedMip = 0; shaderResourceViewDesc.Texture2D.MipLevels = 1; // Create the shader resource view. d3d11Device->CreateShaderResourceView(renderTargetTextureMap, &shaderResourceViewDesc, &shaderResourceViewMap); //////////////////////// Map's camera information // We will have the camera follow the player XMVECTOR mapCamPosition = XMVectorSetY(camPosition, XMVectorGetY(camPosition) + 100.0f); XMVECTOR mapCamTarget = camPosition; XMVECTOR mapCamUp = XMVectorSet( 0.0f, 0.0f, 1.0f, 0.0f ); //Set the View matrix mapView = XMMatrixLookAtLH( mapCamPosition, mapCamTarget, mapCamUp ); // Build an orthographic projection matrix mapProjection = XMMatrixOrthographicLH( 512, 512, 1.0f, 1000.0f); /*************************************************************************************************/ return true; } void StartTimer() { LARGE_INTEGER frequencyCount; QueryPerformanceFrequency(&frequencyCount); countsPerSecond = double(frequencyCount.QuadPart); QueryPerformanceCounter(&frequencyCount); CounterStart = frequencyCount.QuadPart; } double GetTime() { LARGE_INTEGER currentTime; QueryPerformanceCounter(¤tTime); return double(currentTime.QuadPart-CounterStart)/countsPerSecond; } double GetFrameTime() { LARGE_INTEGER currentTime; __int64 tickCount; QueryPerformanceCounter(¤tTime); tickCount = currentTime.QuadPart-frameTimeOld; frameTimeOld = currentTime.QuadPart; if(tickCount < 0.0f) tickCount = 0.0f; return float(tickCount)/countsPerSecond; } void UpdateScene(double time) { /************************************New Stuff****************************************************/ // Update the map's camera XMVECTOR mapCamPosition = XMVectorSetY(camPosition, XMVectorGetY(camPosition) + 100.0f); XMVECTOR mapCamTarget = camPosition; XMVECTOR mapCamUp = XMVectorSet( 0.0f, 0.0f, 1.0f, 0.0f ); //Set the View matrix mapView = XMMatrixLookAtLH( mapCamPosition, mapCamTarget, mapCamUp ); /*************************************************************************************************/ } void RenderText(std::wstring text, int inInt) { d3d11DevCon->PSSetShader(D2D_PS, 0, 0); //Release the D3D 11 Device keyedMutex11->ReleaseSync(0); //Use D3D10.1 device keyedMutex10->AcquireSync(0, 5); //Draw D2D content D2DRenderTarget->BeginDraw(); //Clear D2D Background D2DRenderTarget->Clear(D2D1::ColorF(0.0f, 0.0f, 0.0f, 0.0f)); //Create our string std::wostringstream printString; printString << text << inInt; printText = printString.str(); //Set the Font Color D2D1_COLOR_F FontColor = D2D1::ColorF(1.0f, 1.0f, 1.0f, 1.0f); //Set the brush color D2D will use to draw with Brush->SetColor(FontColor); //Create the D2D Render Area D2D1_RECT_F layoutRect = D2D1::RectF(0, 0, Width, Height); //Draw the Text D2DRenderTarget->DrawText( printText.c_str(), wcslen(printText.c_str()), TextFormat, layoutRect, Brush ); D2DRenderTarget->EndDraw(); //Release the D3D10.1 Device keyedMutex10->ReleaseSync(1); //Use the D3D11 Device keyedMutex11->AcquireSync(1, 5); //Use the shader resource representing the direct2d render target //to texture a square which is rendered in screen space so it //overlays on top of our entire scene. We use alpha blending so //that the entire background of the D2D render target is "invisible", //And only the stuff we draw with D2D will be visible (the text) //Set the blend state for D2D render target texture objects d3d11DevCon->OMSetBlendState(Transparency, NULL, 0xffffffff); //Set the d2d Index buffer d3d11DevCon->IASetIndexBuffer( d2dIndexBuffer, DXGI_FORMAT_R32_UINT, 0); //Set the d2d vertex buffer UINT stride = sizeof( Vertex ); UINT offset = 0; d3d11DevCon->IASetVertexBuffers( 0, 1, &d2dVertBuffer, &stride, &offset ); WVP = XMMatrixIdentity(); cbPerObj.WVP = XMMatrixTranspose(WVP); d3d11DevCon->UpdateSubresource( cbPerObjectBuffer, 0, NULL, &cbPerObj, 0, 0 ); d3d11DevCon->VSSetConstantBuffers( 0, 1, &cbPerObjectBuffer ); d3d11DevCon->PSSetShaderResources( 0, 1, &d2dTexture ); d3d11DevCon->PSSetSamplers( 0, 1, &CubesTexSamplerState ); d3d11DevCon->RSSetState(CWcullMode); //Draw the second cube d3d11DevCon->DrawIndexed( 6, 0, 0 ); } void DrawScene() { //Clear our render target and depth/stencil view float bgColor[4] = { 0.1f, 0.1f, 0.1f, 0.0f }; d3d11DevCon->ClearRenderTargetView(renderTargetView, bgColor); d3d11DevCon->ClearDepthStencilView(depthStencilView, D3D11_CLEAR_DEPTH|D3D11_CLEAR_STENCIL, 1.0f, 0); constbuffPerFrame.light = light; d3d11DevCon->UpdateSubresource( cbPerFrameBuffer, 0, NULL, &constbuffPerFrame, 0, 0 ); d3d11DevCon->PSSetConstantBuffers(0, 1, &cbPerFrameBuffer); //Set our Render Target d3d11DevCon->OMSetRenderTargets( 1, &renderTargetView, depthStencilView ); //Set the default blend state (no blending) for opaque objects d3d11DevCon->OMSetBlendState(0, 0, 0xffffffff); //Set Vertex and Pixel Shaders d3d11DevCon->VSSetShader(VS, 0, 0); d3d11DevCon->PSSetShader(PS, 0, 0); //Set the cubes index buffer d3d11DevCon->IASetIndexBuffer( squareIndexBuffer, DXGI_FORMAT_R32_UINT, 0); //Set the cubes vertex buffer UINT stride = sizeof( Vertex ); UINT offset = 0; d3d11DevCon->IASetVertexBuffers( 0, 1, &squareVertBuffer, &stride, &offset ); //Set the WVP matrix and send it to the constant buffer in effect file WVP = groundWorld * camView * camProjection; cbPerObj.WVP = XMMatrixTranspose(WVP); cbPerObj.World = XMMatrixTranspose(groundWorld); d3d11DevCon->UpdateSubresource( cbPerObjectBuffer, 0, NULL, &cbPerObj, 0, 0 ); d3d11DevCon->VSSetConstantBuffers( 0, 1, &cbPerObjectBuffer ); d3d11DevCon->PSSetShaderResources( 0, 1, &CubesTexture ); d3d11DevCon->PSSetSamplers( 0, 1, &CubesTexSamplerState ); d3d11DevCon->RSSetState(CWcullMode); d3d11DevCon->DrawIndexed( NumFaces * 3, 0, 0 ); /************************************New Stuff****************************************************/ ////////////////////////// Draw Terrain Onto Map // Here we will draw our map, which is just the terrain from the mapCam's view // Set our maps Render Target d3d11DevCon->OMSetRenderTargets( 1, &renderTargetViewMap, depthStencilView ); // Now clear the render target d3d11DevCon->ClearRenderTargetView(renderTargetViewMap, bgColor); // Since we just drew the terrain, and all the states are already set the way we want them // (besides the render target) we just need to provide the shaders with the new WVP and draw the terrain again WVP = groundWorld * mapView * mapProjection; cbPerObj.WVP = XMMatrixTranspose(WVP); cbPerObj.World = XMMatrixTranspose(groundWorld); d3d11DevCon->UpdateSubresource( cbPerObjectBuffer, 0, NULL, &cbPerObj, 0, 0 ); d3d11DevCon->DrawIndexed( NumFaces * 3, 0, 0 ); //////////////////////////// Draw the Map // Make sure to set the render target back d3d11DevCon->OMSetRenderTargets( 1, &renderTargetView, depthStencilView ); // Now lets actually draw the map. We only need a square to put the texture on, so we'll just // use the d2d's square we made in an earlier lesson. We will be drawing this square directly // in screen space so we don't need to use the view or projection matrix. also, the square is // set as -1 to 1 for both the x and y axis's, which will cover the entire screen. We only want // to cover the bottom right corner of the screen, so we will scale the square down and translate // it to the bottom right corner of the screen. // Set it to the D2D_PS so that we do not impliment lighting d3d11DevCon->PSSetShader(D2D_PS, 0, 0); //Set the d2d square's Index buffer d3d11DevCon->IASetIndexBuffer( d2dIndexBuffer, DXGI_FORMAT_R32_UINT, 0); //Set the d2d square's vertex buffer d3d11DevCon->IASetVertexBuffers( 0, 1, &d2dVertBuffer, &stride, &offset ); // Just set the WVP to a scale and translate, which will put the square into the bottom right corner of the screen WVP = XMMatrixScaling( 0.5f, 0.5f, 0.0f ) * XMMatrixTranslation( 0.5f, -0.5f, 0.0f ); cbPerObj.WVP = XMMatrixTranspose(WVP); d3d11DevCon->UpdateSubresource( cbPerObjectBuffer, 0, NULL, &cbPerObj, 0, 0 ); d3d11DevCon->VSSetConstantBuffers( 0, 1, &cbPerObjectBuffer ); d3d11DevCon->PSSetShaderResources( 0, 1, &shaderResourceViewMap ); // Draw the map to the square d3d11DevCon->PSSetSamplers( 0, 1, &CubesTexSamplerState ); d3d11DevCon->RSSetState(CWcullMode); //Draw the second cube d3d11DevCon->DrawIndexed( 6, 0, 0 ); /*************************************************************************************************/ RenderText(L"FPS: ", fps); //Present the backbuffer to the screen SwapChain->Present(0, 0); } int messageloop(){ MSG msg; ZeroMemory(&msg, sizeof(MSG)); while(true) { BOOL PeekMessageL( LPMSG lpMsg, HWND hWnd, UINT wMsgFilterMin, UINT wMsgFilterMax, UINT wRemoveMsg ); if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { if (msg.message == WM_QUIT) break; TranslateMessage(&msg); DispatchMessage(&msg); } else{ // run game code frameCount++; if(GetTime() > 1.0f) { fps = frameCount; frameCount = 0; StartTimer(); } frameTime = GetFrameTime(); DetectInput(frameTime); UpdateScene(frameTime); DrawScene(); } } return msg.wParam; } LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch( msg ) { case WM_KEYDOWN: if( wParam == VK_ESCAPE ){ DestroyWindow(hwnd); } return 0; case WM_DESTROY: PostQuitMessage(0); return 0; } return DefWindowProc(hwnd, msg, wParam, lParam); } Effects.fx struct Light { float3 dir; float4 ambient; float4 diffuse; }; cbuffer cbPerFrame { Light light; }; cbuffer cbPerObject { float4x4 WVP; float4x4 World; }; Texture2D ObjTexture; SamplerState ObjSamplerState; struct VS_OUTPUT { float4 Pos : SV_POSITION; float2 TexCoord : TEXCOORD; float3 normal : NORMAL; }; VS_OUTPUT VS(float4 inPos : POSITION, float2 inTexCoord : TEXCOORD, float3 normal : NORMAL) { VS_OUTPUT output; output.Pos = mul(inPos, WVP); output.normal = mul(normal, World); output.TexCoord = inTexCoord; return output; } float4 PS(VS_OUTPUT input) : SV_TARGET { input.normal = normalize(input.normal); float4 diffuse = ObjTexture.Sample( ObjSamplerState, input.TexCoord ); float3 finalColor; finalColor = diffuse * light.ambient; finalColor += saturate(dot(light.dir, input.normal) * light.diffuse * diffuse); return float4(finalColor, diffuse.a); } float4 D2D_PS(VS_OUTPUT input) : SV_TARGET { float4 diffuse = ObjTexture.Sample( ObjSamplerState, input.TexCoord ); return diffuse; }
Sign in to comment