2010-06-13 19:50:06 +00:00
|
|
|
// Copyright (C) 2003 Dolphin Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, version 2.0.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
|
|
|
|
#include "Common.h"
|
|
|
|
|
|
|
|
#include "D3DBase.h"
|
|
|
|
#include "D3DUtil.h"
|
|
|
|
#include "D3DTexture.h"
|
|
|
|
#include "Render.h"
|
|
|
|
#include "PixelShaderCache.h"
|
|
|
|
#include "VertexShaderCache.h"
|
|
|
|
#include "D3DShader.h"
|
|
|
|
|
|
|
|
namespace D3D
|
|
|
|
{
|
|
|
|
|
|
|
|
CD3DFont font;
|
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
#define MAX_NUM_VERTICES 50*6
|
2010-06-13 19:50:06 +00:00
|
|
|
struct FONT2DVERTEX {
|
|
|
|
float x,y,z;
|
|
|
|
float col[4];
|
|
|
|
float tu, tv;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline FONT2DVERTEX InitFont2DVertex(float x, float y, u32 color, float tu, float tv)
|
|
|
|
{
|
|
|
|
FONT2DVERTEX v; v.x=x; v.y=y; v.z=0; v.tu = tu; v.tv = tv;
|
|
|
|
v.col[0] = ((float)((color >> 16) & 0xFF)) / 255.f;
|
|
|
|
v.col[1] = ((float)((color >> 8) & 0xFF)) / 255.f;
|
|
|
|
v.col[2] = ((float)((color >> 0) & 0xFF)) / 255.f;
|
|
|
|
v.col[3] = ((float)((color >> 24) & 0xFF)) / 255.f;
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
CD3DFont::CD3DFont() : m_dwTexWidth(512), m_dwTexHeight(512)
|
|
|
|
{
|
|
|
|
m_pTexture = NULL;
|
|
|
|
m_pVB = NULL;
|
|
|
|
m_InputLayout = NULL;
|
|
|
|
m_pshader = NULL;
|
|
|
|
m_vshader = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char fontpixshader[] = {
|
|
|
|
"Texture2D tex2D;\n"
|
|
|
|
"SamplerState linearSampler\n"
|
|
|
|
"{\n"
|
|
|
|
" Filter = MIN_MAG_MIP_LINEAR;\n"
|
2010-06-20 23:10:55 +00:00
|
|
|
" AddressU = D3D11_TEXTURE_ADDRESS_BORDER;\n"
|
|
|
|
" AddressV = D3D11_TEXTURE_ADDRESS_BORDER;\n"
|
|
|
|
" BorderColor = float4(0.f, 0.f, 0.f, 0.f);\n"
|
2010-06-13 19:50:06 +00:00
|
|
|
"};\n"
|
|
|
|
"struct PS_INPUT\n"
|
|
|
|
"{\n"
|
|
|
|
" float4 pos : SV_POSITION;\n"
|
|
|
|
" float4 col : COLOR;\n"
|
|
|
|
" float2 tex : TEXCOORD;\n"
|
|
|
|
"};\n"
|
|
|
|
"float4 main( PS_INPUT input ) : SV_Target\n"
|
|
|
|
"{\n"
|
|
|
|
" return tex2D.Sample( linearSampler, input.tex ) * input.col;\n"
|
|
|
|
"};\n"
|
|
|
|
};
|
|
|
|
|
|
|
|
const char fontvertshader[] = {
|
|
|
|
"struct VS_INPUT\n"
|
|
|
|
"{\n"
|
|
|
|
" float4 pos : POSITION;\n"
|
|
|
|
" float4 col : COLOR;\n"
|
|
|
|
" float2 tex : TEXCOORD;\n"
|
|
|
|
"};\n"
|
|
|
|
"struct PS_INPUT\n"
|
|
|
|
"{\n"
|
|
|
|
" float4 pos : SV_POSITION;\n"
|
|
|
|
" float4 col : COLOR;\n"
|
|
|
|
" float2 tex : TEXCOORD;\n"
|
|
|
|
"};\n"
|
|
|
|
"PS_INPUT main( VS_INPUT input )\n"
|
|
|
|
"{\n"
|
|
|
|
" PS_INPUT output;\n"
|
|
|
|
" output.pos = input.pos;\n"
|
|
|
|
" output.col = input.col;\n"
|
|
|
|
" output.tex = input.tex;\n"
|
|
|
|
" return output;\n"
|
|
|
|
"};\n"
|
|
|
|
};
|
|
|
|
|
|
|
|
int CD3DFont::Init()
|
|
|
|
{
|
2010-09-28 02:15:02 +00:00
|
|
|
// Create vertex buffer for the letters
|
2010-06-13 19:50:06 +00:00
|
|
|
HRESULT hr;
|
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Prepare to create a bitmap
|
2010-06-13 19:50:06 +00:00
|
|
|
unsigned int* pBitmapBits;
|
|
|
|
BITMAPINFO bmi;
|
|
|
|
ZeroMemory(&bmi.bmiHeader, sizeof(BITMAPINFOHEADER));
|
|
|
|
bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
|
|
bmi.bmiHeader.biWidth = (int)m_dwTexWidth;
|
|
|
|
bmi.bmiHeader.biHeight = -(int)m_dwTexHeight;
|
|
|
|
bmi.bmiHeader.biPlanes = 1;
|
|
|
|
bmi.bmiHeader.biCompression = BI_RGB;
|
|
|
|
bmi.bmiHeader.biBitCount = 32;
|
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Create a DC and a bitmap for the font
|
2010-06-13 19:50:06 +00:00
|
|
|
HDC hDC = CreateCompatibleDC(NULL);
|
|
|
|
HBITMAP hbmBitmap = CreateDIBSection(hDC, &bmi, DIB_RGB_COLORS, (void**)&pBitmapBits, NULL, 0);
|
|
|
|
SetMapMode(hDC, MM_TEXT);
|
|
|
|
|
|
|
|
// create a GDI font
|
2010-06-20 23:10:55 +00:00
|
|
|
HFONT hFont = CreateFont(24, 0, 0, 0, FW_NORMAL, FALSE,
|
2010-06-13 19:50:06 +00:00
|
|
|
FALSE, FALSE, DEFAULT_CHARSET, OUT_DEFAULT_PRECIS,
|
2010-06-20 23:10:55 +00:00
|
|
|
CLIP_DEFAULT_PRECIS, PROOF_QUALITY,
|
2010-06-13 19:50:06 +00:00
|
|
|
VARIABLE_PITCH, _T("Tahoma"));
|
|
|
|
if (NULL == hFont) return E_FAIL;
|
|
|
|
|
|
|
|
HGDIOBJ hOldbmBitmap = SelectObject(hDC, hbmBitmap);
|
|
|
|
HGDIOBJ hOldFont = SelectObject(hDC, hFont);
|
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Set text properties
|
2010-06-13 19:50:06 +00:00
|
|
|
SetTextColor(hDC, 0xFFFFFF);
|
|
|
|
SetBkColor (hDC, 0);
|
|
|
|
SetTextAlign(hDC, TA_TOP);
|
|
|
|
|
2010-06-20 23:10:55 +00:00
|
|
|
TEXTMETRICW tm;
|
|
|
|
GetTextMetricsW(hDC, &tm);
|
|
|
|
m_LineHeight = tm.tmHeight;
|
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Loop through all printable characters and output them to the bitmap
|
|
|
|
// Meanwhile, keep track of the corresponding tex coords for each character.
|
2010-06-13 19:50:06 +00:00
|
|
|
int x = 0, y = 0;
|
|
|
|
char str[2] = "\0";
|
|
|
|
for (int c = 0; c < 127 - 32; c++)
|
|
|
|
{
|
|
|
|
str[0] = c + 32;
|
|
|
|
SIZE size;
|
|
|
|
GetTextExtentPoint32A(hDC, str, 1, &size);
|
|
|
|
if ((int)(x+size.cx+1) > m_dwTexWidth)
|
|
|
|
{
|
|
|
|
x = 0;
|
2010-06-20 23:10:55 +00:00
|
|
|
y += m_LineHeight;
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ExtTextOutA(hDC, x+1, y+0, ETO_OPAQUE | ETO_CLIPPED, NULL, str, 1, NULL);
|
2010-09-28 02:15:02 +00:00
|
|
|
m_fTexCoords[c][0] = ((float)(x+0))/m_dwTexWidth;
|
|
|
|
m_fTexCoords[c][1] = ((float)(y+0))/m_dwTexHeight;
|
|
|
|
m_fTexCoords[c][2] = ((float)(x+0+size.cx))/m_dwTexWidth;
|
|
|
|
m_fTexCoords[c][3] = ((float)(y+0+size.cy))/m_dwTexHeight;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
x += size.cx + 3; // 3 to work around annoying ij conflict (part of the j ends up with the i)
|
|
|
|
}
|
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Create a new texture for the font
|
2010-06-13 19:50:06 +00:00
|
|
|
// possible optimization: store the converted data in a buffer and fill the texture on creation.
|
|
|
|
// That way, we can use a static texture
|
|
|
|
ID3D11Texture2D* buftex;
|
2010-06-19 01:02:43 +00:00
|
|
|
D3D11_TEXTURE2D_DESC texdesc = CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_R8G8B8A8_UNORM, m_dwTexWidth, m_dwTexHeight,
|
2010-06-13 19:50:06 +00:00
|
|
|
1, 1, D3D11_BIND_SHADER_RESOURCE, D3D11_USAGE_DYNAMIC,
|
|
|
|
D3D11_CPU_ACCESS_WRITE);
|
|
|
|
hr = device->CreateTexture2D(&texdesc, NULL, &buftex);
|
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
PanicAlert("Failed to create font texture");
|
|
|
|
return hr;
|
|
|
|
}
|
2010-06-17 10:42:57 +00:00
|
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)buftex, "texture of a CD3DFont object");
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Lock the surface and write the alpha values for the set pixels
|
2010-06-13 19:50:06 +00:00
|
|
|
D3D11_MAPPED_SUBRESOURCE texmap;
|
|
|
|
hr = context->Map(buftex, 0, D3D11_MAP_WRITE_DISCARD, 0, &texmap);
|
|
|
|
if (FAILED(hr)) PanicAlert("Failed to map a texture at %s %d\n", __FILE__, __LINE__);
|
|
|
|
|
|
|
|
for (y = 0; y < m_dwTexHeight; y++)
|
|
|
|
{
|
|
|
|
u32* pDst32 = (u32*)((u8*)texmap.pData + y * texmap.RowPitch);
|
|
|
|
for (x = 0; x < m_dwTexWidth; x++)
|
|
|
|
{
|
|
|
|
const u8 bAlpha = (pBitmapBits[m_dwTexWidth * y + x] & 0xff);
|
2010-06-20 23:10:55 +00:00
|
|
|
*pDst32++ = (((bAlpha << 4) | bAlpha) << 24) | 0xFFFFFF;
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Done updating texture, so clean up used objects
|
2010-06-13 19:50:06 +00:00
|
|
|
context->Unmap(buftex, 0);
|
|
|
|
hr = D3D::device->CreateShaderResourceView(buftex, NULL, &m_pTexture);
|
|
|
|
if (FAILED(hr)) PanicAlert("Failed to create shader resource view at %s %d\n", __FILE__, __LINE__);
|
2010-07-17 15:18:52 +00:00
|
|
|
SAFE_RELEASE(buftex);
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
SelectObject(hDC, hOldbmBitmap);
|
|
|
|
DeleteObject(hbmBitmap);
|
|
|
|
|
|
|
|
SelectObject(hDC, hOldFont);
|
|
|
|
DeleteObject(hFont);
|
|
|
|
|
|
|
|
// setup device objects for drawing
|
|
|
|
m_pshader = D3D::CompileAndCreatePixelShader(fontpixshader, sizeof(fontpixshader));
|
|
|
|
if (m_pshader == NULL) PanicAlert("Failed to create pixel shader, %s %d\n", __FILE__, __LINE__);
|
2010-06-17 10:42:57 +00:00
|
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_pshader, "pixel shader of a CD3DFont object");
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2010-06-16 23:25:19 +00:00
|
|
|
D3DBlob* vsbytecode;
|
2010-06-13 19:50:06 +00:00
|
|
|
D3D::CompileVertexShader(fontvertshader, sizeof(fontvertshader), &vsbytecode);
|
|
|
|
if (vsbytecode == NULL) PanicAlert("Failed to compile vertex shader, %s %d\n", __FILE__, __LINE__);
|
|
|
|
m_vshader = D3D::CreateVertexShaderFromByteCode(vsbytecode);
|
|
|
|
if (m_vshader == NULL) PanicAlert("Failed to create vertex shader, %s %d\n", __FILE__, __LINE__);
|
2010-06-17 10:42:57 +00:00
|
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_vshader, "vertex shader of a CD3DFont object");
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
const D3D11_INPUT_ELEMENT_DESC desc[] =
|
|
|
|
{
|
|
|
|
{ "POSITION", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 0, D3D11_INPUT_PER_VERTEX_DATA, 0 },
|
|
|
|
{ "COLOR", 0, DXGI_FORMAT_R32G32B32A32_FLOAT, 0, 12, D3D11_INPUT_PER_VERTEX_DATA, 0 },
|
|
|
|
{ "TEXCOORD", 0, DXGI_FORMAT_R32G32_FLOAT, 0, 28, D3D11_INPUT_PER_VERTEX_DATA, 0 },
|
|
|
|
};
|
2010-06-16 23:25:19 +00:00
|
|
|
hr = D3D::device->CreateInputLayout(desc, 3, vsbytecode->Data(), vsbytecode->Size(), &m_InputLayout);
|
2010-06-13 19:50:06 +00:00
|
|
|
if (FAILED(hr)) PanicAlert("Failed to create input layout, %s %d\n", __FILE__, __LINE__);
|
2010-07-17 15:18:52 +00:00
|
|
|
SAFE_RELEASE(vsbytecode);
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
D3D11_BLEND_DESC blenddesc;
|
|
|
|
blenddesc.AlphaToCoverageEnable = FALSE;
|
|
|
|
blenddesc.IndependentBlendEnable = FALSE;
|
|
|
|
blenddesc.RenderTarget[0].BlendEnable = TRUE;
|
|
|
|
blenddesc.RenderTarget[0].RenderTargetWriteMask = D3D11_COLOR_WRITE_ENABLE_ALL;
|
|
|
|
blenddesc.RenderTarget[0].SrcBlend = D3D11_BLEND_SRC_ALPHA;
|
|
|
|
blenddesc.RenderTarget[0].DestBlend = D3D11_BLEND_INV_SRC_ALPHA;
|
|
|
|
blenddesc.RenderTarget[0].BlendOp = D3D11_BLEND_OP_ADD;
|
|
|
|
blenddesc.RenderTarget[0].SrcBlendAlpha = D3D11_BLEND_SRC_ALPHA;
|
|
|
|
blenddesc.RenderTarget[0].DestBlendAlpha = D3D11_BLEND_INV_SRC_ALPHA;
|
|
|
|
blenddesc.RenderTarget[0].BlendOpAlpha = D3D11_BLEND_OP_ADD;
|
2010-06-18 18:40:58 +00:00
|
|
|
hr = D3D::device->CreateBlendState(&blenddesc, &m_blendstate);
|
|
|
|
CHECK(hr==S_OK, "Create font blend state");
|
2010-06-17 10:42:57 +00:00
|
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_blendstate, "blend state of a CD3DFont object");
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
// this might need to be changed when adding multisampling support
|
|
|
|
D3D11_RASTERIZER_DESC rastdesc = CD3D11_RASTERIZER_DESC(D3D11_FILL_SOLID, D3D11_CULL_NONE, false, 0, 0.f, 0.f, false, false, false, false);
|
2010-06-18 18:40:58 +00:00
|
|
|
hr = D3D::device->CreateRasterizerState(&rastdesc, &m_raststate);
|
|
|
|
CHECK(hr==S_OK, "Create font rasterizer state");
|
2010-06-17 10:42:57 +00:00
|
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_raststate, "rasterizer state of a CD3DFont object");
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
D3D11_BUFFER_DESC vbdesc = CD3D11_BUFFER_DESC(MAX_NUM_VERTICES*sizeof(FONT2DVERTEX), D3D11_BIND_VERTEX_BUFFER, D3D11_USAGE_DYNAMIC, D3D11_CPU_ACCESS_WRITE);
|
|
|
|
if (FAILED(hr = device->CreateBuffer(&vbdesc, NULL, &m_pVB)))
|
|
|
|
{
|
2010-06-18 18:40:58 +00:00
|
|
|
PanicAlert("Failed to create font vertex buffer at %s, line %d\n", __FILE__, __LINE__);
|
2010-06-13 19:50:06 +00:00
|
|
|
return hr;
|
|
|
|
}
|
2010-06-17 10:42:57 +00:00
|
|
|
D3D::SetDebugObjectName((ID3D11DeviceChild*)m_pVB, "vertex buffer of a CD3DFont object");
|
2010-06-13 19:50:06 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CD3DFont::Shutdown()
|
|
|
|
{
|
|
|
|
SAFE_RELEASE(m_pVB);
|
|
|
|
SAFE_RELEASE(m_pTexture);
|
|
|
|
SAFE_RELEASE(m_InputLayout);
|
|
|
|
SAFE_RELEASE(m_pshader);
|
|
|
|
SAFE_RELEASE(m_vshader);
|
|
|
|
|
|
|
|
SAFE_RELEASE(m_blendstate);
|
|
|
|
SAFE_RELEASE(m_raststate);
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2010-06-20 23:10:55 +00:00
|
|
|
int CD3DFont::DrawTextScaled(float x, float y, float size, float spacing, u32 dwColor, const char* strText, bool center)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
2010-09-28 02:15:02 +00:00
|
|
|
if (!m_pVB)
|
|
|
|
return 0;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
UINT stride = sizeof(FONT2DVERTEX);
|
|
|
|
UINT bufoffset = 0;
|
|
|
|
|
2010-06-20 23:10:55 +00:00
|
|
|
float scalex = 1 / (float)D3D::GetBackBufferWidth() * 2.f;
|
|
|
|
float scaley = 1 / (float)D3D::GetBackBufferHeight() * 2.f;
|
|
|
|
float sizeratio = size / (float)m_LineHeight;
|
|
|
|
|
2010-06-16 13:36:40 +00:00
|
|
|
// translate starting positions
|
2010-07-06 13:14:51 +00:00
|
|
|
float sx = x * scalex - 1.f;
|
2010-06-20 23:10:55 +00:00
|
|
|
float sy = 1.f - y * scaley;
|
2010-06-16 13:36:40 +00:00
|
|
|
char c;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2010-09-28 02:15:02 +00:00
|
|
|
// Fill vertex buffer
|
2010-06-13 19:50:06 +00:00
|
|
|
FONT2DVERTEX* pVertices;
|
|
|
|
int dwNumTriangles = 0L;
|
|
|
|
|
|
|
|
D3D11_MAPPED_SUBRESOURCE vbmap;
|
|
|
|
HRESULT hr = context->Map(m_pVB, 0, D3D11_MAP_WRITE_DISCARD, 0, &vbmap);
|
|
|
|
if (FAILED(hr)) PanicAlert("Mapping vertex buffer failed, %s %d\n", __FILE__, __LINE__);
|
|
|
|
pVertices = (D3D::FONT2DVERTEX*)vbmap.pData;
|
|
|
|
|
2010-06-16 13:36:40 +00:00
|
|
|
// if center was requested, set current position as centre
|
|
|
|
// this is currently never used
|
2010-06-13 19:50:06 +00:00
|
|
|
if (center)
|
|
|
|
{
|
2010-06-16 13:36:40 +00:00
|
|
|
const char *oldText = strText;
|
|
|
|
float mx=0;
|
|
|
|
float maxx=0;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2010-06-16 13:36:40 +00:00
|
|
|
while (c = *strText++)
|
|
|
|
{
|
2010-06-20 23:10:55 +00:00
|
|
|
if (c == ('\n')) mx = 0;
|
|
|
|
if (c < (' ') ) continue;
|
2010-06-16 13:36:40 +00:00
|
|
|
c -= 32;
|
2010-06-20 23:10:55 +00:00
|
|
|
mx += (m_fTexCoords[c][2]-m_fTexCoords[c][0])/(m_fTexCoords[0][3] - m_fTexCoords[0][1]) + spacing;
|
2010-06-16 13:36:40 +00:00
|
|
|
if (mx > maxx) maxx = mx;
|
|
|
|
}
|
2010-06-20 23:10:55 +00:00
|
|
|
sx -= scalex*maxx*size;
|
2010-06-16 13:36:40 +00:00
|
|
|
strText = oldText;
|
|
|
|
}
|
2010-06-13 19:50:06 +00:00
|
|
|
// set general pipeline state
|
2010-06-18 23:33:07 +00:00
|
|
|
D3D::stateman->PushBlendState(m_blendstate);
|
|
|
|
D3D::stateman->PushRasterizerState(m_raststate);
|
|
|
|
D3D::stateman->Apply();
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
D3D::context->PSSetShader(m_pshader, NULL, 0);
|
|
|
|
D3D::context->VSSetShader(m_vshader, NULL, 0);
|
|
|
|
|
|
|
|
D3D::context->IASetInputLayout(m_InputLayout);
|
|
|
|
D3D::context->IASetPrimitiveTopology(D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST);
|
|
|
|
D3D::context->PSSetShaderResources(0, 1, &m_pTexture);
|
2010-06-20 23:10:55 +00:00
|
|
|
|
|
|
|
float fStartX = sx;
|
2010-06-16 13:36:40 +00:00
|
|
|
while (c = *strText++)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
|
|
|
if (c == ('\n'))
|
|
|
|
{
|
|
|
|
sx = fStartX;
|
2010-06-20 23:10:55 +00:00
|
|
|
sy -= scaley * size;
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
2010-09-28 02:15:02 +00:00
|
|
|
if (c < (' '))
|
|
|
|
continue;
|
2010-06-20 23:10:55 +00:00
|
|
|
|
2010-06-16 13:36:40 +00:00
|
|
|
c -= 32;
|
2010-06-13 19:50:06 +00:00
|
|
|
float tx1 = m_fTexCoords[c][0];
|
|
|
|
float ty1 = m_fTexCoords[c][1];
|
|
|
|
float tx2 = m_fTexCoords[c][2];
|
|
|
|
float ty2 = m_fTexCoords[c][3];
|
|
|
|
|
2010-06-20 23:10:55 +00:00
|
|
|
float w = (float)(tx2-tx1) * m_dwTexWidth * scalex * sizeratio;
|
|
|
|
float h = (float)(ty1-ty2) * m_dwTexHeight * scaley * sizeratio;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
FONT2DVERTEX v[6];
|
2010-09-28 02:15:02 +00:00
|
|
|
v[0] = InitFont2DVertex(sx, sy+h, dwColor, tx1, ty2);
|
|
|
|
v[1] = InitFont2DVertex(sx, sy, dwColor, tx1, ty1);
|
|
|
|
v[2] = InitFont2DVertex(sx+w, sy+h, dwColor, tx2, ty2);
|
|
|
|
v[3] = InitFont2DVertex(sx+w, sy, dwColor, tx2, ty1);
|
2010-06-13 19:50:06 +00:00
|
|
|
v[4] = v[2];
|
|
|
|
v[5] = v[1];
|
|
|
|
|
2010-07-06 13:14:51 +00:00
|
|
|
memcpy(pVertices, v, 6*sizeof(FONT2DVERTEX));
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
pVertices+=6;
|
|
|
|
dwNumTriangles += 2;
|
|
|
|
|
|
|
|
if (dwNumTriangles * 3 > (MAX_NUM_VERTICES - 6))
|
|
|
|
{
|
|
|
|
context->Unmap(m_pVB, 0);
|
|
|
|
|
|
|
|
D3D::context->IASetVertexBuffers(0, 1, &m_pVB, &stride, &bufoffset);
|
|
|
|
D3D::context->Draw(3 * dwNumTriangles, 0);
|
|
|
|
|
|
|
|
dwNumTriangles = 0;
|
|
|
|
D3D11_MAPPED_SUBRESOURCE vbmap;
|
2010-06-20 23:10:55 +00:00
|
|
|
hr = context->Map(m_pVB, 0, D3D11_MAP_WRITE_DISCARD, 0, &vbmap);
|
2010-06-13 19:50:06 +00:00
|
|
|
if (FAILED(hr)) PanicAlert("Mapping vertex buffer failed, %s %d\n", __FILE__, __LINE__);
|
|
|
|
pVertices = (D3D::FONT2DVERTEX*)vbmap.pData;
|
|
|
|
}
|
2010-06-20 23:10:55 +00:00
|
|
|
sx += w + spacing * scalex * size;
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unlock and render the vertex buffer
|
|
|
|
context->Unmap(m_pVB, 0);
|
|
|
|
if (dwNumTriangles > 0)
|
|
|
|
{
|
|
|
|
D3D::context->IASetVertexBuffers(0, 1, &m_pVB, &stride, &bufoffset);
|
|
|
|
D3D::context->Draw(3 * dwNumTriangles, 0);
|
|
|
|
}
|
2010-06-18 23:33:07 +00:00
|
|
|
D3D::stateman->PopBlendState();
|
|
|
|
D3D::stateman->PopRasterizerState();
|
2010-06-13 19:50:06 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2010-10-04 11:09:32 +00:00
|
|
|
ID3D11Buffer* CreateQuadVertexBuffer(unsigned int size, void* data, D3D11_USAGE usage = D3D11_USAGE_DYNAMIC)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
|
|
|
ID3D11Buffer* vb;
|
|
|
|
D3D11_BUFFER_DESC vbdesc;
|
|
|
|
vbdesc.BindFlags = D3D11_BIND_VERTEX_BUFFER;
|
|
|
|
vbdesc.ByteWidth = size;
|
|
|
|
vbdesc.MiscFlags = 0;
|
2010-10-04 11:09:32 +00:00
|
|
|
vbdesc.Usage = usage;
|
|
|
|
switch (usage)
|
|
|
|
{
|
|
|
|
case D3D11_USAGE_DEFAULT:
|
|
|
|
case D3D11_USAGE_IMMUTABLE:
|
|
|
|
vbdesc.CPUAccessFlags = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case D3D11_USAGE_DYNAMIC:
|
|
|
|
vbdesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case D3D11_USAGE_STAGING:
|
|
|
|
vbdesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE|D3D11_CPU_ACCESS_READ;
|
|
|
|
break;
|
|
|
|
}
|
2010-06-13 19:50:06 +00:00
|
|
|
if (data)
|
|
|
|
{
|
|
|
|
D3D11_SUBRESOURCE_DATA bufdata;
|
|
|
|
bufdata.pSysMem = data;
|
|
|
|
if (FAILED(device->CreateBuffer(&vbdesc, &bufdata, &vb))) return NULL;
|
|
|
|
}
|
|
|
|
else if (FAILED(device->CreateBuffer(&vbdesc, NULL, &vb))) return NULL;
|
|
|
|
|
|
|
|
return vb;
|
|
|
|
}
|
|
|
|
|
2010-07-17 15:18:52 +00:00
|
|
|
ID3D11SamplerState* linear_copy_sampler = NULL;
|
|
|
|
ID3D11SamplerState* point_copy_sampler = NULL;
|
2010-06-13 19:50:06 +00:00
|
|
|
ID3D11Buffer* stqvb = NULL;
|
|
|
|
ID3D11Buffer* stsqvb = NULL;
|
|
|
|
ID3D11Buffer* clearvb = NULL;
|
|
|
|
|
|
|
|
typedef struct { float x,y,z,u,v; } STQVertex;
|
|
|
|
typedef struct { float x,y,z,u,v; } STSQVertex;
|
2010-07-06 13:14:51 +00:00
|
|
|
typedef struct { float x,y,z; u32 col; } ClearVertex;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2010-07-17 16:13:37 +00:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
float u1, v1, u2, v2;
|
|
|
|
} tex_quad_data;
|
|
|
|
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
MathUtil::Rectangle<float> rdest;
|
|
|
|
float u1, v1, u2, v2;
|
|
|
|
} tex_sub_quad_data;
|
|
|
|
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
u32 col;
|
|
|
|
float z;
|
|
|
|
} clear_quad_data;
|
|
|
|
|
2010-06-13 19:50:06 +00:00
|
|
|
void InitUtils()
|
|
|
|
{
|
|
|
|
float border[4] = { 0.f, 0.f, 0.f, 0.f };
|
2010-07-17 15:18:52 +00:00
|
|
|
D3D11_SAMPLER_DESC samDesc = CD3D11_SAMPLER_DESC(D3D11_FILTER_MIN_MAG_MIP_POINT, D3D11_TEXTURE_ADDRESS_BORDER, D3D11_TEXTURE_ADDRESS_BORDER, D3D11_TEXTURE_ADDRESS_BORDER, 0.f, 1, D3D11_COMPARISON_ALWAYS, border, -D3D11_FLOAT32_MAX, D3D11_FLOAT32_MAX);
|
|
|
|
HRESULT hr = D3D::device->CreateSamplerState(&samDesc, &point_copy_sampler);
|
2010-06-13 19:50:06 +00:00
|
|
|
if (FAILED(hr)) PanicAlert("Failed to create sampler state at %s %d\n", __FILE__, __LINE__);
|
2010-07-17 15:18:52 +00:00
|
|
|
else SetDebugObjectName((ID3D11DeviceChild*)point_copy_sampler, "point copy sampler state");
|
|
|
|
|
|
|
|
samDesc = CD3D11_SAMPLER_DESC(D3D11_FILTER_MIN_MAG_MIP_LINEAR, D3D11_TEXTURE_ADDRESS_BORDER, D3D11_TEXTURE_ADDRESS_BORDER, D3D11_TEXTURE_ADDRESS_BORDER, 0.f, 1, D3D11_COMPARISON_ALWAYS, border, -D3D11_FLOAT32_MAX, D3D11_FLOAT32_MAX);
|
|
|
|
hr = D3D::device->CreateSamplerState(&samDesc, &linear_copy_sampler);
|
|
|
|
if (FAILED(hr)) PanicAlert("Failed to create sampler state at %s %d\n", __FILE__, __LINE__);
|
|
|
|
else SetDebugObjectName((ID3D11DeviceChild*)linear_copy_sampler, "linear copy sampler state");
|
2010-06-13 19:50:06 +00:00
|
|
|
|
2010-07-17 16:13:37 +00:00
|
|
|
// cached data used to avoid unnecessarily reloading the vertex buffers
|
|
|
|
memset(&tex_quad_data, 0, sizeof(tex_quad_data));
|
|
|
|
memset(&tex_sub_quad_data, 0, sizeof(tex_sub_quad_data));
|
|
|
|
memset(&clear_quad_data, 0, sizeof(clear_quad_data));
|
|
|
|
|
|
|
|
STQVertex stqcoords[4] = {
|
|
|
|
{-1.0f, 1.0f, 0.0f, 0, 0},
|
|
|
|
{ 1.0f, 1.0f, 0.0f, 0, 0},
|
|
|
|
{-1.0f,-1.0f, 0.0f, 0, 0},
|
|
|
|
{ 1.0f,-1.0f, 0.0f, 0, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
STSQVertex stsqcoords[4];
|
|
|
|
memset(stsqcoords, 0, sizeof(stsqcoords));
|
|
|
|
|
|
|
|
ClearVertex cqcoords[4] = {
|
|
|
|
{-1.0f, 1.0f, 0, 0},
|
|
|
|
{ 1.0f, 1.0f, 0, 0},
|
|
|
|
{-1.0f, -1.0f, 0, 0},
|
|
|
|
{ 1.0f, -1.0f, 0, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
stqvb = CreateQuadVertexBuffer(4*sizeof(STQVertex), stqcoords);
|
2010-06-18 18:40:58 +00:00
|
|
|
CHECK(stqvb!=NULL, "Create vertex buffer of drawShadedTexQuad");
|
2010-06-13 19:50:06 +00:00
|
|
|
SetDebugObjectName((ID3D11DeviceChild*)stqvb, "vertex buffer of drawShadedTexQuad");
|
|
|
|
|
2010-07-17 16:13:37 +00:00
|
|
|
stsqvb = CreateQuadVertexBuffer(4*sizeof(STSQVertex), stsqcoords);
|
2010-06-18 18:40:58 +00:00
|
|
|
CHECK(stsqvb!=NULL, "Create vertex buffer of drawShadedTexSubQuad");
|
2010-06-13 19:50:06 +00:00
|
|
|
SetDebugObjectName((ID3D11DeviceChild*)stsqvb, "vertex buffer of drawShadedTexSubQuad");
|
|
|
|
|
2010-07-17 16:13:37 +00:00
|
|
|
clearvb = CreateQuadVertexBuffer(4*sizeof(ClearVertex), cqcoords);
|
2010-06-18 18:40:58 +00:00
|
|
|
CHECK(clearvb!=NULL, "Create vertex buffer of drawClearQuad");
|
2010-06-13 19:50:06 +00:00
|
|
|
SetDebugObjectName((ID3D11DeviceChild*)clearvb, "vertex buffer of drawClearQuad");
|
2010-07-18 10:11:34 +00:00
|
|
|
|
|
|
|
font.Init();
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ShutdownUtils()
|
|
|
|
{
|
2010-07-18 10:11:34 +00:00
|
|
|
font.Shutdown();
|
2010-07-17 15:18:52 +00:00
|
|
|
SAFE_RELEASE(point_copy_sampler);
|
|
|
|
SAFE_RELEASE(linear_copy_sampler);
|
2010-06-13 19:50:06 +00:00
|
|
|
SAFE_RELEASE(stqvb);
|
|
|
|
SAFE_RELEASE(stsqvb);
|
|
|
|
SAFE_RELEASE(clearvb);
|
|
|
|
}
|
|
|
|
|
2010-07-17 15:18:52 +00:00
|
|
|
void SetPointCopySampler()
|
|
|
|
{
|
|
|
|
D3D::context->PSSetSamplers(0, 1, &point_copy_sampler);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetLinearCopySampler()
|
|
|
|
{
|
|
|
|
D3D::context->PSSetSamplers(0, 1, &linear_copy_sampler);
|
|
|
|
}
|
|
|
|
|
2010-06-13 19:50:06 +00:00
|
|
|
void drawShadedTexQuad(ID3D11ShaderResourceView* texture,
|
|
|
|
const D3D11_RECT* rSource,
|
|
|
|
int SourceWidth,
|
|
|
|
int SourceHeight,
|
|
|
|
ID3D11PixelShader* PShader,
|
|
|
|
ID3D11VertexShader* Vshader,
|
|
|
|
ID3D11InputLayout* layout)
|
|
|
|
{
|
|
|
|
float sw = 1.0f /(float) SourceWidth;
|
|
|
|
float sh = 1.0f /(float) SourceHeight;
|
|
|
|
float u1 = ((float)rSource->left) * sw;
|
|
|
|
float u2 = ((float)rSource->right) * sw;
|
2010-07-17 16:13:37 +00:00
|
|
|
float v1 = ((float)rSource->top) * sh;
|
|
|
|
float v2 = ((float)rSource->bottom) * sh;
|
2010-06-13 19:50:06 +00:00
|
|
|
|
|
|
|
STQVertex coords[4] = {
|
|
|
|
{-1.0f, 1.0f, 0.0f, u1, v1},
|
|
|
|
{ 1.0f, 1.0f, 0.0f, u2, v1},
|
|
|
|
{-1.0f,-1.0f, 0.0f, u1, v2},
|
|
|
|
{ 1.0f,-1.0f, 0.0f, u2, v2},
|
|
|
|
};
|
|
|
|
|
|
|
|
// only upload the data to VRAM if it changed
|
2010-07-17 16:13:37 +00:00
|
|
|
if (tex_quad_data.u1 != u1 || tex_quad_data.v1 != v1 ||
|
|
|
|
tex_quad_data.u2 != u2 || tex_quad_data.v2 != v2)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
|
|
|
D3D11_MAPPED_SUBRESOURCE map;
|
|
|
|
D3D::context->Map(stqvb, 0, D3D11_MAP_WRITE_DISCARD, 0, &map);
|
|
|
|
memcpy(map.pData, coords, sizeof(coords));
|
|
|
|
D3D::context->Unmap(stqvb, 0);
|
2010-07-17 16:13:37 +00:00
|
|
|
tex_quad_data.u1 = u1;
|
|
|
|
tex_quad_data.v1 = v1;
|
|
|
|
tex_quad_data.u2 = u2;
|
|
|
|
tex_quad_data.v2 = v2;
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
UINT stride = sizeof(STQVertex);
|
|
|
|
UINT offset = 0;
|
|
|
|
|
|
|
|
D3D::context->IASetPrimitiveTopology(D3D11_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP);
|
|
|
|
D3D::context->IASetInputLayout(layout);
|
|
|
|
D3D::context->IASetVertexBuffers(0, 1, &stqvb, &stride, &offset);
|
2010-07-06 13:14:51 +00:00
|
|
|
D3D::context->PSSetShader(PShader, NULL, 0);
|
2010-06-13 19:50:06 +00:00
|
|
|
D3D::context->PSSetShaderResources(0, 1, &texture);
|
|
|
|
D3D::context->VSSetShader(Vshader, NULL, 0);
|
2010-06-18 23:33:07 +00:00
|
|
|
D3D::stateman->Apply();
|
2010-06-13 19:50:06 +00:00
|
|
|
D3D::context->Draw(4, 0);
|
|
|
|
|
|
|
|
ID3D11ShaderResourceView* texres = NULL;
|
|
|
|
context->PSSetShaderResources(0, 1, &texres); // immediately unbind the texture
|
|
|
|
}
|
|
|
|
|
|
|
|
void drawShadedTexSubQuad(ID3D11ShaderResourceView* texture,
|
|
|
|
const MathUtil::Rectangle<float>* rSource,
|
|
|
|
int SourceWidth,
|
|
|
|
int SourceHeight,
|
|
|
|
const MathUtil::Rectangle<float>* rDest,
|
|
|
|
ID3D11PixelShader* PShader,
|
|
|
|
ID3D11VertexShader* Vshader,
|
|
|
|
ID3D11InputLayout* layout)
|
|
|
|
{
|
|
|
|
float sw = 1.0f /(float) SourceWidth;
|
|
|
|
float sh = 1.0f /(float) SourceHeight;
|
|
|
|
float u1 = (rSource->left ) * sw;
|
|
|
|
float u2 = (rSource->right ) * sw;
|
|
|
|
float v1 = (rSource->top ) * sh;
|
|
|
|
float v2 = (rSource->bottom) * sh;
|
|
|
|
|
|
|
|
STSQVertex coords[4] = {
|
|
|
|
{ rDest->left , rDest->bottom, 0.0f, u1, v1},
|
|
|
|
{ rDest->right, rDest->bottom, 0.0f, u2, v1},
|
|
|
|
{ rDest->left , rDest->top , 0.0f, u1, v2},
|
|
|
|
{ rDest->right, rDest->top , 0.0f, u2, v2},
|
|
|
|
};
|
|
|
|
|
|
|
|
// only upload the data to VRAM if it changed
|
2010-07-17 16:13:37 +00:00
|
|
|
if (memcmp(rDest, &tex_sub_quad_data.rdest, sizeof(rDest)) != 0 ||
|
|
|
|
tex_sub_quad_data.u1 != u1 || tex_sub_quad_data.v1 != v1 ||
|
|
|
|
tex_sub_quad_data.u2 != u2 || tex_sub_quad_data.v2 != v2)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
|
|
|
D3D11_MAPPED_SUBRESOURCE map;
|
|
|
|
D3D::context->Map(stsqvb, 0, D3D11_MAP_WRITE_DISCARD, 0, &map);
|
|
|
|
memcpy(map.pData, coords, sizeof(coords));
|
|
|
|
D3D::context->Unmap(stsqvb, 0);
|
2010-07-17 16:13:37 +00:00
|
|
|
tex_sub_quad_data.u1 = u1;
|
|
|
|
tex_sub_quad_data.v1 = v1;
|
|
|
|
tex_sub_quad_data.u2 = u2;
|
|
|
|
tex_sub_quad_data.v2 = v2;
|
|
|
|
memcpy(&tex_sub_quad_data.rdest, &rDest, sizeof(rDest));
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
UINT stride = sizeof(STSQVertex);
|
|
|
|
UINT offset = 0;
|
|
|
|
|
|
|
|
context->IASetPrimitiveTopology(D3D11_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP);
|
|
|
|
context->IASetVertexBuffers(0, 1, &stsqvb, &stride, &offset);
|
|
|
|
context->IASetInputLayout(layout);
|
|
|
|
context->PSSetShaderResources(0, 1, &texture);
|
|
|
|
context->PSSetShader(PShader, NULL, 0);
|
|
|
|
context->VSSetShader(Vshader, NULL, 0);
|
2010-06-18 23:33:07 +00:00
|
|
|
stateman->Apply();
|
2010-06-13 19:50:06 +00:00
|
|
|
context->Draw(4, 0);
|
|
|
|
|
|
|
|
ID3D11ShaderResourceView* texres = NULL;
|
|
|
|
context->PSSetShaderResources(0, 1, &texres); // immediately unbind the texture
|
|
|
|
}
|
|
|
|
|
|
|
|
void drawClearQuad(u32 Color, float z, ID3D11PixelShader* PShader, ID3D11VertexShader* Vshader, ID3D11InputLayout* layout)
|
|
|
|
{
|
2010-07-17 16:13:37 +00:00
|
|
|
if (clear_quad_data.col != Color || clear_quad_data.z != z)
|
2010-06-13 19:50:06 +00:00
|
|
|
{
|
|
|
|
ClearVertex coords[4] = {
|
2010-06-15 21:19:09 +00:00
|
|
|
{-1.0f, 1.0f, z, Color},
|
|
|
|
{ 1.0f, 1.0f, z, Color},
|
|
|
|
{-1.0f, -1.0f, z, Color},
|
|
|
|
{ 1.0f, -1.0f, z, Color},
|
2010-06-13 19:50:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
D3D11_MAPPED_SUBRESOURCE map;
|
|
|
|
context->Map(clearvb, 0, D3D11_MAP_WRITE_DISCARD, 0, &map);
|
|
|
|
memcpy(map.pData, coords, sizeof(coords));
|
|
|
|
context->Unmap(clearvb, 0);
|
2010-07-17 16:13:37 +00:00
|
|
|
clear_quad_data.col = Color;
|
|
|
|
clear_quad_data.z = z;
|
2010-06-13 19:50:06 +00:00
|
|
|
}
|
|
|
|
context->VSSetShader(Vshader, NULL, 0);
|
|
|
|
context->PSSetShader(PShader, NULL, 0);
|
|
|
|
context->IASetInputLayout(layout);
|
|
|
|
|
|
|
|
UINT stride = sizeof(ClearVertex);
|
|
|
|
UINT offset = 0;
|
|
|
|
context->IASetPrimitiveTopology(D3D11_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP);
|
|
|
|
context->IASetVertexBuffers(0, 1, &clearvb, &stride, &offset);
|
2010-06-18 23:33:07 +00:00
|
|
|
stateman->Apply();
|
2010-06-13 19:50:06 +00:00
|
|
|
context->Draw(4, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace
|