-
Notifications
You must be signed in to change notification settings - Fork 433
Sprites and textures
In this lesson, we will cover the basics of creating a texture from a bitmap file, and then rendering it using a 2D sprite with various drawing options.
If you are planning to write a 2D game, you should be using DirectX 11 rather than DirectX 12. SpriteBatch for DirectX 12 is intended to support HUDs, menus, and font rendering over complex 3D scenes. It certainly can be used for 'sprites' but the additional complexities required to develop, debug, and manage DirectX 12 provides little value in exchange for a 2D only application. You might also consider using Win2D.
First create a new project using the instructions from the first two lessons: The basic game loop and Adding the DirectX Tool Kit which we will use for this lesson.
Start by saving cat.png into your new project's directory, and then from the top menu select Project / Add Existing Item.... Select "cat.png" and click "OK".
In the Game.h file, add the following variable to the bottom of the Game class's private declarations (right after the m_graphicsMemory
variable you already added as part of setup):
std::unique_ptr<DirectX::DescriptorHeap> m_resourceDescriptors;
Microsoft::WRL::ComPtr<ID3D12Resource> m_texture;
enum Descriptors
{
Cat,
Count
};
In Game.cpp, add to the TODO of CreateDevice after where you have created m_graphicsMemory
:
m_resourceDescriptors = std::make_unique<DescriptorHeap>(m_d3dDevice.Get(),
D3D12_DESCRIPTOR_HEAP_TYPE_CBV_SRV_UAV,
D3D12_DESCRIPTOR_HEAP_FLAG_SHADER_VISIBLE,
Descriptors::Count);
ResourceUploadBatch resourceUpload(m_d3dDevice.Get());
resourceUpload.Begin();
DX::ThrowIfFailed(
CreateWICTextureFromFile(m_d3dDevice.Get(), resourceUpload, L"cat.png",
m_texture.ReleaseAndGetAddressOf()));
CreateShaderResourceView(m_d3dDevice.Get(), m_texture.Get(),
m_resourceDescriptors->GetCpuHandle(Descriptors::Cat));
auto uploadResourcesFinished = resourceUpload.End(m_commandQueue.Get());
uploadResourcesFinished.wait();
In Game.cpp, add to the TODO of OnDeviceLost:
m_texture.Reset();
m_resourceDescriptors.reset();
Build and run the application which will still not be displaying anything but the cornflower blue window, but will have a texture loaded.
Troubleshooting: If you get a runtime exception, then you may have the "cat.png" in the wrong folder, have modified the "Working Directory" in the "Debugging" configuration settings, or otherwise changed the expected paths at runtime of the application. You should set a break-point on
CreateWICTextureFromFile
and step into the code to find the exact problem.
In the Game.h file, add the following variables to the bottom of the Game class's private declarations:
std::unique_ptr<DirectX::SpriteBatch> m_spriteBatch;
DirectX::SimpleMath::Vector2 m_screenPos;
DirectX::SimpleMath::Vector2 m_origin;
In Game.cpp, modify TODO of CreateDevice to include after resourceUpload.Begin
and before the resourceUpload.End
:
RenderTargetState rtState(DXGI_FORMAT_B8G8R8A8_UNORM, DXGI_FORMAT_D32_FLOAT);
SpriteBatchPipelineStateDescription pd(rtState);
m_spriteBatch = std::make_unique<SpriteBatch>(m_d3dDevice.Get(), resourceUpload, pd);
XMUINT2 catSize = GetTextureSize(m_texture.Get());
m_origin.x = float(catSize.x / 2);
m_origin.y = float(catSize.y / 2);
In Game.cpp, add to the TODO of CreateResources:
D3D12_VIEWPORT viewport = { 0.0f, 0.0f,
static_cast<float>(backBufferWidth), static_cast<float>(backBufferHeight),
D3D12_MIN_DEPTH, D3D12_MAX_DEPTH };
m_spriteBatch->SetViewport(viewport);
m_screenPos.x = backBufferWidth / 2.f;
m_screenPos.y = backBufferHeight / 2.f;
If using the UWP template, you should also add
m_spriteBatch->SetRotation(m_outputRotation);
to handle display orientation changes.
In Game.cpp, add to the TODO of OnDeviceLost:
m_spriteBatch.reset();
In Game.cpp, add to the TODO of Render:
ID3D12DescriptorHeap* heaps[] = { m_resourceDescriptors->Heap() };
m_commandList->SetDescriptorHeaps(_countof(heaps), heaps);
m_spriteBatch->Begin(m_commandList.Get());
m_spriteBatch->Draw(m_resourceDescriptors->GetGpuHandle(Descriptors::Cat),
GetTextureSize(m_texture.Get()),
m_screenPos, nullptr, Colors::White, 0.f, m_origin);
m_spriteBatch->End();
Build and run, and you should get the following screen:
You'll note that setting the descriptor heap is left to the caller. This allows the application to use their own heaps instead of the
DescriptorHeap
class. You can freely mix and match heaps in the application, but remember that you can have only a single texture descriptor heap and a single sampler descriptor heap active at any given time. To change active heaps, you shouldEnd
the current batch, callSetDescriptorHeaps
, and then callBegin
to start a new one.
One thing you should notice is that the edges of the cat look strange with a bit of white outline. The problem here is that the cat.png
file's alpha channel is straight alpha (i.e. the pixels are of the form (R,G,B,A)
). The default behavior of SpriteBatch, however, is to assume you are using premultiplied alpha (i.e. the pixels are of the form (R*A, G*A, B*A, A)
). There are many reasons why using premultiplied alpha is superior, but for now we can fix this mismatch by changing our use of SpriteBatch to use straight alpha blending instead by supplying our own ID3D11BlendState
object. We'll make use of the CommonStates class to provide one of the built-in blend state objects.
In Game.cpp, modify the creation of SpriteBatchPipelineStateDescription
in CreateDevice:
SpriteBatchPipelineStateDescription pd(rtState,
&CommonStates::NonPremultiplied);
Build and run again, and you'll get a nice clean cat:
Rather than use a PNG
and the Windows Imaging Component (WIC) to load the texture, a more efficient thing for us to do is to make use of a DDS
file instead. A DDS
file is a container for all kinds of Direct3D resources including 1D and 2D textures, cubemaps, volume maps, arrays of 1D or 2D textures or cubemaps each optionally with mipmaps. It can contain a wide-array of pixel formats and hardware-supported 'block-compression' schemes to save on video memory usage at runtime.
Visual Studio has a built-in system for converting images to DDS as part of the build process, which you can read about here.
For this tutorial, we will instead make of use of the DirectXTex texconv command-line tool.
- Download the Texconv.exe from the DirectXTex site save the EXE into your project's folder.
- Open a command-prompt and then change to your project's folder.
Then run the following command-line:
texconv cat.png -pmalpha -m 1 -f BC3_UNORM
Then from the top menu in Visual Studio select Project / Add Existing Item.... Select cat.dds and click "OK".
Now will return to Game.cpp in the CreateDevice and change our use of CreateWICTextureFromFile
to CreateDDSTextureFromFile
:
DX::ThrowIfFailed(
CreateDDSTextureFromFile(m_d3dDevice.Get(), resourceUpload, L"cat.dds",
m_texture.ReleaseAndGetAddressOf()));
Note that since we used the option -pmalpha
, we should also make sure we change back using premultipled alpha because our "cat.dds" has premultiplied alpha in it.
In Game.cpp, modify SpriteBatchPipelineStateDescription
in CreateDevice:
SpriteBatchPipelineStateDescription pd(rtState);
Build and run we are rendering our 'clean' cat with premultiplied alpha:
- The switch
-pmalpha
causes the texconv command-line tool to convert the image to premultiplied alpha before saving the.dds
file. This assumes the source image is in straight-alpha. - The switch
-m 1
disables the generation of mipmaps for the image. By default, the tool generates a full set of mipmaps when converting to a.dds
, but since our source image is not a power of two in width & height, it also generates a warning message about use with feature level 9.x devices. For standard sprites, we typically do not make use of mipmaps. - The switch
-f BC3_UNORM
selects theDXGI_FORMAT_BC3_UNORM
format for the resulting.dds
file. In combination with the-pmalpha
switch, this results in the "DXT4" block-compression format being used.
Now that we have our cat rendering, we can start to animate it. Here's a simple rotation where we are using the cosf function to give us a time-varying value from -1 to 1.
In Game.cpp, modify the TODO of Render:
float time = float(m_timer.GetTotalSeconds());
ID3D12DescriptorHeap* heaps[] = { m_resourceDescriptors->Heap() };
m_commandList->SetDescriptorHeaps(_countof(heaps), heaps);
m_spriteBatch->Begin(m_commandList.Get());
m_spriteBatch->Draw(m_resourceDescriptors->GetGpuHandle(Descriptors::Cat),
GetTextureSize(m_texture.Get()),
m_screenPos, nullptr, Colors::White, cosf(time) * 4.f, m_origin);
m_spriteBatch->End();
Build and run to see the cat spinning.
We can scale a sprite's size as well. Again, we are using cosf to give us a time-varying value between -1 and 1.
In Game.cpp, modify the TODO of Render:
float time = float(m_timer.GetTotalSeconds());
ID3D12DescriptorHeap* heaps[] = { m_resourceDescriptors->Heap() };
m_commandList->SetDescriptorHeaps(_countof(heaps), heaps);
m_spriteBatch->Begin(m_commandList.Get());
m_spriteBatch->Draw(m_resourceDescriptors->GetGpuHandle(Descriptors::Cat),
GetTextureSize(m_texture.Get()),
m_screenPos, nullptr, Colors::White, 0.f, m_origin, cosf(time) + 2.f);
m_spriteBatch->End();
Build and run to see the cat growing and shrinking.
We can modify the color of the sprite with a tint as well:
In Game.cpp, modify the TODO of Render:
ID3D12DescriptorHeap* heaps[] = { m_resourceDescriptors->Heap() };
m_commandList->SetDescriptorHeaps(_countof(heaps), heaps);
m_spriteBatch->Begin(m_commandList.Get());
m_spriteBatch->Draw(m_resourceDescriptors->GetGpuHandle(Descriptors::Cat),
GetTextureSize(m_texture.Get()),
m_screenPos, nullptr, Colors::Green, 0.f, m_origin);
m_spriteBatch->End();
Build and run to see a green-tinged cat.
With the optional source-rectangle parameter, we can tile a sprite.
In the Game.h file, add the following variables to the bottom of the Game class's private declarations:
RECT m_tileRect;
std::unique_ptr<DirectX::CommonStates> m_states;
In the Game.cpp file, add to TODO section of CreateDevice:
m_states = std::make_unique<CommonStates>(m_d3dDevice.Get());
modify the creation of SpriteBatchPipelineStateDescription
:
auto sampler = m_states->LinearWrap();
SpriteBatchPipelineStateDescription pd(
rtState, nullptr, nullptr, nullptr, &sampler);
and then change
m_origin.x = float(catSize.x / 2);
m_origin.y = float(catSize.y / 2);
to
m_origin.x = float(catSize.x * 2);
m_origin.y = float(catSize.y * 2);
m_tileRect.left = catSize.x * 2;
m_tileRect.right = catSize.x * 6;
m_tileRect.top = catSize.y * 2;
m_tileRect.bottom = catSize.y * 6;
In Game.cpp, add to the TODO of OnDeviceLost:
m_states.reset();
In the Game.cpp file, modify in the TODO section of Render:
ID3D12DescriptorHeap* heaps[] = { m_resourceDescriptors->Heap(), m_states->Heap() };
m_commandList->SetDescriptorHeaps(_countof(heaps), heaps);
m_spriteBatch->Begin(m_commandList.Get());
m_spriteBatch->Draw(m_resourceDescriptors->GetGpuHandle(Descriptors::Cat),
GetTextureSize(m_texture.Get()),
m_screenPos, &m_tileRect, Colors::White, 0.f, m_origin);
m_spriteBatch->End();
Build and run to see the sprite as an array of 4x4 cats.
By default SpriteBatch
uses a static sampler that is set to LinearClamp
. In order to do the tiling, we had to override that setting by providing a heap sampler. The CommonStates object provides a heap and pre-defined sampler descriptors that can be used for this purpose.
Our last exercise for this lesson is rendering a sprite as a full background image. Start by saving sunset.jpg to your project directory, and then from the top menu select Project / Add Existing Item.... Select "sunset.jpg" and click "OK".
In the Game.h file, add the following variables to the bottom of the Game class's private declarations:
RECT m_fullscreenRect;
Microsoft::WRL::ComPtr<ID3D12Resource> m_background;
and change
enum Descriptors
{
Cat,
Count
};
to
enum Descriptors
{
Cat,
Background,
Count
};
In Game.cpp, add to the TODO of CreateDevice to include after resourceUpload.Begin
and before the resourceUpload.End
:
DX::ThrowIfFailed(
CreateWICTextureFromFile(m_d3dDevice.Get(), resourceUpload, L"sunset.jpg",
m_background.ReleaseAndGetAddressOf()));
CreateShaderResourceView(m_d3dDevice.Get(), m_background.Get(),
m_resourceDescriptors->GetCpuHandle(Descriptors::Background));
In Game.cpp, add to the TODO of CreateResources:
m_fullscreenRect = { 0, 0, backBufferWidth, backBufferHeight };
and then modify the m_origin
initialization back to:
m_origin.x = float(catSize.x / 2);
m_origin.y = float(catSize.y / 2);
In Game.cpp, add to the TODO of OnDeviceLost:
m_background.Reset();
```cpp
In **Game.cpp**, modify the TODO section of **Render** to be:
```cpp
ID3D12DescriptorHeap* heaps[] = { m_resourceDescriptors->Heap() };
m_commandList->SetDescriptorHeaps(_countof(heaps), heaps);
m_spriteBatch->Begin(m_commandList.Get());
m_spriteBatch->Draw(m_resourceDescriptors->GetGpuHandle(Descriptors::Background),
GetTextureSize(m_background.Get()),
m_fullscreenRect);
m_spriteBatch->Draw(m_resourceDescriptors->GetGpuHandle(Descriptors::Cat),
GetTextureSize(m_texture.Get()),
m_screenPos, nullptr, Colors::White, 0.f, m_origin);
m_spriteBatch->End();
Build and run to see our cat drawing over a sunset background.
Next lesson: Drawing text
DirectX Tool Kit docs CommonStates, DescriptorHeap, DDSTextureLoader, RenderTargetState, ResourceUploadBatch, SpriteBatch, WICTextureLoader
Direc3D 11 Textures and Block Compression
ShawnHar on Premultiplied Alpha
Tom Forsyth on Premultiplied Alpha
DirectX Tool Kit for DX11 tutorial: More tricks with sprites
All content and source code for this package are subject to the terms of the MIT License.
This project has adopted the Microsoft Open Source Code of Conduct. For more information see the Code of Conduct FAQ or contact opencode@microsoft.com with any additional questions or comments.
- Universal Windows Platform apps
- Windows desktop apps
- Windows 11
- Windows 10
- Xbox One
- Xbox Series X|S
- x86
- x64
- ARM64
- Visual Studio 2022
- Visual Studio 2019 (16.11)
- clang/LLVM v12 - v18
- MinGW 12.2, 13.2
- CMake 3.20