metaforce/specter/lib/View.cpp

437 lines
14 KiB
C++
Raw Normal View History

2015-11-21 23:45:02 +00:00
#include "Specter/View.hpp"
#include "Specter/ViewResources.hpp"
2015-11-29 02:55:30 +00:00
#include "Specter/RootView.hpp"
2015-11-21 23:45:02 +00:00
namespace Specter
{
static LogVisor::LogModule Log("Specter::View");
2015-11-21 23:45:02 +00:00
2016-02-23 02:33:59 +00:00
static const char* GLSLSolidVS =
"#version 330\n"
"layout(location=0) in vec3 posIn;\n"
"layout(location=1) in vec4 colorIn;\n"
SPECTER_VIEW_VERT_BLOCK_GLSL
"struct VertToFrag\n"
"{\n"
" vec4 color;\n"
"};\n"
"out VertToFrag vtf;\n"
"void main()\n"
"{\n"
" vtf.color = colorIn * mulColor;\n"
" gl_Position = mv * vec4(posIn, 1.0);\n"
"}\n";
2015-11-26 00:24:01 +00:00
2016-02-23 02:33:59 +00:00
static const char* GLSLSolidFS =
"#version 330\n"
"struct VertToFrag\n"
"{\n"
" vec4 color;\n"
"};\n"
"in VertToFrag vtf;\n"
"layout(location=0) out vec4 colorOut;\n"
"void main()\n"
"{\n"
" colorOut = vtf.color;\n"
"}\n";
2015-11-26 00:24:01 +00:00
2016-02-23 02:33:59 +00:00
static const char* GLSLTexVS =
"#version 330\n"
"layout(location=0) in vec3 posIn;\n"
"layout(location=1) in vec2 uvIn;\n"
SPECTER_VIEW_VERT_BLOCK_GLSL
"struct VertToFrag\n"
"{\n"
" vec4 color;\n"
" vec2 uv;\n"
"};\n"
"out VertToFrag vtf;\n"
"void main()\n"
"{\n"
" vtf.uv = uvIn;\n"
" vtf.color = mulColor;\n"
" gl_Position = mv * vec4(posIn, 1.0);\n"
"}\n";
2015-11-30 00:21:42 +00:00
2016-02-23 02:33:59 +00:00
static const char* GLSLTexFS =
"#version 330\n"
"struct VertToFrag\n"
"{\n"
" vec4 color;\n"
" vec2 uv;\n"
"};\n"
"in VertToFrag vtf;\n"
"uniform sampler2D tex;\n"
"layout(location=0) out vec4 colorOut;\n"
"void main()\n"
"{\n"
" colorOut = texture(tex, vtf.uv) * vtf.color;\n"
"}\n";
2015-11-30 00:21:42 +00:00
2016-02-23 02:33:59 +00:00
void View::Resources::init(boo::GLDataFactory* factory, const IThemeData& theme)
{
2015-11-26 00:24:01 +00:00
static const char* BlockNames[] = {"SpecterViewBlock"};
2016-02-23 02:33:59 +00:00
m_solidShader = factory->newShaderPipeline(GLSLSolidVS, GLSLSolidFS, 0, nullptr, 1, BlockNames,
2015-11-26 00:24:01 +00:00
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
2015-11-30 00:21:42 +00:00
2016-02-23 02:33:59 +00:00
m_texShader = factory->newShaderPipeline(GLSLTexVS, GLSLTexFS, 1, "tex", 1, BlockNames,
2015-11-30 00:21:42 +00:00
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
2015-11-26 00:24:01 +00:00
}
2015-11-28 04:05:27 +00:00
#if _WIN32
2015-11-26 00:24:01 +00:00
2016-01-11 02:48:56 +00:00
void View::Resources::init(boo::ID3DDataFactory* factory, const IThemeData& theme)
2015-11-27 22:20:22 +00:00
{
2015-11-30 00:21:42 +00:00
static const char* SolidVS =
2015-11-27 22:20:22 +00:00
"struct VertData\n"
"{\n"
" float3 posIn : POSITION;\n"
" float4 colorIn : COLOR;\n"
"};\n"
SPECTER_VIEW_VERT_BLOCK_HLSL
"struct VertToFrag\n"
"{\n"
" float4 position : SV_Position;\n"
" float4 color : COLOR;\n"
"};\n"
"VertToFrag main(in VertData v)\n"
"{\n"
" VertToFrag vtf;\n"
" vtf.color = v.colorIn * mulColor;\n"
2015-11-27 22:20:22 +00:00
" vtf.position = mul(mv, float4(v.posIn, 1.0));\n"
" return vtf;\n"
"}\n";
2015-11-30 00:21:42 +00:00
static const char* SolidFS =
2015-11-27 22:20:22 +00:00
"struct VertToFrag\n"
"{\n"
" float4 position : SV_Position;\n"
" float4 color : COLOR;\n"
"};\n"
"float4 main(in VertToFrag vtf) : SV_Target0\n"
"{\n"
" return vtf.color;\n"
"}\n";
2015-11-30 00:21:42 +00:00
static const char* TexVS =
"struct VertData\n"
"{\n"
" float3 posIn : POSITION;\n"
" float2 uvIn : UV;\n"
"};\n"
SPECTER_VIEW_VERT_BLOCK_HLSL
"struct VertToFrag\n"
"{\n"
" float4 position : SV_Position;\n"
" float4 color : COLOR;\n"
2015-11-30 00:21:42 +00:00
" float2 uv : UV;\n"
"};\n"
"VertToFrag main(in VertData v)\n"
"{\n"
" VertToFrag vtf;\n"
" vtf.uv = v.uvIn;\n"
" vtf.color = mulColor;\n"
2015-11-30 00:21:42 +00:00
" vtf.position = mul(mv, float4(v.posIn, 1.0));\n"
" return vtf;\n"
"}\n";
static const char* TexFS =
"struct VertToFrag\n"
"{\n"
" float4 position : SV_Position;\n"
" float4 color : COLOR;\n"
2015-11-30 00:21:42 +00:00
" float2 uv : UV;\n"
"};\n"
"Texture2D tex : register(t0);\n"
"SamplerState samp : register(s0);\n"
"float4 main(in VertToFrag vtf) : SV_Target0\n"
"{\n"
" return tex.Sample(samp, vtf.uv) * vtf.color;\n"
2015-11-30 00:21:42 +00:00
"}\n";
2015-11-30 03:41:53 +00:00
boo::VertexElementDescriptor solidvdescs[] =
2015-11-27 22:20:22 +00:00
{
{nullptr, nullptr, boo::VertexSemantic::Position4},
2015-12-05 00:42:46 +00:00
{nullptr, nullptr, boo::VertexSemantic::Color}
2015-11-27 22:20:22 +00:00
};
2015-11-30 03:41:53 +00:00
m_solidVtxFmt = factory->newVertexFormat(2, solidvdescs);
2015-11-27 22:20:22 +00:00
ComPtr<ID3DBlob> vertBlob;
ComPtr<ID3DBlob> fragBlob;
ComPtr<ID3DBlob> pipeBlob;
2015-11-30 00:21:42 +00:00
m_solidShader = factory->newShaderPipeline(SolidVS, SolidFS, vertBlob, fragBlob, pipeBlob, m_solidVtxFmt,
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
2015-11-30 03:41:53 +00:00
boo::VertexElementDescriptor texvdescs[] =
2015-11-30 00:21:42 +00:00
{
{nullptr, nullptr, boo::VertexSemantic::Position4},
{nullptr, nullptr, boo::VertexSemantic::UV4}
};
2015-11-30 03:41:53 +00:00
m_texVtxFmt = factory->newVertexFormat(2, texvdescs);
2015-11-30 00:21:42 +00:00
vertBlob.Reset();
fragBlob.Reset();
pipeBlob.Reset();
m_texShader = factory->newShaderPipeline(TexVS, TexFS, vertBlob, fragBlob, pipeBlob, m_texVtxFmt,
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
2015-11-27 22:20:22 +00:00
}
2015-11-28 04:05:27 +00:00
2016-02-23 02:33:59 +00:00
#endif
#if BOO_HAS_METAL
2015-11-28 04:05:27 +00:00
void View::Resources::init(boo::MetalDataFactory* factory, const IThemeData& theme)
2015-11-28 04:05:27 +00:00
{
2015-11-30 00:21:42 +00:00
static const char* SolidVS =
2015-11-28 04:05:27 +00:00
"#include <metal_stdlib>\n"
"using namespace metal;\n"
"struct VertData\n"
"{\n"
" float3 posIn [[ attribute(0) ]];\n"
" float4 colorIn [[ attribute(1) ]];\n"
"};\n"
SPECTER_VIEW_VERT_BLOCK_METAL
"struct VertToFrag\n"
"{\n"
" float4 position [[ position ]];\n"
" float4 color;\n"
"};\n"
"vertex VertToFrag vmain(VertData v [[ stage_in ]], constant SpecterViewBlock& view [[ buffer(2) ]])\n"
"{\n"
" VertToFrag vtf;\n"
" vtf.color = v.colorIn * view.mulColor;\n"
2015-11-28 04:05:27 +00:00
" vtf.position = view.mv * float4(v.posIn, 1.0);\n"
" return vtf;\n"
"}\n";
2015-11-30 00:21:42 +00:00
static const char* SolidFS =
2015-11-28 04:05:27 +00:00
"#include <metal_stdlib>\n"
"using namespace metal;\n"
"struct VertToFrag\n"
"{\n"
" float4 position [[ position ]];\n"
" float4 color;\n"
"};\n"
"fragment float4 fmain(VertToFrag vtf [[ stage_in ]])\n"
"{\n"
" return vtf.color;\n"
"}\n";
2015-11-30 00:21:42 +00:00
static const char* TexVS =
"#include <metal_stdlib>\n"
"using namespace metal;\n"
"struct VertData\n"
"{\n"
" float3 posIn [[ attribute(0) ]];\n"
" float2 uvIn [[ attribute(1) ]];\n"
"};\n"
SPECTER_VIEW_VERT_BLOCK_METAL
"struct VertToFrag\n"
"{\n"
" float4 position [[ position ]];\n"
" float4 color;\n"
2015-11-30 00:21:42 +00:00
" float2 uv;\n"
"};\n"
"vertex VertToFrag vmain(VertData v [[ stage_in ]], constant SpecterViewBlock& view [[ buffer(2) ]])\n"
"{\n"
" VertToFrag vtf;\n"
" vtf.uv = v.uvIn;\n"
" vtf.color = view.mulColor;\n"
2015-11-30 00:21:42 +00:00
" vtf.position = view.mv * float4(v.posIn, 1.0);\n"
" return vtf;\n"
"}\n";
static const char* TexFS =
"#include <metal_stdlib>\n"
"using namespace metal;\n"
"constexpr sampler samp(address::repeat);\n"
"struct VertToFrag\n"
"{\n"
" float4 position [[ position ]];\n"
" float4 color;\n"
2015-11-30 00:21:42 +00:00
" float2 uv;\n"
"};\n"
"fragment float4 fmain(VertToFrag vtf [[ stage_in ]], texture2d<float> tex [[ texture(0) ]])\n"
"{\n"
" return tex.sample(samp, vtf.uv) * vtf.color;\n"
2015-11-30 00:21:42 +00:00
"}\n";
2015-11-28 04:05:27 +00:00
2015-11-30 03:41:53 +00:00
boo::VertexElementDescriptor solidvdescs[] =
2015-11-28 04:05:27 +00:00
{
{nullptr, nullptr, boo::VertexSemantic::Position4},
2015-12-05 00:42:46 +00:00
{nullptr, nullptr, boo::VertexSemantic::Color}
2015-11-28 04:05:27 +00:00
};
2015-11-30 03:41:53 +00:00
m_solidVtxFmt = factory->newVertexFormat(2, solidvdescs);
2015-11-28 04:05:27 +00:00
2015-11-30 00:21:42 +00:00
m_solidShader = factory->newShaderPipeline(SolidVS, SolidFS, m_solidVtxFmt, 1,
2015-11-28 04:05:27 +00:00
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
2015-11-30 00:21:42 +00:00
2015-11-30 03:41:53 +00:00
boo::VertexElementDescriptor texvdescs[] =
2015-11-30 00:21:42 +00:00
{
{nullptr, nullptr, boo::VertexSemantic::Position4},
{nullptr, nullptr, boo::VertexSemantic::UV4}
};
2015-11-30 03:41:53 +00:00
m_texVtxFmt = factory->newVertexFormat(2, texvdescs);
2015-11-30 00:21:42 +00:00
m_texShader = factory->newShaderPipeline(TexVS, TexFS, m_texVtxFmt, 1,
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
2015-11-28 04:05:27 +00:00
}
2016-02-23 02:33:59 +00:00
#endif
#if BOO_HAS_VULKAN
void View::Resources::init(boo::VulkanDataFactory* factory, const IThemeData& theme)
{
boo::VertexElementDescriptor solidvdescs[] =
{
{nullptr, nullptr, boo::VertexSemantic::Position4},
{nullptr, nullptr, boo::VertexSemantic::Color}
};
m_solidVtxFmt = factory->newVertexFormat(2, solidvdescs);
m_solidShader = factory->newShaderPipeline(GLSLSolidVS, GLSLSolidFS, m_solidVtxFmt,
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
boo::VertexElementDescriptor texvdescs[] =
{
{nullptr, nullptr, boo::VertexSemantic::Position4},
{nullptr, nullptr, boo::VertexSemantic::UV4}
};
m_texVtxFmt = factory->newVertexFormat(2, texvdescs);
m_texShader = factory->newShaderPipeline(GLSLTexVS, GLSLTexFS, m_texVtxFmt,
boo::BlendFactor::SrcAlpha, boo::BlendFactor::InvSrcAlpha,
false, false, false);
}
2015-11-28 04:05:27 +00:00
#endif
2015-11-27 22:20:22 +00:00
2015-12-29 02:02:43 +00:00
void View::buildResources(ViewResources& res)
2015-11-26 00:24:01 +00:00
{
2015-11-26 07:35:43 +00:00
m_viewVertBlockBuf =
2015-12-29 02:02:43 +00:00
res.m_factory->newDynamicBuffer(boo::BufferUse::Uniform,
sizeof(ViewBlock), 1);
m_bgVertsBinding.initSolid(res, 4, m_viewVertBlockBuf);
2015-11-29 02:55:30 +00:00
}
2015-12-29 02:02:43 +00:00
View::View(ViewResources& res)
: m_rootView(*static_cast<RootView*>(this)),
m_parentView(*static_cast<RootView*>(this))
2015-11-29 02:55:30 +00:00
{
2015-12-29 02:02:43 +00:00
buildResources(res);
2015-11-29 02:55:30 +00:00
}
2015-11-26 00:24:01 +00:00
2015-12-29 02:02:43 +00:00
View::View(ViewResources& res, View& parentView)
2015-12-05 00:42:46 +00:00
: m_rootView(parentView.rootView()),
2015-11-29 02:55:30 +00:00
m_parentView(parentView)
{
2015-12-29 02:02:43 +00:00
buildResources(res);
2015-11-29 02:55:30 +00:00
}
void View::updateSize()
{
resized(m_rootView.rootRect(), m_subRect);
2015-11-26 00:24:01 +00:00
}
2015-11-26 23:03:56 +00:00
void View::resized(const boo::SWindowRect& root, const boo::SWindowRect& sub)
2015-11-26 07:35:43 +00:00
{
2015-11-29 02:55:30 +00:00
m_subRect = sub;
2015-11-26 23:03:56 +00:00
m_viewVertBlock.setViewRect(root, sub);
2015-12-05 00:42:46 +00:00
m_bgRect[0].m_pos.assign(0.f, sub.size[1], 0.f);
m_bgRect[1].m_pos.assign(0.f, 0.f, 0.f);
m_bgRect[2].m_pos.assign(sub.size[0], sub.size[1], 0.f);
m_bgRect[3].m_pos.assign(sub.size[0], 0.f, 0.f);
m_viewVertBlockBuf->load(&m_viewVertBlock, sizeof(ViewBlock));
m_bgVertsBinding.load(m_bgRect, sizeof(m_bgRect));
2015-11-26 07:35:43 +00:00
}
void View::resized(const ViewBlock& vb, const boo::SWindowRect& sub)
{
m_subRect = sub;
m_bgRect[0].m_pos.assign(0.f, sub.size[1], 0.f);
m_bgRect[1].m_pos.assign(0.f, 0.f, 0.f);
m_bgRect[2].m_pos.assign(sub.size[0], sub.size[1], 0.f);
m_bgRect[3].m_pos.assign(sub.size[0], 0.f, 0.f);
m_viewVertBlockBuf->load(&vb, sizeof(ViewBlock));
m_bgVertsBinding.load(m_bgRect, sizeof(m_bgRect));
}
2015-11-26 00:24:01 +00:00
void View::draw(boo::IGraphicsCommandQueue* gfxQ)
{
gfxQ->setShaderDataBinding(m_bgVertsBinding);
2015-11-26 00:24:01 +00:00
gfxQ->draw(0, 4);
}
2015-12-29 02:02:43 +00:00
void View::commitResources(ViewResources& res)
2015-11-30 00:21:42 +00:00
{
if (m_gfxData)
Log.report(LogVisor::FatalError, "multiple resource commits not allowed");
2015-12-29 02:02:43 +00:00
m_gfxData = res.m_factory->commit();
2015-11-30 00:21:42 +00:00
}
void View::VertexBufferBinding::initSolid(ViewResources& res, size_t count, boo::IGraphicsBuffer* viewBlockBuf)
{
m_vertsBuf = res.m_factory->newDynamicBuffer(boo::BufferUse::Vertex, sizeof(SolidShaderVert), count);
if (!res.m_viewRes.m_solidVtxFmt)
{
boo::VertexElementDescriptor vdescs[] =
{
{m_vertsBuf, nullptr, boo::VertexSemantic::Position4},
{m_vertsBuf, nullptr, boo::VertexSemantic::Color}
};
m_vtxFmt = res.m_factory->newVertexFormat(2, vdescs);
boo::IGraphicsBuffer* bufs[] = {viewBlockBuf};
m_shaderBinding = res.m_factory->newShaderDataBinding(res.m_viewRes.m_solidShader,
m_vtxFmt, m_vertsBuf, nullptr,
nullptr, 1, bufs, 0, nullptr);
}
else
{
boo::IGraphicsBuffer* bufs[] = {viewBlockBuf};
m_shaderBinding = res.m_factory->newShaderDataBinding(res.m_viewRes.m_solidShader,
res.m_viewRes.m_solidVtxFmt,
m_vertsBuf, nullptr,
nullptr, 1, bufs, 0, nullptr);
}
}
void View::VertexBufferBinding::initTex(ViewResources& res, size_t count, boo::IGraphicsBuffer* viewBlockBuf, boo::ITexture* texture)
{
m_vertsBuf = res.m_factory->newDynamicBuffer(boo::BufferUse::Vertex, sizeof(TexShaderVert), count);
if (!res.m_viewRes.m_texVtxFmt)
{
boo::VertexElementDescriptor vdescs[] =
{
{m_vertsBuf, nullptr, boo::VertexSemantic::Position4},
{m_vertsBuf, nullptr, boo::VertexSemantic::UV4}
};
m_vtxFmt = res.m_factory->newVertexFormat(2, vdescs);
boo::IGraphicsBuffer* bufs[] = {viewBlockBuf};
boo::ITexture* tex[] = {texture};
m_shaderBinding = res.m_factory->newShaderDataBinding(res.m_viewRes.m_texShader,
m_vtxFmt, m_vertsBuf, nullptr,
nullptr, 1, bufs, 1, tex);
}
else
{
boo::IGraphicsBuffer* bufs[] = {viewBlockBuf};
boo::ITexture* tex[] = {texture};
m_shaderBinding = res.m_factory->newShaderDataBinding(res.m_viewRes.m_texShader,
res.m_viewRes.m_texVtxFmt,
m_vertsBuf, nullptr,
nullptr, 1, bufs, 1, tex);
}
}
2015-11-30 00:21:42 +00:00
2015-11-21 23:45:02 +00:00
}