mirror of
https://bitbucket.org/CPMADevs/cnq3
synced 2024-12-11 13:02:23 +00:00
296 lines
8.8 KiB
C++
296 lines
8.8 KiB
C++
/*
|
|
===========================================================================
|
|
Copyright (C) 2022-2024 Gian 'myT' Schellenbaum
|
|
|
|
This file is part of Challenge Quake 3 (CNQ3).
|
|
|
|
Challenge Quake 3 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; either version 2 of the License,
|
|
or (at your option) any later version.
|
|
|
|
Challenge Quake 3 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 for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Challenge Quake 3. If not, see <https://www.gnu.org/licenses/>.
|
|
===========================================================================
|
|
*/
|
|
// Shared Rendering Pipeline - Dear ImGUI integration
|
|
|
|
|
|
#include "srp_local.h"
|
|
#include "../imgui/imgui.h"
|
|
|
|
|
|
#define MAX_VERTEX_COUNT (64 << 10)
|
|
#define MAX_INDEX_COUNT (MAX_VERTEX_COUNT << 3)
|
|
|
|
|
|
#pragma pack(push, 4)
|
|
struct VertexRC
|
|
{
|
|
float mvp[16];
|
|
};
|
|
|
|
struct PixelRC
|
|
{
|
|
uint32_t texture;
|
|
uint32_t sampler;
|
|
float mip;
|
|
};
|
|
#pragma pack(pop)
|
|
|
|
|
|
HTexture ImGUI::Init(bool ddhi_, const ShaderByteCode& vs, const ShaderByteCode& ps, TextureFormat::Id rtFormat, HDescriptorTable descTable, RootSignatureDesc* rootSigDesc)
|
|
{
|
|
ddhi = ddhi_;
|
|
descriptorTable = descTable;
|
|
|
|
ImGuiIO& io = ImGui::GetIO();
|
|
io.DisplaySize.x = glConfig.vidWidth;
|
|
io.DisplaySize.y = glConfig.vidHeight;
|
|
|
|
if(srp.firstInit)
|
|
{
|
|
io.BackendRendererUserData = this;
|
|
io.BackendRendererName = "CNQ3 Direct3D 12";
|
|
io.BackendFlags |= ImGuiBackendFlags_RendererHasVtxOffset; // We can honor the ImDrawCmd::VtxOffset field, allowing for large meshes.
|
|
|
|
for(int i = 0; i < FrameCount; i++)
|
|
{
|
|
FrameResources* fr = &frameResources[i];
|
|
|
|
BufferDesc vtx("Dear ImGUI index", MAX_INDEX_COUNT * sizeof(ImDrawIdx), ResourceStates::IndexBufferBit);
|
|
vtx.memoryUsage = MemoryUsage::Upload;
|
|
fr->indexBuffer = CreateBuffer(vtx);
|
|
|
|
BufferDesc idx("Dear ImGUI vertex", MAX_VERTEX_COUNT * sizeof(ImDrawData), ResourceStates::VertexBufferBit);
|
|
idx.memoryUsage = MemoryUsage::Upload;
|
|
fr->vertexBuffer = CreateBuffer(idx);
|
|
}
|
|
|
|
if(ddhi)
|
|
{
|
|
rootSignature = RHI_MAKE_NULL_HANDLE();
|
|
}
|
|
else
|
|
{
|
|
RootSignatureDesc desc = *rootSigDesc;
|
|
desc.name = "Dear ImGUI";
|
|
desc.constants[ShaderStage::Vertex].byteCount = sizeof(VertexRC);
|
|
desc.constants[ShaderStage::Pixel].byteCount = sizeof(PixelRC);
|
|
rootSignature = CreateRootSignature(desc);
|
|
}
|
|
|
|
{
|
|
unsigned char* pixels;
|
|
int width, height;
|
|
io.Fonts->GetTexDataAsRGBA32(&pixels, &width, &height);
|
|
|
|
TextureDesc desc("Dear ImGUI font atlas", width, height, 1);
|
|
fontAtlas = CreateTexture(desc);
|
|
|
|
MappedTexture update;
|
|
BeginTextureUpload(update, fontAtlas);
|
|
for(uint32_t r = 0; r < update.rowCount; ++r)
|
|
{
|
|
memcpy(update.mappedData + r * update.dstRowByteCount, pixels + r * update.srcRowByteCount, update.srcRowByteCount);
|
|
}
|
|
EndTextureUpload();
|
|
}
|
|
}
|
|
|
|
{
|
|
GraphicsPipelineDesc desc("Dear ImGUI", rootSignature);
|
|
desc.shortLifeTime = true;
|
|
desc.vertexShader = vs;
|
|
desc.pixelShader = ps;
|
|
desc.vertexLayout.bindingStrides[0] = sizeof(ImDrawVert);
|
|
desc.vertexLayout.AddAttribute(0, ShaderSemantic::Position,
|
|
DataType::Float32, 2, offsetof(ImDrawVert, pos));
|
|
desc.vertexLayout.AddAttribute(0, ShaderSemantic::TexCoord,
|
|
DataType::Float32, 2, offsetof(ImDrawVert, uv));
|
|
desc.vertexLayout.AddAttribute(0, ShaderSemantic::Color,
|
|
DataType::UNorm8, 4, offsetof(ImDrawVert, col));
|
|
desc.depthStencil.depthComparison = ComparisonFunction::Always;
|
|
desc.depthStencil.depthStencilFormat = TextureFormat::Depth32_Float;
|
|
desc.depthStencil.enableDepthTest = false;
|
|
desc.depthStencil.enableDepthWrites = false;
|
|
desc.rasterizer.cullMode = CT_TWO_SIDED;
|
|
desc.AddRenderTarget(GLS_SRCBLEND_SRC_ALPHA | GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA, rtFormat);
|
|
pipeline = CreateGraphicsPipeline(desc);
|
|
}
|
|
|
|
if(ddhi)
|
|
{
|
|
const uint32_t fontIndex = GetTextureIndexSRV(fontAtlas);
|
|
io.Fonts->SetTexID((ImTextureID)fontIndex);
|
|
}
|
|
|
|
return fontAtlas;
|
|
}
|
|
|
|
void ImGUI::RegisterFontAtlas(uint32_t fontIndex)
|
|
{
|
|
ImGui::GetIO().Fonts->SetTexID((ImTextureID)fontIndex);
|
|
}
|
|
|
|
void ImGUI::Draw(HTexture renderTarget)
|
|
{
|
|
if(r_debugUI->integer == 0)
|
|
{
|
|
EndFrame();
|
|
return;
|
|
}
|
|
|
|
srp.renderMode = RenderMode::ImGui;
|
|
|
|
ImGuiIO& io = ImGui::GetIO();
|
|
io.DisplaySize.x = glConfig.vidWidth;
|
|
io.DisplaySize.y = glConfig.vidHeight;
|
|
|
|
EndFrame();
|
|
ImGui::Render();
|
|
|
|
const ImDrawData* drawData = ImGui::GetDrawData();
|
|
|
|
// avoid rendering when minimized
|
|
if(drawData->DisplaySize.x <= 0.0f || drawData->DisplaySize.y <= 0.0f)
|
|
{
|
|
srp.renderMode = RenderMode::None;
|
|
return;
|
|
}
|
|
|
|
SCOPED_RENDER_PASS("Dear ImGUI", 0.5f, 0.5f, 1.0f);
|
|
|
|
FrameResources* fr = &frameResources[GetFrameIndex()];
|
|
|
|
// Upload vertex/index data into a single contiguous GPU buffer
|
|
ImDrawVert* vtxDst = (ImDrawVert*)MapBuffer(fr->vertexBuffer);
|
|
ImDrawIdx* idxDst = (ImDrawIdx*)MapBuffer(fr->indexBuffer);
|
|
ImDrawVert* const vtxDstEnd = vtxDst + MAX_VERTEX_COUNT;
|
|
ImDrawIdx* const idxDstEnd = idxDst + MAX_INDEX_COUNT;
|
|
for(int cl = 0; cl < drawData->CmdListsCount; cl++)
|
|
{
|
|
const ImDrawList* cmdList = drawData->CmdLists[cl];
|
|
if(vtxDst + cmdList->VtxBuffer.Size > vtxDstEnd)
|
|
{
|
|
Q_assert(!"Dear ImGui vertex buffer too small");
|
|
break;
|
|
}
|
|
if(idxDst + cmdList->IdxBuffer.Size > idxDstEnd)
|
|
{
|
|
Q_assert(!"Dear ImGui index buffer too small");
|
|
break;
|
|
}
|
|
memcpy(vtxDst, cmdList->VtxBuffer.Data, cmdList->VtxBuffer.Size * sizeof(ImDrawVert));
|
|
memcpy(idxDst, cmdList->IdxBuffer.Data, cmdList->IdxBuffer.Size * sizeof(ImDrawIdx));
|
|
vtxDst += cmdList->VtxBuffer.Size;
|
|
idxDst += cmdList->IdxBuffer.Size;
|
|
}
|
|
UnmapBuffer(fr->vertexBuffer);
|
|
UnmapBuffer(fr->indexBuffer);
|
|
|
|
// Setup orthographic projection matrix into our constant buffer
|
|
// Our visible imgui space lies from draw_data->DisplayPos (top left) to draw_data->DisplayPos+data_data->DisplaySize (bottom right).
|
|
const float L = drawData->DisplayPos.x;
|
|
const float R = drawData->DisplayPos.x + drawData->DisplaySize.x;
|
|
const float T = drawData->DisplayPos.y;
|
|
const float B = drawData->DisplayPos.y + drawData->DisplaySize.y;
|
|
const VertexRC vertexRC =
|
|
{
|
|
2.0f / (R - L), 0.0f, 0.0f, 0.0f,
|
|
0.0f, 2.0f / (T - B), 0.0f, 0.0f,
|
|
0.0f, 0.0f, 0.5f, 0.0f,
|
|
(R + L) / (L - R), (T + B) / (B - T), 0.5f, 1.0f
|
|
};
|
|
const uint32_t vertexStride = sizeof(ImDrawVert);
|
|
static_assert(sizeof(ImDrawIdx) == 4, "uint32 indices expected!");
|
|
|
|
CmdBindRenderTargets(1, &renderTarget, NULL);
|
|
if(!ddhi)
|
|
{
|
|
CmdBindRootSignature(rootSignature);
|
|
}
|
|
CmdBindPipeline(pipeline);
|
|
if(!ddhi)
|
|
{
|
|
CmdBindDescriptorTable(rootSignature, descriptorTable);
|
|
}
|
|
CmdBindVertexBuffers(1, &fr->vertexBuffer, &vertexStride, NULL);
|
|
CmdBindIndexBuffer(fr->indexBuffer, IndexType::UInt32, 0);
|
|
CmdSetViewport(0, 0, drawData->DisplaySize.x, drawData->DisplaySize.y);
|
|
if(ddhi)
|
|
{
|
|
CmdSetGraphicsRootConstants(0, sizeof(vertexRC), &vertexRC);
|
|
}
|
|
else
|
|
{
|
|
CmdSetRootConstants(rootSignature, ShaderStage::Vertex, &vertexRC);
|
|
}
|
|
|
|
// Render command lists
|
|
// (Because we merged all buffers into a single one, we maintain our own offset into them)
|
|
int globalVtxOffset = 0;
|
|
int globalIdxOffset = 0;
|
|
ImVec2 clipOff = drawData->DisplayPos;
|
|
for(int cl = 0; cl < drawData->CmdListsCount; cl++)
|
|
{
|
|
const ImDrawList* cmdList = drawData->CmdLists[cl];
|
|
for(int c = 0; c < cmdList->CmdBuffer.Size; c++)
|
|
{
|
|
const ImDrawCmd* cmd = &cmdList->CmdBuffer[c];
|
|
|
|
// Project scissor/clipping rectangles into framebuffer space
|
|
ImVec2 clip_min(cmd->ClipRect.x - clipOff.x, cmd->ClipRect.y - clipOff.y);
|
|
ImVec2 clip_max(cmd->ClipRect.z - clipOff.x, cmd->ClipRect.w - clipOff.y);
|
|
if(clip_max.x <= clip_min.x || clip_max.y <= clip_min.y)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
PixelRC pixelRC = {};
|
|
pixelRC.texture = (uint32_t)cmd->TextureId & 0xFFFF;
|
|
pixelRC.sampler = GetSamplerIndex(TW_CLAMP_TO_EDGE, TextureFilter::Linear);
|
|
pixelRC.mip = (float)(((uint32_t)cmd->TextureId >> 16) & 0xFFFF);
|
|
if(ddhi)
|
|
{
|
|
CmdSetGraphicsRootConstants(sizeof(vertexRC), sizeof(pixelRC), &pixelRC);
|
|
}
|
|
else
|
|
{
|
|
CmdSetRootConstants(rootSignature, ShaderStage::Pixel, &pixelRC);
|
|
}
|
|
|
|
// Apply Scissor/clipping rectangle, Draw
|
|
CmdSetScissor(clip_min.x, clip_min.y, clip_max.x - clip_min.x, clip_max.y - clip_min.y);
|
|
CmdDrawIndexed(cmd->ElemCount, cmd->IdxOffset + globalIdxOffset, cmd->VtxOffset + globalVtxOffset);
|
|
}
|
|
|
|
globalIdxOffset += cmdList->IdxBuffer.Size;
|
|
globalVtxOffset += cmdList->VtxBuffer.Size;
|
|
}
|
|
|
|
srp.renderMode = RenderMode::None;
|
|
}
|
|
|
|
void ImGUI::BeginFrame()
|
|
{
|
|
if(!frameStarted)
|
|
{
|
|
ImGui::NewFrame();
|
|
frameStarted = true;
|
|
}
|
|
}
|
|
|
|
void ImGUI::EndFrame()
|
|
{
|
|
if(frameStarted)
|
|
{
|
|
ImGui::EndFrame();
|
|
frameStarted = false;
|
|
}
|
|
}
|