Finished UI Render Context
This commit is contained in:
@ -5,9 +5,8 @@
|
|||||||
|
|
||||||
#include "UICanvas.hpp"
|
#include "UICanvas.hpp"
|
||||||
#include "display/pass/RenderPass.hpp"
|
#include "display/pass/RenderPass.hpp"
|
||||||
#include "display/shader/UIShader.hpp"
|
|
||||||
|
|
||||||
#include "display/mesh/QuadMesh.hpp"
|
#include "display/mesh/QuadMesh.hpp"
|
||||||
|
#include "ui/UIComponent.hpp"
|
||||||
|
|
||||||
using namespace Dawn;
|
using namespace Dawn;
|
||||||
|
|
||||||
@ -37,30 +36,68 @@ void UICanvas::onDispose() {
|
|||||||
std::vector<std::shared_ptr<IRenderPass>> UICanvas::getPasses(
|
std::vector<std::shared_ptr<IRenderPass>> UICanvas::getPasses(
|
||||||
struct RenderPassContext &ctx
|
struct RenderPassContext &ctx
|
||||||
) {
|
) {
|
||||||
std::vector<std::shared_ptr<IRenderPass>> passes;
|
if(this->components.empty()) return {};
|
||||||
std::unordered_map<shadertexturebinding_t, std::shared_ptr<Texture>> textures;
|
|
||||||
UIShaderData data = {
|
data.projection = ctx.camera->getProjection();
|
||||||
.projection = ctx.camera->getProjection(),
|
data.view = ctx.camera->getItem()->getWorldTransform();
|
||||||
.view = ctx.camera->getItem()->getWorldTransform(),
|
data.model = this->getItem()->getWorldTransform();
|
||||||
.model = this->getItem()->getWorldTransform()
|
|
||||||
};
|
this->passes.clear();
|
||||||
|
this->textureBindings.clear();
|
||||||
|
this->textures.clear();
|
||||||
|
quadCount = 0;
|
||||||
|
nextBinding = 0;
|
||||||
|
|
||||||
size_t quadCount = 0;
|
|
||||||
auto itComponents = components.begin();
|
auto itComponents = components.begin();
|
||||||
|
auto self = std::ref(*this);
|
||||||
while(itComponents != components.end()) {
|
while(itComponents != components.end()) {
|
||||||
auto component = *itComponents;
|
auto component = *itComponents;
|
||||||
|
component->getQuads({ 0, 0 }, self);
|
||||||
// Get this components' quads.
|
|
||||||
auto quads = component->getQuads({ 0, 0 });
|
|
||||||
for(auto quad : quads) {
|
|
||||||
data.quads[quadCount++] = quad;
|
|
||||||
assertTrue(quadCount <= UI_SHADER_QUAD_COUNT, "Too many UI quads!");
|
|
||||||
}
|
|
||||||
++itComponents;
|
++itComponents;
|
||||||
}
|
}
|
||||||
|
flushPass();
|
||||||
|
|
||||||
// No render passes if no quads.
|
return passes;
|
||||||
if(quadCount == 0) return passes;
|
}
|
||||||
|
|
||||||
|
void UICanvas::addQuad(
|
||||||
|
const glm::vec4 quad,
|
||||||
|
const glm::vec4 uvs,
|
||||||
|
const struct Color color,
|
||||||
|
const std::shared_ptr<Texture> text
|
||||||
|
) {
|
||||||
|
float_t fTexture = -1;
|
||||||
|
if(text == nullptr) {
|
||||||
|
fTexture = -1;
|
||||||
|
} else {
|
||||||
|
shadertexturebinding_t texture;
|
||||||
|
auto bindingIt = textureBindings.find(text);
|
||||||
|
if(bindingIt == textureBindings.end()) {
|
||||||
|
if(nextBinding >= UI_SHADER_TEXTURE_COUNT) {
|
||||||
|
flushPass();
|
||||||
|
}
|
||||||
|
textureBindings[text] = nextBinding;
|
||||||
|
textures[nextBinding] = text;
|
||||||
|
data.textures[nextBinding] = nextBinding;
|
||||||
|
texture = nextBinding++;
|
||||||
|
} else {
|
||||||
|
texture = bindingIt->second;
|
||||||
|
}
|
||||||
|
fTexture = (float_t)texture;
|
||||||
|
}
|
||||||
|
|
||||||
|
data.quads[quadCount] = {
|
||||||
|
quad,
|
||||||
|
uvs,
|
||||||
|
color,
|
||||||
|
fTexture
|
||||||
|
};
|
||||||
|
quadCount++;
|
||||||
|
if(quadCount == UI_SHADER_QUAD_COUNT) flushPass();
|
||||||
|
}
|
||||||
|
|
||||||
|
void UICanvas::flushPass() {
|
||||||
|
if(quadCount == 0) return;
|
||||||
|
|
||||||
auto pass = createRenderPass<UIShader, UIShaderData>(
|
auto pass = createRenderPass<UIShader, UIShaderData>(
|
||||||
std::ref(*this),
|
std::ref(*this),
|
||||||
@ -73,6 +110,8 @@ std::vector<std::shared_ptr<IRenderPass>> UICanvas::getPasses(
|
|||||||
);
|
);
|
||||||
passes.push_back(pass);
|
passes.push_back(pass);
|
||||||
|
|
||||||
|
quadCount = 0;
|
||||||
return passes;
|
nextBinding = 0;
|
||||||
|
textures.clear();
|
||||||
|
textureBindings.clear();
|
||||||
}
|
}
|
@ -6,25 +6,47 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include "scene/SceneItem.hpp"
|
#include "scene/SceneItem.hpp"
|
||||||
#include "component/display/IRenderableComponent.hpp"
|
#include "component/display/IRenderableComponent.hpp"
|
||||||
#include "ui/UIComponent.hpp"
|
#include "display/shader/UIShader.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
|
class UIComponent;
|
||||||
|
|
||||||
class UICanvas :
|
class UICanvas :
|
||||||
public SceneComponent,
|
public SceneComponent,
|
||||||
public IRenderableComponent
|
public IRenderableComponent
|
||||||
{
|
{
|
||||||
protected:
|
private:
|
||||||
std::shared_ptr<Mesh> mesh;
|
std::shared_ptr<Mesh> mesh;
|
||||||
|
UIShaderData data;
|
||||||
|
|
||||||
|
size_t quadCount = 0;
|
||||||
|
shadertexturebinding_t nextBinding = 0;
|
||||||
|
std::unordered_map<
|
||||||
|
shadertexturebinding_t, std::shared_ptr<Texture>
|
||||||
|
> textures;
|
||||||
|
std::map<
|
||||||
|
std::shared_ptr<Texture>, shadertexturebinding_t
|
||||||
|
> textureBindings;
|
||||||
|
|
||||||
|
std::vector<std::shared_ptr<IRenderPass>> passes;
|
||||||
|
|
||||||
|
protected:
|
||||||
virtual void onInit() override;
|
virtual void onInit() override;
|
||||||
virtual void onDispose() override;
|
virtual void onDispose() override;
|
||||||
|
void flushPass();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
std::vector<std::shared_ptr<UIComponent>> components;
|
std::vector<std::shared_ptr<UIComponent>> components;
|
||||||
|
|
||||||
|
|
||||||
std::vector<std::shared_ptr<IRenderPass>> getPasses(
|
std::vector<std::shared_ptr<IRenderPass>> getPasses(
|
||||||
struct RenderPassContext &ctx
|
struct RenderPassContext &ctx
|
||||||
) override;
|
) override;
|
||||||
|
|
||||||
|
void addQuad(
|
||||||
|
const glm::vec4 quad,
|
||||||
|
const glm::vec4 uvs,
|
||||||
|
const struct Color color,
|
||||||
|
const std::shared_ptr<Texture> texture
|
||||||
|
);
|
||||||
};
|
};
|
||||||
}
|
}
|
@ -11,15 +11,12 @@ std::vector<std::shared_ptr<UIComponent>> UIComponent::getChildren() {
|
|||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<struct UIShaderQuad> UIComponent::getQuads(const glm::vec2 parent) {
|
void UIComponent::getQuads(const glm::vec2 parent, UICanvas &ctx) {
|
||||||
glm::vec2 transform = parent + position;
|
glm::vec2 transform = parent + position;
|
||||||
std::vector<struct UIShaderQuad> quads = this->getSelfQuads(transform);
|
this->getSelfQuads(transform, ctx);
|
||||||
|
|
||||||
auto children = getChildren();
|
auto children = getChildren();
|
||||||
for(auto &c : children) {
|
for(auto &c : children) {
|
||||||
auto childQuads = c->getQuads(transform);
|
c->getQuads(transform, ctx);
|
||||||
quads.insert(quads.end(), childQuads.begin(), childQuads.end());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return quads;
|
|
||||||
}
|
}
|
@ -6,19 +6,14 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include "ui/UIAlign.hpp"
|
#include "ui/UIAlign.hpp"
|
||||||
#include "display/shader/UIShader.hpp"
|
#include "display/shader/UIShader.hpp"
|
||||||
|
#include "component/ui/UICanvas.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
class UICanvas;
|
|
||||||
|
|
||||||
class UIComponent {
|
class UIComponent {
|
||||||
protected:
|
protected:
|
||||||
virtual std::vector<struct UIShaderQuad> getSelfQuads(
|
virtual void getSelfQuads(const glm::vec2 t, UICanvas &ctx) = 0;
|
||||||
const glm::vec2 t
|
|
||||||
) = 0;
|
|
||||||
|
|
||||||
virtual std::vector<std::shared_ptr<UIComponent>> getChildren();
|
virtual std::vector<std::shared_ptr<UIComponent>> getChildren();
|
||||||
|
void getQuads(const glm::vec2 parent, UICanvas &ctx);
|
||||||
std::vector<struct UIShaderQuad> getQuads(const glm::vec2 parent);
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
glm::vec2 position;
|
glm::vec2 position;
|
||||||
|
@ -7,30 +7,29 @@
|
|||||||
|
|
||||||
using namespace Dawn;
|
using namespace Dawn;
|
||||||
|
|
||||||
std::vector<struct UIShaderQuad> UILabel::getSelfQuads(const glm::vec2 t) {
|
void UILabel::getSelfQuads(const glm::vec2 t, UICanvas &ctx) {
|
||||||
std::vector<struct UIShaderQuad> quads;
|
std::vector<struct UIShaderQuad> quads;
|
||||||
if(this->texture == nullptr) return quads;
|
if(this->texture == nullptr) return;
|
||||||
|
|
||||||
const std::wstring text = L"Hello World!";
|
const std::wstring text = L"He";
|
||||||
glm::vec2 position = t;
|
glm::vec2 position = t;
|
||||||
glm::vec4 quad;
|
glm::vec4 quad;
|
||||||
|
|
||||||
for(wchar_t c : text) {
|
for(wchar_t c : text) {
|
||||||
auto info = texture->getCharacterData(c);
|
auto info = texture->getCharacterData(c);
|
||||||
quads.push_back({
|
ctx.addQuad(
|
||||||
.quad = {
|
{
|
||||||
position.x,
|
position.x,
|
||||||
position.y,
|
position.y,
|
||||||
position.x + info.size.x,
|
position.x + info.size.x,
|
||||||
position.y + info.size.y
|
position.y + info.size.y
|
||||||
},
|
},
|
||||||
.uv = info.quad,
|
info.quad,
|
||||||
.color = COLOR_WHITE
|
COLOR_WHITE,
|
||||||
});
|
texture->texture
|
||||||
|
);
|
||||||
position += info.advance;
|
position += info.advance;
|
||||||
}
|
}
|
||||||
|
|
||||||
return quads;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void UILabel::setFont(std::shared_ptr<TrueTypeTexture> texture) {
|
void UILabel::setFont(std::shared_ptr<TrueTypeTexture> texture) {
|
||||||
|
@ -13,7 +13,7 @@ namespace Dawn {
|
|||||||
std::shared_ptr<TrueTypeTexture> texture = nullptr;
|
std::shared_ptr<TrueTypeTexture> texture = nullptr;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
std::vector<struct UIShaderQuad> getSelfQuads(const glm::vec2 t) override;
|
void getSelfQuads(const glm::vec2 t, UICanvas &ctx) override;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void setFont(std::shared_ptr<TrueTypeTexture> texture);
|
void setFont(std::shared_ptr<TrueTypeTexture> texture);
|
||||||
|
@ -7,12 +7,12 @@
|
|||||||
|
|
||||||
using namespace Dawn;
|
using namespace Dawn;
|
||||||
|
|
||||||
std::vector<struct UIShaderQuad> UIRectangle::getSelfQuads(const glm::vec2 t) {
|
void UIRectangle::getSelfQuads(const glm::vec2 t, UICanvas &ctx) {
|
||||||
std::vector<struct UIShaderQuad> quads;
|
std::vector<struct UIShaderQuad> quads;
|
||||||
quads.push_back({
|
ctx.addQuad(
|
||||||
glm::vec4(t, t + size),
|
glm::vec4(t, t + size),
|
||||||
uv,
|
uv,
|
||||||
color
|
color,
|
||||||
});
|
nullptr
|
||||||
return quads;
|
);
|
||||||
}
|
}
|
@ -9,7 +9,7 @@
|
|||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
class UIRectangle final : public UIComponent {
|
class UIRectangle final : public UIComponent {
|
||||||
protected:
|
protected:
|
||||||
std::vector<struct UIShaderQuad> getSelfQuads(const glm::vec2 t) override;
|
void getSelfQuads(const glm::vec2 t, UICanvas &ctx) override;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
struct Color color = COLOR_WHITE;
|
struct Color color = COLOR_WHITE;
|
||||||
|
@ -52,10 +52,10 @@ void Dawn::helloWorldScene(Scene &s) {
|
|||||||
auto uiCanvasItem = s.createSceneItem();
|
auto uiCanvasItem = s.createSceneItem();
|
||||||
auto uiCanvas = uiCanvasItem->addComponent<UICanvas>();
|
auto uiCanvas = uiCanvasItem->addComponent<UICanvas>();
|
||||||
|
|
||||||
auto rect = std::make_shared<UIRectangle>();
|
// auto rect = std::make_shared<UIRectangle>();
|
||||||
rect->position = { -32, -32 };
|
// rect->position = { -32, -32 };
|
||||||
rect->color = COLOR_MAGENTA;
|
// rect->color = COLOR_MAGENTA;
|
||||||
uiCanvas->components.push_back(rect);
|
// uiCanvas->components.push_back(rect);
|
||||||
|
|
||||||
auto label = std::make_shared<UILabel>();
|
auto label = std::make_shared<UILabel>();
|
||||||
label->setFont(texture);
|
label->setFont(texture);
|
||||||
|
@ -57,7 +57,7 @@ ShaderStage::ShaderStage(
|
|||||||
GLchar *log = new GLchar[logLength];
|
GLchar *log = new GLchar[logLength];
|
||||||
glGetShaderInfoLog(this->id, logLength, NULL, log);
|
glGetShaderInfoLog(this->id, logLength, NULL, log);
|
||||||
assertNoGLError();
|
assertNoGLError();
|
||||||
assertUnreachable("Failed to compile shader stage:\n%s", log);
|
assertUnreachable("Failed to compile shader stage %i:\n%s", type, log);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,19 +33,21 @@ void UIShader::getStages(
|
|||||||
"vec4 quad;\n"
|
"vec4 quad;\n"
|
||||||
"vec4 uv;\n"
|
"vec4 uv;\n"
|
||||||
"vec4 color;\n"
|
"vec4 color;\n"
|
||||||
|
"float texture;\n"
|
||||||
"};\n"
|
"};\n"
|
||||||
"layout (std140) uniform ub_Quad {\n"
|
"layout (std140) uniform ub_Quad {\n"
|
||||||
"UIShaderQuad quads[" MACRO_STRINGIFY(UI_SHADER_QUAD_COUNT) "];\n"
|
"UIShaderQuad quads[" MACRO_STRINGIFY(UI_SHADER_QUAD_COUNT) "];\n"
|
||||||
"};\n"
|
"};\n"
|
||||||
"out vec2 o_TextCoord;\n"
|
"out vec2 v_TextCoord;\n"
|
||||||
"out vec4 v_Color;\n"
|
"out vec4 v_Color;\n"
|
||||||
|
"out float v_TextureIndex;\n"
|
||||||
"void main() {\n"
|
"void main() {\n"
|
||||||
"vec4 pos;\n"
|
"vec4 pos;\n"
|
||||||
"vec2 coord;\n"
|
"vec2 coord;\n"
|
||||||
"int index = int(aPos.z);\n"
|
"int index = int(aPos.z);\n"
|
||||||
"int quadIndex = index / 4;\n"
|
"int quadIndex = index / 4;\n"
|
||||||
"int vertexIndex = index % 4;\n"
|
"int vertexIndex = index % 4;\n"
|
||||||
"struct UIShaderQuad quad = quads[quadIndex];\n"
|
"UIShaderQuad quad = quads[quadIndex];\n"
|
||||||
"if(vertexIndex == 0) {\n"
|
"if(vertexIndex == 0) {\n"
|
||||||
"pos.x = quad.quad.x;\n"
|
"pos.x = quad.quad.x;\n"
|
||||||
"pos.y = quad.quad.y;\n"
|
"pos.y = quad.quad.y;\n"
|
||||||
@ -70,19 +72,44 @@ void UIShader::getStages(
|
|||||||
"pos.z = 0;\n"
|
"pos.z = 0;\n"
|
||||||
"pos.w = 1;\n"
|
"pos.w = 1;\n"
|
||||||
"gl_Position = u_Projection * u_View * u_Model * pos;\n"
|
"gl_Position = u_Projection * u_View * u_Model * pos;\n"
|
||||||
"o_TextCoord = coord;\n"
|
"v_TextCoord = coord;\n"
|
||||||
"v_Color = quad.color;\n"
|
"v_Color = quad.color;\n"
|
||||||
|
"v_TextureIndex = quad.texture;\n"
|
||||||
"}"
|
"}"
|
||||||
);
|
);
|
||||||
|
|
||||||
fragment = std::make_shared<ShaderStage>(
|
fragment = std::make_shared<ShaderStage>(
|
||||||
ShaderStageType::FRAGMENT,
|
ShaderStageType::FRAGMENT,
|
||||||
"#version 330 core\n"
|
"#version 330 core\n"
|
||||||
"in vec2 o_TextCoord;\n"
|
"in vec2 v_TextCoord;\n"
|
||||||
"in vec4 v_Color;\n"
|
"in vec4 v_Color;\n"
|
||||||
|
"in float v_TextureIndex;\n"
|
||||||
|
"uniform sampler2D u_Texture[" MACRO_STRINGIFY(UI_SHADER_TEXTURE_COUNT) "];\n"
|
||||||
"out vec4 o_Color;\n"
|
"out vec4 o_Color;\n"
|
||||||
"void main() {\n"
|
"void main() {\n"
|
||||||
"o_Color = v_Color;\n"
|
"vec4 texColor = vec4(1, 1, 1, 1);\n"
|
||||||
|
// "switch(int(v_TextureIndex)) {\n"
|
||||||
|
// "case 0:\n"
|
||||||
|
// "texColor = texture(u_Texture[0], v_TextCoord);\n"
|
||||||
|
// "break;\n"
|
||||||
|
// "case 1:\n"
|
||||||
|
// "texColor = texture(u_Texture[1], v_TextCoord);\n"
|
||||||
|
// "break;\n"
|
||||||
|
// "case 2:\n"
|
||||||
|
// "texColor = texture(u_Texture[2], v_TextCoord);\n"
|
||||||
|
// "break;\n"
|
||||||
|
// "case 3:\n"
|
||||||
|
// "texColor = texture(u_Texture[3], v_TextCoord);\n"
|
||||||
|
// "break;\n"
|
||||||
|
// "case 4:\n"
|
||||||
|
// "texColor = texture(u_Texture[4], v_TextCoord);\n"
|
||||||
|
// "break;\n"
|
||||||
|
// "case 5:\n"
|
||||||
|
// "texColor = texture(u_Texture[5], v_TextCoord);\n"
|
||||||
|
// "break;\n"
|
||||||
|
// "}\n"
|
||||||
|
"texColor = texture(u_Texture[0], v_TextCoord);\n"
|
||||||
|
"o_Color = texColor * v_Color;\n"
|
||||||
"}\n"
|
"}\n"
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
@ -114,6 +141,13 @@ void UIShader::getStages(
|
|||||||
ShaderParameterType::MAT4
|
ShaderParameterType::MAT4
|
||||||
));
|
));
|
||||||
|
|
||||||
|
parameters.push_back(ShaderParameter(
|
||||||
|
"u_Texture",
|
||||||
|
&rel->textures,
|
||||||
|
ShaderParameterType::TEXTURE,
|
||||||
|
UI_SHADER_TEXTURE_COUNT
|
||||||
|
));
|
||||||
|
|
||||||
structures.push_back(ShaderStructure<struct UIShaderQuad>(
|
structures.push_back(ShaderStructure<struct UIShaderQuad>(
|
||||||
"ub_Quad",
|
"ub_Quad",
|
||||||
&rel->quads,
|
&rel->quads,
|
||||||
@ -136,6 +170,12 @@ void UIShader::getStages(
|
|||||||
&rel.color,
|
&rel.color,
|
||||||
ShaderParameterType::COLOR
|
ShaderParameterType::COLOR
|
||||||
));
|
));
|
||||||
|
|
||||||
|
parameters.push_back(ShaderParameter(
|
||||||
|
"u_Texture",
|
||||||
|
&rel.texture,
|
||||||
|
ShaderParameterType::TEXTURE
|
||||||
|
));
|
||||||
},
|
},
|
||||||
UI_SHADER_QUAD_COUNT
|
UI_SHADER_QUAD_COUNT
|
||||||
));
|
));
|
||||||
|
@ -8,17 +8,20 @@
|
|||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
#define UI_SHADER_QUAD_COUNT 32
|
#define UI_SHADER_QUAD_COUNT 32
|
||||||
|
#define UI_SHADER_TEXTURE_COUNT 6
|
||||||
|
|
||||||
struct UIShaderQuad {
|
struct UIShaderQuad {
|
||||||
glm::vec4 quad;
|
glm::vec4 quad;
|
||||||
glm::vec4 uv;
|
glm::vec4 uv;
|
||||||
struct Color color;
|
struct Color color;
|
||||||
|
float_t texture;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct UIShaderData {
|
struct UIShaderData {
|
||||||
glm::mat4 projection;
|
glm::mat4 projection;
|
||||||
glm::mat4 view;
|
glm::mat4 view;
|
||||||
glm::mat4 model;
|
glm::mat4 model;
|
||||||
|
shadertexturebinding_t textures[UI_SHADER_TEXTURE_COUNT];
|
||||||
struct UIShaderQuad quads[UI_SHADER_QUAD_COUNT];
|
struct UIShaderQuad quads[UI_SHADER_QUAD_COUNT];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user