Initial commit of starter code project
This commit is contained in:
parent
52cda213a7
commit
70a860173a
11 changed files with 1519 additions and 0 deletions
31
DX11Starter.sln
Normal file
31
DX11Starter.sln
Normal file
|
@ -0,0 +1,31 @@
|
|||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio Version 16
|
||||
VisualStudioVersion = 16.0.29209.62
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "DX11Starter", "DX11Starter.vcxproj", "{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|x64 = Debug|x64
|
||||
Debug|x86 = Debug|x86
|
||||
Release|x64 = Release|x64
|
||||
Release|x86 = Release|x86
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Debug|x64.ActiveCfg = Debug|x64
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Debug|x64.Build.0 = Debug|x64
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Debug|x86.ActiveCfg = Debug|Win32
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Debug|x86.Build.0 = Debug|Win32
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Release|x64.ActiveCfg = Release|x64
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Release|x64.Build.0 = Release|x64
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Release|x86.ActiveCfg = Release|Win32
|
||||
{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}.Release|x86.Build.0 = Release|Win32
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
EndGlobalSection
|
||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||
SolutionGuid = {0783F28E-8634-436C-B05A-98F92CB21105}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
160
DX11Starter.vcxproj
Normal file
160
DX11Starter.vcxproj
Normal file
|
@ -0,0 +1,160 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||
<ItemGroup Label="ProjectConfigurations">
|
||||
<ProjectConfiguration Include="Debug|Win32">
|
||||
<Configuration>Debug</Configuration>
|
||||
<Platform>Win32</Platform>
|
||||
</ProjectConfiguration>
|
||||
<ProjectConfiguration Include="Release|Win32">
|
||||
<Configuration>Release</Configuration>
|
||||
<Platform>Win32</Platform>
|
||||
</ProjectConfiguration>
|
||||
<ProjectConfiguration Include="Debug|x64">
|
||||
<Configuration>Debug</Configuration>
|
||||
<Platform>x64</Platform>
|
||||
</ProjectConfiguration>
|
||||
<ProjectConfiguration Include="Release|x64">
|
||||
<Configuration>Release</Configuration>
|
||||
<Platform>x64</Platform>
|
||||
</ProjectConfiguration>
|
||||
</ItemGroup>
|
||||
<PropertyGroup Label="Globals">
|
||||
<VCProjectVersion>16.0</VCProjectVersion>
|
||||
<ProjectGuid>{7B07137C-8E03-4F0C-BEDA-4C9915CD667C}</ProjectGuid>
|
||||
<RootNamespace>DX11Starter</RootNamespace>
|
||||
<WindowsTargetPlatformVersion>10.0</WindowsTargetPlatformVersion>
|
||||
</PropertyGroup>
|
||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>true</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<CharacterSet>MultiByte</CharacterSet>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>false</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<WholeProgramOptimization>true</WholeProgramOptimization>
|
||||
<CharacterSet>MultiByte</CharacterSet>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>true</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<CharacterSet>MultiByte</CharacterSet>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>false</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<WholeProgramOptimization>true</WholeProgramOptimization>
|
||||
<CharacterSet>MultiByte</CharacterSet>
|
||||
</PropertyGroup>
|
||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
|
||||
<ImportGroup Label="ExtensionSettings">
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="Shared">
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<PropertyGroup Label="UserMacros" />
|
||||
<PropertyGroup />
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<Optimization>Disabled</Optimization>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Windows</SubSystem>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<Optimization>Disabled</Optimization>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Windows</SubSystem>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<Optimization>MaxSpeed</Optimization>
|
||||
<FunctionLevelLinking>true</FunctionLevelLinking>
|
||||
<IntrinsicFunctions>true</IntrinsicFunctions>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Windows</SubSystem>
|
||||
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
||||
<OptimizeReferences>true</OptimizeReferences>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<Optimization>MaxSpeed</Optimization>
|
||||
<FunctionLevelLinking>true</FunctionLevelLinking>
|
||||
<IntrinsicFunctions>true</IntrinsicFunctions>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Windows</SubSystem>
|
||||
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
||||
<OptimizeReferences>true</OptimizeReferences>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemGroup>
|
||||
<ClCompile Include="DXCore.cpp" />
|
||||
<ClCompile Include="Game.cpp" />
|
||||
<ClCompile Include="Main.cpp" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ClInclude Include="DXCore.h" />
|
||||
<ClInclude Include="Game.h" />
|
||||
<ClInclude Include="Vertex.h" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<FxCompile Include="PixelShader.hlsl">
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">Pixel</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">5.0</ShaderModel>
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">Pixel</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">5.0</ShaderModel>
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">Pixel</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">5.0</ShaderModel>
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Release|x64'">Pixel</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Release|x64'">5.0</ShaderModel>
|
||||
</FxCompile>
|
||||
<FxCompile Include="VertexShader.hlsl">
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">Vertex</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">5.0</ShaderModel>
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">Vertex</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">5.0</ShaderModel>
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">Vertex</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">5.0</ShaderModel>
|
||||
<ShaderType Condition="'$(Configuration)|$(Platform)'=='Release|x64'">Vertex</ShaderType>
|
||||
<ShaderModel Condition="'$(Configuration)|$(Platform)'=='Release|x64'">5.0</ShaderModel>
|
||||
</FxCompile>
|
||||
</ItemGroup>
|
||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
|
||||
<ImportGroup Label="ExtensionTargets">
|
||||
</ImportGroup>
|
||||
</Project>
|
50
DX11Starter.vcxproj.filters
Normal file
50
DX11Starter.vcxproj.filters
Normal file
|
@ -0,0 +1,50 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||
<ItemGroup>
|
||||
<Filter Include="Source Files">
|
||||
<UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>
|
||||
<Extensions>cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx</Extensions>
|
||||
</Filter>
|
||||
<Filter Include="Header Files">
|
||||
<UniqueIdentifier>{93995380-89BD-4b04-88EB-625FBE52EBFB}</UniqueIdentifier>
|
||||
<Extensions>h;hh;hpp;hxx;hm;inl;inc;ipp;xsd</Extensions>
|
||||
</Filter>
|
||||
<Filter Include="Resource Files">
|
||||
<UniqueIdentifier>{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}</UniqueIdentifier>
|
||||
<Extensions>rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms</Extensions>
|
||||
</Filter>
|
||||
<Filter Include="Shaders">
|
||||
<UniqueIdentifier>{e76f01f5-08db-40d4-8720-b5b21f7ec0a3}</UniqueIdentifier>
|
||||
</Filter>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ClCompile Include="DXCore.cpp">
|
||||
<Filter>Source Files</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="Game.cpp">
|
||||
<Filter>Source Files</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="Main.cpp">
|
||||
<Filter>Source Files</Filter>
|
||||
</ClCompile>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ClInclude Include="Vertex.h">
|
||||
<Filter>Header Files</Filter>
|
||||
</ClInclude>
|
||||
<ClInclude Include="DXCore.h">
|
||||
<Filter>Header Files</Filter>
|
||||
</ClInclude>
|
||||
<ClInclude Include="Game.h">
|
||||
<Filter>Header Files</Filter>
|
||||
</ClInclude>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<FxCompile Include="PixelShader.hlsl">
|
||||
<Filter>Shaders</Filter>
|
||||
</FxCompile>
|
||||
<FxCompile Include="VertexShader.hlsl">
|
||||
<Filter>Shaders</Filter>
|
||||
</FxCompile>
|
||||
</ItemGroup>
|
||||
</Project>
|
673
DXCore.cpp
Normal file
673
DXCore.cpp
Normal file
|
@ -0,0 +1,673 @@
|
|||
#include "DXCore.h"
|
||||
|
||||
#include <WindowsX.h>
|
||||
#include <sstream>
|
||||
|
||||
// Define the static instance variable so our OS-level
|
||||
// message handling function below can talk to our object
|
||||
DXCore* DXCore::DXCoreInstance = 0;
|
||||
|
||||
// --------------------------------------------------------
|
||||
// The global callback function for handling windows OS-level messages.
|
||||
//
|
||||
// This needs to be a global function (not part of a class), but we want
|
||||
// to forward the parameters to our class to properly handle them.
|
||||
// --------------------------------------------------------
|
||||
LRESULT DXCore::WindowProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||||
{
|
||||
return DXCoreInstance->ProcessMessage(hWnd, uMsg, wParam, lParam);
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Constructor - Set up fields and timer
|
||||
//
|
||||
// hInstance - The application's OS-level handle (unique ID)
|
||||
// titleBarText - Text for the window's title bar
|
||||
// windowWidth - Width of the window's client (internal) area
|
||||
// windowHeight - Height of the window's client (internal) area
|
||||
// debugTitleBarStats - Show debug stats in the title bar, like FPS?
|
||||
// --------------------------------------------------------
|
||||
DXCore::DXCore(
|
||||
HINSTANCE hInstance, // The application's handle
|
||||
const char* titleBarText, // Text for the window's title bar
|
||||
unsigned int windowWidth, // Width of the window's client area
|
||||
unsigned int windowHeight, // Height of the window's client area
|
||||
bool debugTitleBarStats) // Show extra stats (fps) in title bar?
|
||||
{
|
||||
// Save a static reference to this object.
|
||||
// - Since the OS-level message function must be a non-member (global) function,
|
||||
// it won't be able to directly interact with our DXCore object otherwise.
|
||||
// - (Yes, a singleton might be a safer choice here).
|
||||
DXCoreInstance = this;
|
||||
|
||||
// Save params
|
||||
this->hInstance = hInstance;
|
||||
this->titleBarText = titleBarText;
|
||||
this->width = windowWidth;
|
||||
this->height = windowHeight;
|
||||
this->titleBarStats = debugTitleBarStats;
|
||||
|
||||
// Initialize fields
|
||||
this->hasFocus = true;
|
||||
|
||||
this->fpsFrameCount = 0;
|
||||
this->fpsTimeElapsed = 0.0f;
|
||||
this->currentTime = 0;
|
||||
this->deltaTime = 0;
|
||||
this->startTime = 0;
|
||||
this->totalTime = 0;
|
||||
|
||||
// Query performance counter for accurate timing information
|
||||
__int64 perfFreq;
|
||||
QueryPerformanceFrequency((LARGE_INTEGER*)&perfFreq);
|
||||
perfCounterSeconds = 1.0 / (double)perfFreq;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Destructor - Clean up (release) all DirectX references
|
||||
// --------------------------------------------------------
|
||||
DXCore::~DXCore()
|
||||
{
|
||||
// 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 DXCore
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Created the actual window for our application
|
||||
// --------------------------------------------------------
|
||||
HRESULT DXCore::InitWindow()
|
||||
{
|
||||
// Start window creation by filling out the
|
||||
// appropriate window class struct
|
||||
WNDCLASS wndClass = {}; // Zero out the memory
|
||||
wndClass.style = CS_HREDRAW | CS_VREDRAW; // Redraw on horizontal or vertical movement/adjustment
|
||||
wndClass.lpfnWndProc = DXCore::WindowProc;
|
||||
wndClass.cbClsExtra = 0;
|
||||
wndClass.cbWndExtra = 0;
|
||||
wndClass.hInstance = hInstance; // Our app's handle
|
||||
wndClass.hIcon = LoadIcon(NULL, IDI_APPLICATION); // Default icon
|
||||
wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); // Default arrow cursor
|
||||
wndClass.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH);
|
||||
wndClass.lpszMenuName = NULL;
|
||||
wndClass.lpszClassName = "Direct3DWindowClass";
|
||||
|
||||
// Attempt to register the window class we've defined
|
||||
if (!RegisterClass(&wndClass))
|
||||
{
|
||||
// Get the most recent error
|
||||
DWORD error = GetLastError();
|
||||
|
||||
// If the class exists, that's actually fine. Otherwise,
|
||||
// we can't proceed with the next step.
|
||||
if (error != ERROR_CLASS_ALREADY_EXISTS)
|
||||
return HRESULT_FROM_WIN32(error);
|
||||
}
|
||||
|
||||
// Adjust the width and height so the "client size" matches
|
||||
// the width and height given (the inner-area of the window)
|
||||
RECT clientRect;
|
||||
SetRect(&clientRect, 0, 0, width, height);
|
||||
AdjustWindowRect(
|
||||
&clientRect,
|
||||
WS_OVERLAPPEDWINDOW, // Has a title bar, border, min and max buttons, etc.
|
||||
false); // No menu bar
|
||||
|
||||
// Center the window to the screen
|
||||
RECT desktopRect;
|
||||
GetClientRect(GetDesktopWindow(), &desktopRect);
|
||||
int centeredX = (desktopRect.right / 2) - (clientRect.right / 2);
|
||||
int centeredY = (desktopRect.bottom / 2) - (clientRect.bottom / 2);
|
||||
|
||||
// Actually ask Windows to create the window itself
|
||||
// using our settings so far. This will return the
|
||||
// handle of the window, which we'll keep around for later
|
||||
hWnd = CreateWindow(
|
||||
wndClass.lpszClassName,
|
||||
titleBarText.c_str(),
|
||||
WS_OVERLAPPEDWINDOW,
|
||||
centeredX,
|
||||
centeredY,
|
||||
clientRect.right - clientRect.left, // Calculated width
|
||||
clientRect.bottom - clientRect.top, // Calculated height
|
||||
0, // No parent window
|
||||
0, // No menu
|
||||
hInstance, // The app's handle
|
||||
0); // No other windows in our application
|
||||
|
||||
// Ensure the window was created properly
|
||||
if (hWnd == NULL)
|
||||
{
|
||||
DWORD error = GetLastError();
|
||||
return HRESULT_FROM_WIN32(error);
|
||||
}
|
||||
|
||||
// The window exists but is not visible yet
|
||||
// We need to tell Windows to show it, and how to show it
|
||||
ShowWindow(hWnd, SW_SHOW);
|
||||
|
||||
// Return an "everything is ok" HRESULT value
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Initializes DirectX, which requires a window. This method
|
||||
// also creates several DirectX objects we'll need to start
|
||||
// drawing things to the screen.
|
||||
// --------------------------------------------------------
|
||||
HRESULT DXCore::InitDirectX()
|
||||
{
|
||||
// This will hold options for DirectX initialization
|
||||
unsigned int deviceFlags = 0;
|
||||
|
||||
#if defined(DEBUG) || defined(_DEBUG)
|
||||
// If we're in debug mode in visual studio, we also
|
||||
// want to make a "Debug DirectX Device" to see some
|
||||
// errors and warnings in Visual Studio's output window
|
||||
// when things go wrong!
|
||||
deviceFlags |= D3D11_CREATE_DEVICE_DEBUG;
|
||||
#endif
|
||||
|
||||
// Create a description of how our swap
|
||||
// chain should work
|
||||
DXGI_SWAP_CHAIN_DESC swapDesc = {};
|
||||
swapDesc.BufferCount = 2;
|
||||
swapDesc.BufferDesc.Width = width;
|
||||
swapDesc.BufferDesc.Height = height;
|
||||
swapDesc.BufferDesc.RefreshRate.Numerator = 60;
|
||||
swapDesc.BufferDesc.RefreshRate.Denominator = 1;
|
||||
swapDesc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM;
|
||||
swapDesc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED;
|
||||
swapDesc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED;
|
||||
swapDesc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT;
|
||||
swapDesc.Flags = 0;
|
||||
swapDesc.OutputWindow = hWnd;
|
||||
swapDesc.SampleDesc.Count = 1;
|
||||
swapDesc.SampleDesc.Quality = 0;
|
||||
swapDesc.SwapEffect = DXGI_SWAP_EFFECT_FLIP_DISCARD;
|
||||
swapDesc.Windowed = true;
|
||||
|
||||
// Result variable for below function calls
|
||||
HRESULT hr = S_OK;
|
||||
|
||||
// Attempt to initialize DirectX
|
||||
hr = D3D11CreateDeviceAndSwapChain(
|
||||
0, // Video adapter (physical GPU) to use, or null for default
|
||||
D3D_DRIVER_TYPE_HARDWARE, // We want to use the hardware (GPU)
|
||||
0, // Used when doing software rendering
|
||||
deviceFlags, // Any special options
|
||||
0, // Optional array of possible verisons we want as fallbacks
|
||||
0, // The number of fallbacks in the above param
|
||||
D3D11_SDK_VERSION, // Current version of the SDK
|
||||
&swapDesc, // Address of swap chain options
|
||||
swapChain.GetAddressOf(), // Pointer to our Swap Chain pointer
|
||||
device.GetAddressOf(), // Pointer to our Device pointer
|
||||
&dxFeatureLevel, // This will hold the actual feature level the app will use
|
||||
context.GetAddressOf()); // Pointer to our Device Context pointer
|
||||
if (FAILED(hr)) return hr;
|
||||
|
||||
// The above function created the back buffer render target
|
||||
// for us, but we need a reference to it
|
||||
ID3D11Texture2D* backBufferTexture = 0;
|
||||
swapChain->GetBuffer(
|
||||
0,
|
||||
__uuidof(ID3D11Texture2D),
|
||||
(void**)&backBufferTexture);
|
||||
|
||||
// Now that we have the texture, create a render target view
|
||||
// for the back buffer so we can render into it. Then release
|
||||
// our local reference to the texture, since we have the view.
|
||||
if (backBufferTexture != 0)
|
||||
{
|
||||
device->CreateRenderTargetView(
|
||||
backBufferTexture,
|
||||
0,
|
||||
backBufferRTV.GetAddressOf());
|
||||
backBufferTexture->Release();
|
||||
}
|
||||
|
||||
// Set up the description of the texture to use for the depth 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.Usage = D3D11_USAGE_DEFAULT;
|
||||
depthStencilDesc.BindFlags = D3D11_BIND_DEPTH_STENCIL;
|
||||
depthStencilDesc.CPUAccessFlags = 0;
|
||||
depthStencilDesc.MiscFlags = 0;
|
||||
depthStencilDesc.SampleDesc.Count = 1;
|
||||
depthStencilDesc.SampleDesc.Quality = 0;
|
||||
|
||||
// Create the depth buffer and its view, then
|
||||
// release our reference to the texture
|
||||
ID3D11Texture2D* depthBufferTexture = 0;
|
||||
device->CreateTexture2D(&depthStencilDesc, 0, &depthBufferTexture);
|
||||
if (depthBufferTexture != 0)
|
||||
{
|
||||
device->CreateDepthStencilView(
|
||||
depthBufferTexture,
|
||||
0,
|
||||
depthStencilView.GetAddressOf());
|
||||
depthBufferTexture->Release();
|
||||
}
|
||||
|
||||
// Bind the views to the pipeline, so rendering properly
|
||||
// uses their underlying textures
|
||||
context->OMSetRenderTargets(
|
||||
1,
|
||||
backBufferRTV.GetAddressOf(),
|
||||
depthStencilView.Get());
|
||||
|
||||
// Lastly, set up a viewport so we render into
|
||||
// to correct portion of the window
|
||||
D3D11_VIEWPORT viewport = {};
|
||||
viewport.TopLeftX = 0;
|
||||
viewport.TopLeftY = 0;
|
||||
viewport.Width = (float)width;
|
||||
viewport.Height = (float)height;
|
||||
viewport.MinDepth = 0.0f;
|
||||
viewport.MaxDepth = 1.0f;
|
||||
context->RSSetViewports(1, &viewport);
|
||||
|
||||
// Return the "everything is ok" HRESULT value
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// When the window is resized, the underlying
|
||||
// buffers (textures) must also be resized to match.
|
||||
//
|
||||
// If we don't do this, the window size and our rendering
|
||||
// resolution won't match up. This can result in odd
|
||||
// stretching/skewing.
|
||||
// --------------------------------------------------------
|
||||
void DXCore::OnResize()
|
||||
{
|
||||
// Release the buffers before resizing the swap chain
|
||||
backBufferRTV.Reset();
|
||||
depthStencilView.Reset();
|
||||
|
||||
// Resize the underlying swap chain buffers
|
||||
swapChain->ResizeBuffers(
|
||||
2,
|
||||
width,
|
||||
height,
|
||||
DXGI_FORMAT_R8G8B8A8_UNORM,
|
||||
0);
|
||||
|
||||
// Recreate the render target view for the back buffer
|
||||
// texture, then release our local texture reference
|
||||
ID3D11Texture2D* backBufferTexture = 0;
|
||||
swapChain->GetBuffer(0, __uuidof(ID3D11Texture2D), reinterpret_cast<void**>(&backBufferTexture));
|
||||
if (backBufferTexture != 0)
|
||||
{
|
||||
device->CreateRenderTargetView(
|
||||
backBufferTexture,
|
||||
0,
|
||||
backBufferRTV.ReleaseAndGetAddressOf()); // ReleaseAndGetAddressOf() cleans up the old object before giving us the pointer
|
||||
backBufferTexture->Release();
|
||||
}
|
||||
|
||||
// Set up the description of the texture to use for the depth 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.Usage = D3D11_USAGE_DEFAULT;
|
||||
depthStencilDesc.BindFlags = D3D11_BIND_DEPTH_STENCIL;
|
||||
depthStencilDesc.CPUAccessFlags = 0;
|
||||
depthStencilDesc.MiscFlags = 0;
|
||||
depthStencilDesc.SampleDesc.Count = 1;
|
||||
depthStencilDesc.SampleDesc.Quality = 0;
|
||||
|
||||
// Create the depth buffer and its view, then
|
||||
// release our reference to the texture
|
||||
ID3D11Texture2D* depthBufferTexture = 0;
|
||||
device->CreateTexture2D(&depthStencilDesc, 0, &depthBufferTexture);
|
||||
if (depthBufferTexture != 0)
|
||||
{
|
||||
device->CreateDepthStencilView(
|
||||
depthBufferTexture,
|
||||
0,
|
||||
depthStencilView.ReleaseAndGetAddressOf()); // ReleaseAndGetAddressOf() cleans up the old object before giving us the pointer
|
||||
depthBufferTexture->Release();
|
||||
}
|
||||
|
||||
// Bind the views to the pipeline, so rendering properly
|
||||
// uses their underlying textures
|
||||
context->OMSetRenderTargets(
|
||||
1,
|
||||
backBufferRTV.GetAddressOf(), // This requires a pointer to a pointer (an array of pointers), so we get the address of the pointer
|
||||
depthStencilView.Get());
|
||||
|
||||
// Lastly, set up a viewport so we render into
|
||||
// to correct portion of the window
|
||||
D3D11_VIEWPORT viewport = {};
|
||||
viewport.TopLeftX = 0;
|
||||
viewport.TopLeftY = 0;
|
||||
viewport.Width = (float)width;
|
||||
viewport.Height = (float)height;
|
||||
viewport.MinDepth = 0.0f;
|
||||
viewport.MaxDepth = 1.0f;
|
||||
context->RSSetViewports(1, &viewport);
|
||||
}
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// This is the main game loop, handling the following:
|
||||
// - OS-level messages coming in from Windows itself
|
||||
// - Calling update & draw back and forth, forever
|
||||
// --------------------------------------------------------
|
||||
HRESULT DXCore::Run()
|
||||
{
|
||||
// Grab the start time now that
|
||||
// the game loop is running
|
||||
__int64 now;
|
||||
QueryPerformanceCounter((LARGE_INTEGER*)&now);
|
||||
startTime = now;
|
||||
currentTime = now;
|
||||
previousTime = now;
|
||||
|
||||
// Give subclass a chance to initialize
|
||||
Init();
|
||||
|
||||
// Our overall game and message loop
|
||||
MSG msg = {};
|
||||
while (msg.message != WM_QUIT)
|
||||
{
|
||||
// Determine if there is a message waiting
|
||||
if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
|
||||
{
|
||||
// Translate and dispatch the message
|
||||
// to our custom WindowProc function
|
||||
TranslateMessage(&msg);
|
||||
DispatchMessage(&msg);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Update timer and title bar (if necessary)
|
||||
UpdateTimer();
|
||||
if(titleBarStats)
|
||||
UpdateTitleBarStats();
|
||||
|
||||
// The game loop
|
||||
Update(deltaTime, totalTime);
|
||||
Draw(deltaTime, totalTime);
|
||||
}
|
||||
}
|
||||
|
||||
// We'll end up here once we get a WM_QUIT message,
|
||||
// which usually comes from the user closing the window
|
||||
return (HRESULT)msg.wParam;
|
||||
}
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Sends an OS-level window close message to our process, which
|
||||
// will be handled by our message processing function
|
||||
// --------------------------------------------------------
|
||||
void DXCore::Quit()
|
||||
{
|
||||
PostMessage(this->hWnd, WM_CLOSE, NULL, NULL);
|
||||
}
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Uses high resolution time stamps to get very accurate
|
||||
// timing information, and calculates useful time stats
|
||||
// --------------------------------------------------------
|
||||
void DXCore::UpdateTimer()
|
||||
{
|
||||
// Grab the current time
|
||||
__int64 now;
|
||||
QueryPerformanceCounter((LARGE_INTEGER*)&now);
|
||||
currentTime = now;
|
||||
|
||||
// Calculate delta time and clamp to zero
|
||||
// - Could go negative if CPU goes into power save mode
|
||||
// or the process itself gets moved to another core
|
||||
deltaTime = max((float)((currentTime - previousTime) * perfCounterSeconds), 0.0f);
|
||||
|
||||
// Calculate the total time from start to now
|
||||
totalTime = (float)((currentTime - startTime) * perfCounterSeconds);
|
||||
|
||||
// Save current time for next frame
|
||||
previousTime = currentTime;
|
||||
}
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Updates the window's title bar with several stats once
|
||||
// per second, including:
|
||||
// - The window's width & height
|
||||
// - The current FPS and ms/frame
|
||||
// - The version of DirectX actually being used (usually 11)
|
||||
// --------------------------------------------------------
|
||||
void DXCore::UpdateTitleBarStats()
|
||||
{
|
||||
fpsFrameCount++;
|
||||
|
||||
// Only calc FPS and update title bar once per second
|
||||
float timeDiff = totalTime - fpsTimeElapsed;
|
||||
if (timeDiff < 1.0f)
|
||||
return;
|
||||
|
||||
// How long did each frame take? (Approx)
|
||||
float mspf = 1000.0f / (float)fpsFrameCount;
|
||||
|
||||
// Quick and dirty title bar text (mostly for debugging)
|
||||
std::ostringstream output;
|
||||
output.precision(6);
|
||||
output << titleBarText <<
|
||||
" Width: " << width <<
|
||||
" Height: " << height <<
|
||||
" FPS: " << fpsFrameCount <<
|
||||
" Frame Time: " << mspf << "ms";
|
||||
|
||||
// Append the version of DirectX the app is using
|
||||
switch (dxFeatureLevel)
|
||||
{
|
||||
case D3D_FEATURE_LEVEL_11_1: output << " DX 11.1"; break;
|
||||
case D3D_FEATURE_LEVEL_11_0: output << " DX 11.0"; break;
|
||||
case D3D_FEATURE_LEVEL_10_1: output << " DX 10.1"; break;
|
||||
case D3D_FEATURE_LEVEL_10_0: output << " DX 10.0"; break;
|
||||
case D3D_FEATURE_LEVEL_9_3: output << " DX 9.3"; break;
|
||||
case D3D_FEATURE_LEVEL_9_2: output << " DX 9.2"; break;
|
||||
case D3D_FEATURE_LEVEL_9_1: output << " DX 9.1"; break;
|
||||
default: output << " DX ???"; break;
|
||||
}
|
||||
|
||||
// Actually update the title bar and reset fps data
|
||||
SetWindowText(hWnd, output.str().c_str());
|
||||
fpsFrameCount = 0;
|
||||
fpsTimeElapsed += 1.0f;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Allocates a console window we can print to for debugging
|
||||
//
|
||||
// bufferLines - Number of lines in the overall console buffer
|
||||
// bufferColumns - Numbers of columns in the overall console buffer
|
||||
// windowLines - Number of lines visible at once in the window
|
||||
// windowColumns - Number of columns visible at once in the window
|
||||
// --------------------------------------------------------
|
||||
void DXCore::CreateConsoleWindow(int bufferLines, int bufferColumns, int windowLines, int windowColumns)
|
||||
{
|
||||
// Our temp console info struct
|
||||
CONSOLE_SCREEN_BUFFER_INFO coninfo;
|
||||
|
||||
// Get the console info and set the number of lines
|
||||
AllocConsole();
|
||||
GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &coninfo);
|
||||
coninfo.dwSize.Y = bufferLines;
|
||||
coninfo.dwSize.X = bufferColumns;
|
||||
SetConsoleScreenBufferSize(GetStdHandle(STD_OUTPUT_HANDLE), coninfo.dwSize);
|
||||
|
||||
SMALL_RECT rect;
|
||||
rect.Left = 0;
|
||||
rect.Top = 0;
|
||||
rect.Right = windowColumns;
|
||||
rect.Bottom = windowLines;
|
||||
SetConsoleWindowInfo(GetStdHandle(STD_OUTPUT_HANDLE), TRUE, &rect);
|
||||
|
||||
FILE *stream;
|
||||
freopen_s(&stream, "CONIN$", "r", stdin);
|
||||
freopen_s(&stream, "CONOUT$", "w", stdout);
|
||||
freopen_s(&stream, "CONOUT$", "w", stderr);
|
||||
|
||||
// Prevent accidental console window close
|
||||
HWND consoleHandle = GetConsoleWindow();
|
||||
HMENU hmenu = GetSystemMenu(consoleHandle, FALSE);
|
||||
EnableMenuItem(hmenu, SC_CLOSE, MF_GRAYED);
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------------
|
||||
// Gets the actual path to this executable
|
||||
//
|
||||
// - As it turns out, the relative path for a program is different when
|
||||
// running through VS and when running the .exe directly, which makes
|
||||
// it a pain to properly load files external files (like textures)
|
||||
// - Running through VS: Current Dir is the *project folder*
|
||||
// - Running from .exe: Current Dir is the .exe's folder
|
||||
// - This has nothing to do with DEBUG and RELEASE modes - it's purely a
|
||||
// Visual Studio "thing", and isn't obvious unless you know to look
|
||||
// for it. In fact, it could be fixed by changing a setting in VS, but
|
||||
// the option is stored in a user file (.suo), which is ignored by most
|
||||
// version control packages by default. Meaning: the option must be
|
||||
// changed every on every PC. Ugh. So instead, here's a helper.
|
||||
// --------------------------------------------------------------------------
|
||||
std::string DXCore::GetExePath()
|
||||
{
|
||||
// Assume the path is just the "current directory" for now
|
||||
std::string path = ".\\";
|
||||
|
||||
// Get the real, full path to this executable
|
||||
char currentDir[1024] = {};
|
||||
GetModuleFileName(0, currentDir, 1024);
|
||||
|
||||
// Find the location of the last slash charaacter
|
||||
char* lastSlash = strrchr(currentDir, '\\');
|
||||
if (lastSlash)
|
||||
{
|
||||
// End the string at the last slash character, essentially
|
||||
// chopping off the exe's file name. Remember, c-strings
|
||||
// are null-terminated, so putting a "zero" character in
|
||||
// there simply denotes the end of the string.
|
||||
*lastSlash = 0;
|
||||
|
||||
// Set the remainder as the path
|
||||
path = currentDir;
|
||||
}
|
||||
|
||||
// Toss back whatever we've found
|
||||
return path;
|
||||
}
|
||||
|
||||
|
||||
// ---------------------------------------------------
|
||||
// Same as GetExePath(), except it returns a wide character
|
||||
// string, which most of the Windows API requires.
|
||||
// ---------------------------------------------------
|
||||
std::wstring DXCore::GetExePath_Wide()
|
||||
{
|
||||
// Grab the path as a standard string
|
||||
std::string path = GetExePath();
|
||||
|
||||
// Convert to a wide string
|
||||
wchar_t widePath[1024] = {};
|
||||
mbstowcs_s(0, widePath, path.c_str(), 1024);
|
||||
|
||||
// Create a wstring for it and return
|
||||
return std::wstring(widePath);
|
||||
}
|
||||
|
||||
|
||||
// ----------------------------------------------------
|
||||
// Gets the full path to a given file. NOTE: This does
|
||||
// NOT "find" the file, it simply concatenates the given
|
||||
// relative file path onto the executable's path
|
||||
// ----------------------------------------------------
|
||||
std::string DXCore::GetFullPathTo(std::string relativeFilePath)
|
||||
{
|
||||
return GetExePath() + "\\" + relativeFilePath;
|
||||
}
|
||||
|
||||
|
||||
|
||||
// ----------------------------------------------------
|
||||
// Same as GetFullPathTo, but with wide char strings.
|
||||
//
|
||||
// Gets the full path to a given file. NOTE: This does
|
||||
// NOT "find" the file, it simply concatenates the given
|
||||
// relative file path onto the executable's path
|
||||
// ----------------------------------------------------
|
||||
std::wstring DXCore::GetFullPathTo_Wide(std::wstring relativeFilePath)
|
||||
{
|
||||
return GetExePath_Wide() + L"\\" + relativeFilePath;
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Handles messages that are sent to our window by the
|
||||
// operating system. Ignoring these messages would cause
|
||||
// our program to hang and Windows would think it was
|
||||
// unresponsive.
|
||||
// --------------------------------------------------------
|
||||
LRESULT DXCore::ProcessMessage(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||||
{
|
||||
// Check the incoming message and handle any we care about
|
||||
switch (uMsg)
|
||||
{
|
||||
// This is the message that signifies the window closing
|
||||
case WM_DESTROY:
|
||||
PostQuitMessage(0); // Send a quit message to our own program
|
||||
return 0;
|
||||
|
||||
// Prevent beeping when we "alt-enter" into fullscreen
|
||||
case WM_MENUCHAR:
|
||||
return MAKELRESULT(0, MNC_CLOSE);
|
||||
|
||||
// Prevent the overall window from becoming too small
|
||||
case WM_GETMINMAXINFO:
|
||||
((MINMAXINFO*)lParam)->ptMinTrackSize.x = 200;
|
||||
((MINMAXINFO*)lParam)->ptMinTrackSize.y = 200;
|
||||
return 0;
|
||||
|
||||
// Sent when the window size changes
|
||||
case WM_SIZE:
|
||||
// Don't adjust anything when minimizing,
|
||||
// since we end up with a width/height of zero
|
||||
// and that doesn't play well with the GPU
|
||||
if (wParam == SIZE_MINIMIZED)
|
||||
return 0;
|
||||
|
||||
// Save the new client area dimensions.
|
||||
width = LOWORD(lParam);
|
||||
height = HIWORD(lParam);
|
||||
|
||||
// If DX is initialized, resize
|
||||
// our required buffers
|
||||
if (device)
|
||||
OnResize();
|
||||
|
||||
return 0;
|
||||
|
||||
// Is our focus state changing?
|
||||
case WM_SETFOCUS: hasFocus = true; return 0;
|
||||
case WM_KILLFOCUS: hasFocus = false; return 0;
|
||||
case WM_ACTIVATE: hasFocus = (LOWORD(wParam) != WA_INACTIVE); return 0;
|
||||
}
|
||||
|
||||
// Let Windows handle any messages we're not touching
|
||||
return DefWindowProc(hWnd, uMsg, wParam, lParam);
|
||||
}
|
||||
|
97
DXCore.h
Normal file
97
DXCore.h
Normal file
|
@ -0,0 +1,97 @@
|
|||
#pragma once
|
||||
|
||||
#include <Windows.h>
|
||||
#include <d3d11.h>
|
||||
#include <string>
|
||||
#include <wrl/client.h> // Used for ComPtr - a smart pointer for COM objects
|
||||
|
||||
// We can include the correct library files here
|
||||
// instead of in Visual Studio settings if we want
|
||||
#pragma comment(lib, "d3d11.lib")
|
||||
|
||||
class DXCore
|
||||
{
|
||||
public:
|
||||
DXCore(
|
||||
HINSTANCE hInstance, // The application's handle
|
||||
const char* titleBarText, // Text for the window's title bar
|
||||
unsigned int windowWidth, // Width of the window's client area
|
||||
unsigned int windowHeight, // Height of the window's client area
|
||||
bool debugTitleBarStats); // Show extra stats (fps) in title bar?
|
||||
~DXCore();
|
||||
|
||||
// Static requirements for OS-level message processing
|
||||
static DXCore* DXCoreInstance;
|
||||
static LRESULT CALLBACK WindowProc(
|
||||
HWND hWnd, // Window handle
|
||||
UINT uMsg, // Message
|
||||
WPARAM wParam, // Message's first parameter
|
||||
LPARAM lParam // Message's second parameter
|
||||
);
|
||||
|
||||
// Internal method for message handling
|
||||
LRESULT ProcessMessage(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam);
|
||||
|
||||
// Initialization and game-loop related methods
|
||||
HRESULT InitWindow();
|
||||
HRESULT InitDirectX();
|
||||
HRESULT Run();
|
||||
void Quit();
|
||||
virtual void OnResize();
|
||||
|
||||
// Pure virtual methods for setup and game functionality
|
||||
virtual void Init() = 0;
|
||||
virtual void Update(float deltaTime, float totalTime) = 0;
|
||||
virtual void Draw(float deltaTime, float totalTime) = 0;
|
||||
|
||||
protected:
|
||||
HINSTANCE hInstance; // The handle to the application
|
||||
HWND hWnd; // The handle to the window itself
|
||||
std::string titleBarText; // Custom text in window's title bar
|
||||
bool titleBarStats; // Show extra stats in title bar?
|
||||
|
||||
// Size of the window's client area
|
||||
unsigned int width;
|
||||
unsigned int height;
|
||||
|
||||
// Does our window currently have focus?
|
||||
// Helpful if we want to pause while not the active window
|
||||
bool hasFocus;
|
||||
|
||||
// DirectX related objects and variables
|
||||
D3D_FEATURE_LEVEL dxFeatureLevel;
|
||||
Microsoft::WRL::ComPtr<IDXGISwapChain> swapChain;
|
||||
Microsoft::WRL::ComPtr<ID3D11Device> device;
|
||||
Microsoft::WRL::ComPtr<ID3D11DeviceContext> context;
|
||||
|
||||
Microsoft::WRL::ComPtr<ID3D11RenderTargetView> backBufferRTV;
|
||||
Microsoft::WRL::ComPtr<ID3D11DepthStencilView> depthStencilView;
|
||||
|
||||
// Helper function for allocating a console window
|
||||
void CreateConsoleWindow(int bufferLines, int bufferColumns, int windowLines, int windowColumns);
|
||||
|
||||
// Helpers for determining the actual path to the executable
|
||||
std::string GetExePath();
|
||||
std::wstring GetExePath_Wide();
|
||||
|
||||
std::string GetFullPathTo(std::string relativeFilePath);
|
||||
std::wstring GetFullPathTo_Wide(std::wstring relativeFilePath);
|
||||
|
||||
|
||||
private:
|
||||
// Timing related data
|
||||
double perfCounterSeconds;
|
||||
float totalTime;
|
||||
float deltaTime;
|
||||
__int64 startTime;
|
||||
__int64 currentTime;
|
||||
__int64 previousTime;
|
||||
|
||||
// FPS calculation
|
||||
int fpsFrameCount;
|
||||
float fpsTimeElapsed;
|
||||
|
||||
void UpdateTimer(); // Updates the timer for this frame
|
||||
void UpdateTitleBarStats(); // Puts debug info in the title bar
|
||||
};
|
||||
|
311
Game.cpp
Normal file
311
Game.cpp
Normal file
|
@ -0,0 +1,311 @@
|
|||
#include "Game.h"
|
||||
#include "Vertex.h"
|
||||
|
||||
// Needed for a helper function to read compiled shader files from the hard drive
|
||||
#pragma comment(lib, "d3dcompiler.lib")
|
||||
#include <d3dcompiler.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?
|
||||
{
|
||||
|
||||
#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
|
||||
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// 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();
|
||||
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()
|
||||
{
|
||||
// Blob for reading raw data
|
||||
// - This is a simplified way of handling raw data
|
||||
ID3DBlob* shaderBlob;
|
||||
|
||||
// Read our compiled vertex shader code into a blob
|
||||
// - Essentially just "open the file and plop its contents here"
|
||||
D3DReadFileToBlob(
|
||||
GetFullPathTo_Wide(L"VertexShader.cso").c_str(), // Using a custom helper for file paths
|
||||
&shaderBlob);
|
||||
|
||||
// Create a vertex shader from the information we
|
||||
// have read into the blob above
|
||||
// - A blob can give a pointer to its contents, and knows its own size
|
||||
device->CreateVertexShader(
|
||||
shaderBlob->GetBufferPointer(), // Get a pointer to the blob's contents
|
||||
shaderBlob->GetBufferSize(), // How big is that data?
|
||||
0, // No classes in this shader
|
||||
vertexShader.GetAddressOf()); // The address of the ID3D11VertexShader*
|
||||
|
||||
|
||||
// Create an input layout that describes the vertex format
|
||||
// used by the vertex shader we're using
|
||||
// - This is used by the pipeline to know how to interpret the raw data
|
||||
// sitting inside a vertex buffer
|
||||
// - Doing this NOW because it requires a vertex shader's byte code to verify against!
|
||||
// - Luckily, we already have that loaded (the blob above)
|
||||
D3D11_INPUT_ELEMENT_DESC inputElements[2] = {};
|
||||
|
||||
// Set up the first element - a position, which is 3 float values
|
||||
inputElements[0].Format = DXGI_FORMAT_R32G32B32_FLOAT; // Most formats are described as color channels; really it just means "Three 32-bit floats"
|
||||
inputElements[0].SemanticName = "POSITION"; // This is "POSITION" - needs to match the semantics in our vertex shader input!
|
||||
inputElements[0].AlignedByteOffset = D3D11_APPEND_ALIGNED_ELEMENT; // How far into the vertex is this? Assume it's after the previous element
|
||||
|
||||
// Set up the second element - a color, which is 4 more float values
|
||||
inputElements[1].Format = DXGI_FORMAT_R32G32B32A32_FLOAT; // 4x 32-bit floats
|
||||
inputElements[1].SemanticName = "COLOR"; // Match our vertex shader input!
|
||||
inputElements[1].AlignedByteOffset = D3D11_APPEND_ALIGNED_ELEMENT; // After the previous element
|
||||
|
||||
// Create the input layout, verifying our description against actual shader code
|
||||
device->CreateInputLayout(
|
||||
inputElements, // An array of descriptions
|
||||
2, // How many elements in that array
|
||||
shaderBlob->GetBufferPointer(), // Pointer to the code of a shader that uses this layout
|
||||
shaderBlob->GetBufferSize(), // Size of the shader code that uses this layout
|
||||
inputLayout.GetAddressOf()); // Address of the resulting ID3D11InputLayout*
|
||||
|
||||
|
||||
|
||||
// Read and create the pixel shader
|
||||
// - Reusing the same blob here, since we're done with the vert shader code
|
||||
D3DReadFileToBlob(
|
||||
GetFullPathTo_Wide(L"PixelShader.cso").c_str(), // Using a custom helper for file paths
|
||||
&shaderBlob);
|
||||
|
||||
device->CreatePixelShader(
|
||||
shaderBlob->GetBufferPointer(),
|
||||
shaderBlob->GetBufferSize(),
|
||||
0,
|
||||
pixelShader.GetAddressOf());
|
||||
}
|
||||
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Creates the geometry we're going to draw - a single triangle for now
|
||||
// --------------------------------------------------------
|
||||
void Game::CreateBasicGeometry()
|
||||
{
|
||||
// Create some temporary variables to represent colors
|
||||
// - Not necessary, just makes things more readable
|
||||
XMFLOAT4 red = XMFLOAT4(1.0f, 0.0f, 0.0f, 1.0f);
|
||||
XMFLOAT4 green = XMFLOAT4(0.0f, 1.0f, 0.0f, 1.0f);
|
||||
XMFLOAT4 blue = XMFLOAT4(0.0f, 0.0f, 1.0f, 1.0f);
|
||||
|
||||
// Set up the vertices of the triangle we would like to draw
|
||||
// - We're going to copy this array, exactly as it exists in memory
|
||||
// over to a DirectX-controlled data structure (the vertex buffer)
|
||||
// - Note: Since we don't have a camera or really any concept of
|
||||
// a "3d world" yet, we're simply describing positions within the
|
||||
// bounds of how the rasterizer sees our screen: [-1 to +1] on X and Y
|
||||
// - This means (0,0) is at the very center of the screen.
|
||||
// - These are known as "Normalized Device Coordinates" or "Homogeneous
|
||||
// Screen Coords", which are ways to describe a position without
|
||||
// knowing the exact size (in pixels) of the image/window/etc.
|
||||
// - Long story short: Resizing the window also resizes the triangle,
|
||||
// since we're describing the triangle in terms of the window itself
|
||||
Vertex vertices[] =
|
||||
{
|
||||
{ XMFLOAT3(+0.0f, +0.5f, +0.0f), red },
|
||||
{ XMFLOAT3(+0.5f, -0.5f, +0.0f), blue },
|
||||
{ XMFLOAT3(-0.5f, -0.5f, +0.0f), green },
|
||||
};
|
||||
|
||||
// Set up the indices, which tell us which vertices to use and in which order
|
||||
// - This is somewhat redundant for just 3 vertices (it's a simple example)
|
||||
// - Indices are technically not required if the vertices are in the buffer
|
||||
// in the correct order and each one will be used exactly once
|
||||
// - But just to see how it's done...
|
||||
unsigned int indices[] = { 0, 1, 2 };
|
||||
|
||||
|
||||
// Create the VERTEX BUFFER description -----------------------------------
|
||||
// - The description is created on the stack because we only need
|
||||
// it to create the buffer. The description is then useless.
|
||||
D3D11_BUFFER_DESC vbd;
|
||||
vbd.Usage = D3D11_USAGE_IMMUTABLE;
|
||||
vbd.ByteWidth = sizeof(Vertex) * 3; // 3 = number of vertices in the buffer
|
||||
vbd.BindFlags = D3D11_BIND_VERTEX_BUFFER; // Tells DirectX this is a vertex buffer
|
||||
vbd.CPUAccessFlags = 0;
|
||||
vbd.MiscFlags = 0;
|
||||
vbd.StructureByteStride = 0;
|
||||
|
||||
// Create the proper struct to hold the initial vertex data
|
||||
// - This is how we put the initial data into the buffer
|
||||
D3D11_SUBRESOURCE_DATA initialVertexData;
|
||||
initialVertexData.pSysMem = vertices;
|
||||
|
||||
// Actually create the buffer with the initial data
|
||||
// - Once we do this, we'll NEVER CHANGE THE BUFFER AGAIN
|
||||
device->CreateBuffer(&vbd, &initialVertexData, vertexBuffer.GetAddressOf());
|
||||
|
||||
|
||||
|
||||
// Create the INDEX BUFFER description ------------------------------------
|
||||
// - The description is created on the stack because we only need
|
||||
// it to create the buffer. The description is then useless.
|
||||
D3D11_BUFFER_DESC ibd;
|
||||
ibd.Usage = D3D11_USAGE_IMMUTABLE;
|
||||
ibd.ByteWidth = sizeof(unsigned int) * 3; // 3 = number of indices in the buffer
|
||||
ibd.BindFlags = D3D11_BIND_INDEX_BUFFER; // Tells DirectX this is an index buffer
|
||||
ibd.CPUAccessFlags = 0;
|
||||
ibd.MiscFlags = 0;
|
||||
ibd.StructureByteStride = 0;
|
||||
|
||||
// Create the proper struct to hold the initial index data
|
||||
// - This is how we put the initial data into the buffer
|
||||
D3D11_SUBRESOURCE_DATA initialIndexData;
|
||||
initialIndexData.pSysMem = indices;
|
||||
|
||||
// Actually create the buffer with the initial data
|
||||
// - Once we do this, we'll NEVER CHANGE THE BUFFER AGAIN
|
||||
device->CreateBuffer(&ibd, &initialIndexData, indexBuffer.GetAddressOf());
|
||||
|
||||
}
|
||||
|
||||
|
||||
// --------------------------------------------------------
|
||||
// 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();
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Update your game here - user input, move objects, AI, etc.
|
||||
// --------------------------------------------------------
|
||||
void Game::Update(float deltaTime, float totalTime)
|
||||
{
|
||||
// Quit if the escape key is pressed
|
||||
if (GetAsyncKeyState(VK_ESCAPE))
|
||||
Quit();
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Clear the screen, redraw everything, present to the user
|
||||
// --------------------------------------------------------
|
||||
void Game::Draw(float deltaTime, float totalTime)
|
||||
{
|
||||
// Background color (Cornflower Blue in this case) for clearing
|
||||
const float color[4] = { 0.4f, 0.6f, 0.75f, 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);
|
||||
|
||||
|
||||
// Set the vertex and pixel shaders to use for the next Draw() command
|
||||
// - These don't technically need to be set every frame
|
||||
// - Once you start applying different shaders to different objects,
|
||||
// you'll need to swap the current shaders before each draw
|
||||
context->VSSetShader(vertexShader.Get(), 0, 0);
|
||||
context->PSSetShader(pixelShader.Get(), 0, 0);
|
||||
|
||||
|
||||
// Ensure the pipeline knows how to interpret the data (numbers)
|
||||
// from the vertex buffer.
|
||||
// - If all of your 3D models use the exact same vertex layout,
|
||||
// this could simply be done once in Init()
|
||||
// - However, this isn't always the case (but might be for this course)
|
||||
context->IASetInputLayout(inputLayout.Get());
|
||||
|
||||
|
||||
// Set buffers in the input assembler
|
||||
// - Do this ONCE PER OBJECT you're drawing, since each object might
|
||||
// have different geometry.
|
||||
// - for this demo, this step *could* simply be done once during Init(),
|
||||
// but I'm doing it here because it's often done multiple times per frame
|
||||
// in a larger application/game
|
||||
UINT stride = sizeof(Vertex);
|
||||
UINT offset = 0;
|
||||
context->IASetVertexBuffers(0, 1, vertexBuffer.GetAddressOf(), &stride, &offset);
|
||||
context->IASetIndexBuffer(indexBuffer.Get(), DXGI_FORMAT_R32_UINT, 0);
|
||||
|
||||
|
||||
// Finally do the actual drawing
|
||||
// - Do this ONCE PER OBJECT you intend to draw
|
||||
// - This will use all of the currently set DirectX "stuff" (shaders, buffers, etc)
|
||||
// - DrawIndexed() uses the currently set INDEX BUFFER to look up corresponding
|
||||
// vertices in the currently set VERTEX BUFFER
|
||||
context->DrawIndexed(
|
||||
3, // The number of indices to use (we could draw a subset if we wanted)
|
||||
0, // Offset to the first index we want to use
|
||||
0); // Offset to add to each index when looking up vertices
|
||||
|
||||
|
||||
|
||||
// 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(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());
|
||||
}
|
44
Game.h
Normal file
44
Game.h
Normal file
|
@ -0,0 +1,44 @@
|
|||
#pragma once
|
||||
|
||||
#include "DXCore.h"
|
||||
#include <DirectXMath.h>
|
||||
#include <wrl/client.h> // Used for ComPtr - a smart pointer for COM objects
|
||||
|
||||
class Game
|
||||
: public DXCore
|
||||
{
|
||||
|
||||
public:
|
||||
Game(HINSTANCE hInstance);
|
||||
~Game();
|
||||
|
||||
// Overridden setup and game loop methods, which
|
||||
// will be called automatically
|
||||
void Init();
|
||||
void OnResize();
|
||||
void Update(float deltaTime, float totalTime);
|
||||
void Draw(float deltaTime, float totalTime);
|
||||
|
||||
private:
|
||||
|
||||
// Initialization helper methods - feel free to customize, combine, etc.
|
||||
void LoadShaders();
|
||||
void CreateBasicGeometry();
|
||||
|
||||
|
||||
// Note the usage of ComPtr below
|
||||
// - This is a smart pointer for objects that abide by the
|
||||
// Component Object Model, which DirectX objects do
|
||||
// - More info here: https://github.com/Microsoft/DirectXTK/wiki/ComPtr
|
||||
|
||||
// Buffers to hold actual geometry data
|
||||
Microsoft::WRL::ComPtr<ID3D11Buffer> vertexBuffer;
|
||||
Microsoft::WRL::ComPtr<ID3D11Buffer> indexBuffer;
|
||||
|
||||
// Shaders and shader-related constructs
|
||||
Microsoft::WRL::ComPtr<ID3D11PixelShader> pixelShader;
|
||||
Microsoft::WRL::ComPtr<ID3D11VertexShader> vertexShader;
|
||||
Microsoft::WRL::ComPtr<ID3D11InputLayout> inputLayout;
|
||||
|
||||
};
|
||||
|
41
Main.cpp
Normal file
41
Main.cpp
Normal file
|
@ -0,0 +1,41 @@
|
|||
|
||||
#include <Windows.h>
|
||||
#include "Game.h"
|
||||
|
||||
// --------------------------------------------------------
|
||||
// Entry point for a graphical (non-console) Windows application
|
||||
// --------------------------------------------------------
|
||||
int WINAPI WinMain(
|
||||
_In_ HINSTANCE hInstance, // The handle to this app's instance
|
||||
_In_opt_ HINSTANCE hPrevInstance, // A handle to the previous instance of the app (always NULL)
|
||||
_In_ LPSTR lpCmdLine, // Command line params
|
||||
_In_ int nCmdShow) // How the window should be shown (we ignore this)
|
||||
{
|
||||
#if defined(DEBUG) | defined(_DEBUG)
|
||||
// Enable memory leak detection as a quick and dirty
|
||||
// way of determining if we forgot to clean something up
|
||||
// - You may want to use something more advanced, like Visual Leak Detector
|
||||
_CrtSetDbgFlag( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
|
||||
#endif
|
||||
|
||||
// Create the Game object using
|
||||
// the app handle we got from WinMain
|
||||
Game dxGame(hInstance);
|
||||
|
||||
// Result variable for function calls below
|
||||
HRESULT hr = S_OK;
|
||||
|
||||
// Attempt to create the window for our program, and
|
||||
// exit early if something failed
|
||||
hr = dxGame.InitWindow();
|
||||
if(FAILED(hr)) return hr;
|
||||
|
||||
// Attempt to initialize DirectX, and exit
|
||||
// early if something failed
|
||||
hr = dxGame.InitDirectX();
|
||||
if(FAILED(hr)) return hr;
|
||||
|
||||
// Begin the message and game loop, and then return
|
||||
// whatever we get back once the game loop is over
|
||||
return dxGame.Run();
|
||||
}
|
34
PixelShader.hlsl
Normal file
34
PixelShader.hlsl
Normal file
|
@ -0,0 +1,34 @@
|
|||
|
||||
// Struct representing the data we expect to receive from earlier pipeline stages
|
||||
// - Should match the output of our corresponding vertex shader
|
||||
// - The name of the struct itself is unimportant
|
||||
// - The variable names don't have to match other shaders (just the semantics)
|
||||
// - Each variable must have a semantic, which defines its usage
|
||||
struct VertexToPixel
|
||||
{
|
||||
// Data type
|
||||
// |
|
||||
// | Name Semantic
|
||||
// | | |
|
||||
// v v v
|
||||
float4 position : SV_POSITION;
|
||||
float4 color : COLOR;
|
||||
};
|
||||
|
||||
// --------------------------------------------------------
|
||||
// The entry point (main method) for our pixel shader
|
||||
//
|
||||
// - Input is the data coming down the pipeline (defined by the struct)
|
||||
// - Output is a single color (float4)
|
||||
// - Has a special semantic (SV_TARGET), which means
|
||||
// "put the output of this into the current render target"
|
||||
// - Named "main" because that's the default the shader compiler looks for
|
||||
// --------------------------------------------------------
|
||||
float4 main(VertexToPixel input) : SV_TARGET
|
||||
{
|
||||
// Just return the input color
|
||||
// - This color (like most values passing through the rasterizer) is
|
||||
// interpolated for each pixel between the corresponding vertices
|
||||
// of the triangle we're rendering
|
||||
return input.color;
|
||||
}
|
14
Vertex.h
Normal file
14
Vertex.h
Normal file
|
@ -0,0 +1,14 @@
|
|||
#pragma once
|
||||
|
||||
#include <DirectXMath.h>
|
||||
|
||||
// --------------------------------------------------------
|
||||
// A custom vertex definition
|
||||
//
|
||||
// You will eventually ADD TO this, and/or make more of these!
|
||||
// --------------------------------------------------------
|
||||
struct Vertex
|
||||
{
|
||||
DirectX::XMFLOAT3 Position; // The position of the vertex
|
||||
DirectX::XMFLOAT4 Color; // The color of the vertex
|
||||
};
|
64
VertexShader.hlsl
Normal file
64
VertexShader.hlsl
Normal file
|
@ -0,0 +1,64 @@
|
|||
|
||||
// Struct representing a single vertex worth of data
|
||||
// - This should match the vertex definition in our C++ code
|
||||
// - By "match", I mean the size, order and number of members
|
||||
// - The name of the struct itself is unimportant, but should be descriptive
|
||||
// - Each variable must have a semantic, which defines its usage
|
||||
struct VertexShaderInput
|
||||
{
|
||||
// Data type
|
||||
// |
|
||||
// | Name Semantic
|
||||
// | | |
|
||||
// v v v
|
||||
float3 position : POSITION; // XYZ position
|
||||
float4 color : COLOR; // RGBA color
|
||||
};
|
||||
|
||||
// Struct representing the data we're sending down the pipeline
|
||||
// - Should match our pixel shader's input (hence the name: Vertex to Pixel)
|
||||
// - At a minimum, we need a piece of data defined tagged as SV_POSITION
|
||||
// - The name of the struct itself is unimportant, but should be descriptive
|
||||
// - Each variable must have a semantic, which defines its usage
|
||||
struct VertexToPixel
|
||||
{
|
||||
// Data type
|
||||
// |
|
||||
// | Name Semantic
|
||||
// | | |
|
||||
// v v v
|
||||
float4 position : SV_POSITION; // XYZW position (System Value Position)
|
||||
float4 color : COLOR; // RGBA color
|
||||
};
|
||||
|
||||
// --------------------------------------------------------
|
||||
// The entry point (main method) for our vertex shader
|
||||
//
|
||||
// - Input is exactly one vertex worth of data (defined by a struct)
|
||||
// - Output is a single struct of data to pass down the pipeline
|
||||
// - Named "main" because that's the default the shader compiler looks for
|
||||
// --------------------------------------------------------
|
||||
VertexToPixel main( VertexShaderInput input )
|
||||
{
|
||||
// Set up output struct
|
||||
VertexToPixel output;
|
||||
|
||||
// Here we're essentially passing the input position directly through to the next
|
||||
// stage (rasterizer), though it needs to be a 4-component vector now.
|
||||
// - To be considered within the bounds of the screen, the X and Y components
|
||||
// must be between -1 and 1.
|
||||
// - The Z component must be between 0 and 1.
|
||||
// - Each of these components is then automatically divided by the W component,
|
||||
// which we're leaving at 1.0 for now (this is more useful when dealing with
|
||||
// a perspective projection matrix, which we'll get to in the future).
|
||||
output.position = float4(input.position, 1.0f);
|
||||
|
||||
// Pass the color through
|
||||
// - The values will be interpolated per-pixel by the rasterizer
|
||||
// - We don't need to alter it here, but we do need to send it to the pixel shader
|
||||
output.color = input.color;
|
||||
|
||||
// Whatever we return will make its way through the pipeline to the
|
||||
// next programmable stage we're using (the pixel shader for now)
|
||||
return output;
|
||||
}
|
Reference in a new issue