Basic UI Alignment controls examples

This commit is contained in:
2023-12-21 11:21:16 -06:00
parent 457d740c73
commit 92f8b17db6
19 changed files with 497 additions and 101 deletions

View File

@ -6,7 +6,7 @@
#include "UICanvas.hpp" #include "UICanvas.hpp"
#include "display/pass/RenderPass.hpp" #include "display/pass/RenderPass.hpp"
#include "display/mesh/QuadMesh.hpp" #include "display/mesh/QuadMesh.hpp"
#include "ui/UIComponent.hpp" #include "ui/UIElement.hpp"
using namespace Dawn; using namespace Dawn;
@ -36,15 +36,12 @@ 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
) { ) {
if(this->components.empty()) return {}; if(this->elements.empty()) return {};
glm::mat4 projection; glm::mat4 projection;
glm::mat4 view; glm::mat4 view;
// data.projection = ctx.camera->getProjection(); // Setup the projection and views
// data.view = ctx.camera->getItem()->getWorldTransform();
// data.model = this->getItem()->getWorldTransform();
data.projection = glm::ortho( data.projection = glm::ortho(
0.0f, ctx.renderTarget->getWidth(), 0.0f, ctx.renderTarget->getWidth(),
ctx.renderTarget->getHeight(), 0.0f, ctx.renderTarget->getHeight(), 0.0f,
@ -53,24 +50,27 @@ struct RenderPassContext &ctx
data.view = glm::mat4(1.0f); data.view = glm::mat4(1.0f);
data.model = glm::mat4(1.0f); data.model = glm::mat4(1.0f);
// Reset the passes
this->passes.clear(); this->passes.clear();
this->textureBindings.clear(); this->textureBindings.clear();
this->textures.clear(); this->textures.clear();
quadCount = 0; quadCount = 0;
nextBinding = 0; nextBinding = 0;
// Define the root alignment // Alignment root
const glm::vec2 rootPosition = { 0, 0 };
const glm::vec2 rootSize = {
ctx.renderTarget->getWidth(),
ctx.renderTarget->getHeight()
};
const float_t rootScale = 1.0f;
// Get the quads for each component // Get the quads for each component
auto itComponents = components.begin(); auto itComponents = elements.begin();
auto self = std::ref(*this); auto self = std::ref(*this);
while(itComponents != components.end()) { while(itComponents != elements.end()) {
auto component = *itComponents; auto component = *itComponents;
component->updateAlignment( component->updateAlignment(rootPosition, rootSize, rootScale);
glm::vec2(0, 0),
glm::vec2(ctx.renderTarget->getWidth(), ctx.renderTarget->getHeight()),
1.0f
);
component->getQuads(self); component->getQuads(self);
++itComponents; ++itComponents;
} }
@ -139,6 +139,6 @@ void UICanvas::flushPass() {
textureBindings.clear(); textureBindings.clear();
} }
void UICanvas::addComponent(std::shared_ptr<UIComponent> component) { void UICanvas::addElement(std::shared_ptr<UIElement> element) {
components.push_back(component); elements.push_back(element);
} }

View File

@ -9,7 +9,7 @@
#include "display/shader/UIShader.hpp" #include "display/shader/UIShader.hpp"
namespace Dawn { namespace Dawn {
class UIComponent; class UIElement;
class UICanvas : class UICanvas :
public SceneComponent, public SceneComponent,
@ -18,7 +18,7 @@ namespace Dawn {
private: private:
std::shared_ptr<Mesh> mesh; std::shared_ptr<Mesh> mesh;
UIShaderData data; UIShaderData data;
std::vector<std::shared_ptr<UIComponent>> components; std::vector<std::shared_ptr<UIElement>> elements;
size_t quadCount = 0; size_t quadCount = 0;
shadertexturebinding_t nextBinding = 0; shadertexturebinding_t nextBinding = 0;
@ -68,6 +68,6 @@ namespace Dawn {
* *
* @param component The component to add. * @param component The component to add.
*/ */
void addComponent(std::shared_ptr<UIComponent> component); void addElement(std::shared_ptr<UIElement> component);
}; };
} }

View File

@ -5,7 +5,12 @@
target_sources(${DAWN_TARGET_NAME} target_sources(${DAWN_TARGET_NAME}
PRIVATE PRIVATE
UIComponent.cpp UIAlignableElement.cpp
UIElement.cpp
UIRectangle.cpp UIRectangle.cpp
UILabel.cpp UILabel.cpp
) UIMenu.cpp
)
# Subdirs
add_subdirectory(container)

View File

@ -3,12 +3,11 @@
// This software is released under the MIT License. // This software is released under the MIT License.
// https://opensource.org/licenses/MIT // https://opensource.org/licenses/MIT
#include "UIComponent.hpp" #include "UIAlignableElement.hpp"
#include "assert/assert.hpp"
using namespace Dawn; using namespace Dawn;
UIComponent::UIComponent() { UIAlignableElement::UIAlignableElement() {
alignUnit[0] = UIAlignmentUnit::SCALE; alignUnit[0] = UIAlignmentUnit::SCALE;
alignUnit[1] = UIAlignmentUnit::SCALE; alignUnit[1] = UIAlignmentUnit::SCALE;
alignUnit[2] = UIAlignmentUnit::SCALE; alignUnit[2] = UIAlignmentUnit::SCALE;
@ -18,7 +17,7 @@ UIComponent::UIComponent() {
}; };
} }
void UIComponent::updateAlignment( void UIAlignableElement::updateSelfAlignment(
const glm::vec2 pPos, const glm::vec2 pPos,
const glm::vec2 pSize, const glm::vec2 pSize,
const float_t canvasScale const float_t canvasScale
@ -180,18 +179,20 @@ void UIComponent::updateAlignment(
); );
} }
this->position += pPos;
this->eventAlignmentUpdated(position, size); this->eventAlignmentUpdated(position, size);
} }
std::vector<std::shared_ptr<UIComponent>> UIComponent::getChildren() { void UIAlignableElement::updateAlignment(
return {}; const glm::vec2 pPos,
} const glm::vec2 pSize,
const float_t canvasScale
void UIComponent::getQuads(UICanvas &ctx) { ) {
this->getSelfQuads(ctx); this->updateSelfAlignment(pPos, pSize, canvasScale);
// Now update children alignment
auto children = getChildren(); auto children = getChildren();
for(auto &c : children) { for(auto &c : children) {
c->getQuads(ctx); c->updateAlignment(this->position, this->size, canvasScale);
} }
} }

View File

@ -0,0 +1,64 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "ui/UIElement.hpp"
namespace Dawn {
enum class UIAlignmentType {
START,
MIDDLE,
END,
STRETCH
};
enum class UIAlignmentUnit {
PIXEL,
SCALE,
PERCENT,
RATIO
};
class UIAlignableElement : public UIElement {
protected:
glm::vec2 position;
glm::vec2 size;
/**
* Updates the alignment of this element ONLY.
*
* @param parentPosition The position of the parent.
* @param parentSize The size of the parent.
* @param canvasScale The scale of the canvas.
*/
virtual void updateSelfAlignment(
const glm::vec2 parentPosition,
const glm::vec2 parentSize,
const float_t canvasScale
);
public:
// Primary alignment controls
glm::vec4 align = glm::vec4(0, 0, 0, 0);
enum UIAlignmentType alignX = UIAlignmentType::STRETCH;
enum UIAlignmentType alignY = UIAlignmentType::STRETCH;
enum UIAlignmentUnit alignUnit[4];
std::function<
void(const glm::vec2, const glm::vec2)
> eventAlignmentUpdated;
/**
* Constructor for the UIAlignableElement.
*/
UIAlignableElement();
void updateAlignment(
const glm::vec2 parentPosition,
const glm::vec2 parentSize,
const float_t canvasScale
) override;
};
}

37
src/dawn/ui/UIElement.cpp Normal file
View File

@ -0,0 +1,37 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "UIElement.hpp"
#include "assert/assert.hpp"
using namespace Dawn;
std::vector<std::shared_ptr<UIElement>> UIElement::getChildren() {
return {};
}
void UIElement::getSelfQuads(UICanvas &ctx) {
//Do nothing
}
void UIElement::getQuads(UICanvas &ctx) {
this->getSelfQuads(ctx);
auto children = getChildren();
for(auto &c : children) {
c->getQuads(ctx);
}
}
void UIElement::updateAlignment(
const glm::vec2 parentPosition,
const glm::vec2 parentSize,
const float_t canvasScale
) {
auto children = getChildren();
for(auto &c : children) {
c->updateAlignment(parentPosition, parentSize, canvasScale);
}
}

View File

@ -8,66 +8,23 @@
#include "component/ui/UICanvas.hpp" #include "component/ui/UICanvas.hpp"
namespace Dawn { namespace Dawn {
enum class UIAlignmentType { class UIElement {
START,
MIDDLE,
END,
STRETCH
};
enum class UIAlignmentUnit {
PIXEL,
SCALE,
PERCENT,
RATIO
};
class UIComponent {
protected: protected:
glm::vec2 position;
glm::vec2 size;
/** /**
* Virtual method overridden by the UIComponent to get the quads for the * Virtual method overridden by the UIElement to get the quads for the
* component. * component.
* *
* @param alignment The alignment of this component. * @param alignment The alignment of this component.
* @param ctx The canvas to add the quads to. * @param ctx The canvas to add the quads to.
*/ */
virtual void getSelfQuads(UICanvas &ctx) = 0; virtual void getSelfQuads(UICanvas &ctx);
/**
* Updates the alignment of this component based on the parent.
*
* @param parentPosition The position of the parent.
* @param parentSize The size of the parent.
*/
void updateAlignment(
const glm::vec2 parentPosition,
const glm::vec2 parentSize,
const float_t canvasScale
);
public: public:
glm::vec4 align = glm::vec4(0, 0, 32, 32);
enum UIAlignmentType alignX = UIAlignmentType::START;
enum UIAlignmentType alignY = UIAlignmentType::START;
enum UIAlignmentUnit alignUnit[4];
std::function<
void(const glm::vec2, const glm::vec2)
> eventAlignmentUpdated;
/** /**
* Instantiates a new UIComponent. * Virtual method overridden by the UIElement to get the children of
*/
UIComponent();
/**
* Virtual method overridden by the UIComponent to get the children of
* this component. * this component.
*/ */
virtual std::vector<std::shared_ptr<UIComponent>> getChildren(); virtual std::vector<std::shared_ptr<UIElement>> getChildren();
/** /**
* Method called by the UICanvas to get the quads for this component. * Method called by the UICanvas to get the quads for this component.
@ -76,6 +33,18 @@ namespace Dawn {
*/ */
void getQuads(UICanvas &ctx); void getQuads(UICanvas &ctx);
friend class UICanvas; /**
* Updates the alignment of this component based on the parent. Typically
* left to the UIAlignableElement to implement, default implementation
* does nothing but invoke children.
*
* @param parentPosition The position of the parent.
* @param parentSize The size of the parent.
*/
virtual void updateAlignment(
const glm::vec2 parentPosition,
const glm::vec2 parentSize,
const float_t canvasScale
);
}; };
} }

View File

@ -75,7 +75,7 @@ void UILabel::getSelfQuads(UICanvas &ctx) {
info.quad.z, info.quad.z,
info.quad.w info.quad.w
}, },
COLOR_WHITE, this->color,
UIShaderQuadStyle::FONT, UIShaderQuadStyle::FONT,
texture->texture texture->texture
); );

View File

@ -4,11 +4,11 @@
// https://opensource.org/licenses/MIT // https://opensource.org/licenses/MIT
#pragma once #pragma once
#include "ui/UIComponent.hpp" #include "ui/UIAlignableElement.hpp"
#include "display/font/TrueTypeTexture.hpp" #include "display/font/TrueTypeTexture.hpp"
namespace Dawn { namespace Dawn {
class UILabel final : public UIComponent { class UILabel final : public UIAlignableElement {
private: private:
std::shared_ptr<TrueTypeTexture> texture = nullptr; std::shared_ptr<TrueTypeTexture> texture = nullptr;
std::wstring text = L"Hello World"; std::wstring text = L"Hello World";
@ -18,6 +18,7 @@ namespace Dawn {
public: public:
bool_t wordWrap = true; bool_t wordWrap = true;
struct Color color = COLOR_WHITE;
/** /**
* Returns the font used for this label. * Returns the font used for this label.

54
src/dawn/ui/UIMenu.cpp Normal file
View File

@ -0,0 +1,54 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "UIMenu.hpp"
using namespace Dawn;
std::vector<std::shared_ptr<UIElement>> UIMenu::getChildren() {
return children;
}
void UIMenu::setPosition(int32_t x, int32_t y) {
assertTrue(x >= 0, "X position must be greater than or equal to 0.");
assertTrue(y >= 0, "Y position must be greater than or equal to 0.");
assertTrue(x < columns, "X must be less than the number of columns.");
assertTrue(y < rows, "Y must be less than the number of rows.");
if(this->x == x && this->y == y) return;
this->x = x;
this->y = y;
eventPositionChanged.emit(x, y);
}
void UIMenu::setSize(int32_t columns, int32_t rows) {
assertTrue(columns > 0, "Columns must be greater than 0.");
assertTrue(rows > 0, "Rows must be greater than 0.");
assertTrue(columns > x, "Columns must be greater than current x position.");
assertTrue(rows > y, "Rows must be greater than current y position.");
if(this->columns == columns && this->rows == rows) return;
this->columns = columns;
this->rows = rows;
}
int32_t UIMenu::getX() {
return x;
}
int32_t UIMenu::getY() {
return y;
}
int32_t UIMenu::getColumns() {
return columns;
}
int32_t UIMenu::getRows() {
return rows;
}

67
src/dawn/ui/UIMenu.hpp Normal file
View File

@ -0,0 +1,67 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "ui/UIElement.hpp"
namespace Dawn {
class UIMenu final : public UIElement {
private:
int32_t x = 0;
int32_t y = 0;
int32_t columns = 1;
int32_t rows = 1;
public:
Event<int32_t, int32_t> eventPositionChanged;
std::vector<std::shared_ptr<UIElement>> children;
std::vector<std::shared_ptr<UIElement>> getChildren() override;
/**
* Sets the position of this menu.
*
* @param x The x position of this menu.
* @param y The y position of this menu.
*/
void setPosition(int32_t x, int32_t y);
/**
* Sets the size of this menu.
*
* @param columns The number of columns in this menu.
* @param rows The number of rows in this menu.
*/
void setSize(int32_t columns, int32_t rows);
/**
* Gets the x position of this menu.
*
* @return The x position of this menu.
*/
int32_t getX();
/**
* Gets the y position of this menu.
*
* @return The y position of this menu.
*/
int32_t getY();
/**
* Gets the number of columns in this menu.
*
* @return The number of columns in this menu.
*/
int32_t getColumns();
/**
* Gets the number of rows in this menu.
*
* @return The number of rows in this menu.
*/
int32_t getRows();
};
}

View File

@ -4,10 +4,10 @@
// https://opensource.org/licenses/MIT // https://opensource.org/licenses/MIT
#pragma once #pragma once
#include "ui/UIComponent.hpp" #include "ui/UIAlignableElement.hpp"
namespace Dawn { namespace Dawn {
class UIRectangle final : public UIComponent { class UIRectangle final : public UIAlignableElement {
protected: protected:
void getSelfQuads(UICanvas &ctx) override; void getSelfQuads(UICanvas &ctx) override;

View File

@ -0,0 +1,10 @@
# Copyright (c) 2023 Dominic Masters
#
# This software is released under the MIT License.
# https://opensource.org/licenses/MIT
target_sources(${DAWN_TARGET_NAME}
PRIVATE
UIGroupContainer.cpp
UIRowContainer.cpp
)

View File

@ -4,10 +4,11 @@
// https://opensource.org/licenses/MIT // https://opensource.org/licenses/MIT
#pragma once #pragma once
#include "dawnlibs.hpp" #include "ui/UIAlignableElement.hpp"
namespace Dawn { namespace Dawn {
struct UIAlign { class UIContainer : public UIAlignableElement {
glm::vec2 position; public:
}; };
} }

View File

@ -0,0 +1,28 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "UIGroupContainer.hpp"
using namespace Dawn;
std::vector<std::shared_ptr<UIElement>> UIGroupContainer::getChildren() {
return this->children;
}
void UIGroupContainer::appendChild(std::shared_ptr<UIElement> child) {
assertNotNull(child, "Cannot append a null child!");
this->children.push_back(child);
}
void UIGroupContainer::removeChild(std::shared_ptr<UIElement> child) {
assertNotNull(child, "Cannot remove a null child!");
auto it = std::find(this->children.begin(), this->children.end(), child);
if(it == this->children.end()) return;
this->children.erase(it);
}
void UIGroupContainer::clearChildren() {
this->children.clear();
}

View File

@ -0,0 +1,36 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "ui/container/UIContainer.hpp"
namespace Dawn {
class UIGroupContainer : public UIContainer {
private:
std::vector<std::shared_ptr<UIElement>> children;
public:
std::vector<std::shared_ptr<UIElement>> getChildren() override;
/**
* Appends a child to this container.
*
* @param child Child to append.
*/
void appendChild(std::shared_ptr<UIElement> child);
/**
* Removes a child from this container.
*
* @param child Child to remove.
*/
void removeChild(std::shared_ptr<UIElement> child);
/**
* Removes all children from this container.
*/
void clearChildren();
};
}

View File

@ -0,0 +1,56 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "assert/assert.hpp"
#include "UIRowContainer.hpp"
using namespace Dawn;
std::vector<std::shared_ptr<UIElement>> UIRowContainer::getChildren() {
return this->children;
}
void UIRowContainer::appendChild(std::shared_ptr<UIElement> child) {
assertNotNull(child, "Cannot append a null child!");
this->children.push_back(child);
}
void UIRowContainer::removeChild(std::shared_ptr<UIElement> child) {
assertNotNull(child, "Cannot remove a null child!");
auto it = std::find(this->children.begin(), this->children.end(), child);
if(it == this->children.end()) return;
this->children.erase(it);
}
void UIRowContainer::clearChildren() {
this->children.clear();
}
void UIRowContainer::updateAlignment(
const glm::vec2 parentPosition,
const glm::vec2 parentSize,
const float_t canvasScale
) {
this->updateSelfAlignment(parentPosition, parentSize, canvasScale);
// Now we have our dimensions, divide evenly
auto children = this->getChildren();
float_t y = 0.0f;
float_t yPiece = this->size.y / (float_t)children.size();
// Update all children
for(auto &child : this->children) {
child->updateAlignment(
this->position + glm::vec2(0, y),
glm::vec2(
this->size.x,
yPiece
),
canvasScale
);
y += yPiece;
}
}

View File

@ -0,0 +1,42 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "ui/container/UIContainer.hpp"
namespace Dawn {
class UIRowContainer final : public UIContainer {
private:
std::vector<std::shared_ptr<UIElement>> children;
public:
std::vector<std::shared_ptr<UIElement>> getChildren() override;
/**
* Appends a child to this container.
*
* @param child Child to append.
*/
void appendChild(std::shared_ptr<UIElement> child);
/**
* Removes a child from this container.
*
* @param child Child to remove.
*/
void removeChild(std::shared_ptr<UIElement> child);
/**
* Removes all children from this container.
*/
void clearChildren();
void updateAlignment(
const glm::vec2 parentPosition,
const glm::vec2 parentSize,
const float_t canvasScale
) override;
};
}

View File

@ -14,6 +14,9 @@
#include "component/ui/UICanvas.hpp" #include "component/ui/UICanvas.hpp"
#include "ui/UIRectangle.hpp" #include "ui/UIRectangle.hpp"
#include "ui/UILabel.hpp" #include "ui/UILabel.hpp"
#include "ui/UIMenu.hpp"
#include "ui/container/UIRowContainer.hpp"
#include "ui/container/UIGroupContainer.hpp"
#include <ft2build.h> #include <ft2build.h>
#include FT_FREETYPE_H #include FT_FREETYPE_H
@ -55,18 +58,40 @@ void Dawn::helloWorldScene(Scene &s) {
auto uiCanvasItem = s.createSceneItem(); auto uiCanvasItem = s.createSceneItem();
auto uiCanvas = uiCanvasItem->addComponent<UICanvas>(); auto uiCanvas = uiCanvasItem->addComponent<UICanvas>();
auto container = std::make_shared<UIGroupContainer>();
container->align = { 32, 32, 300, 250 };
container->alignX = UIAlignmentType::START;
container->alignY = UIAlignmentType::START;
uiCanvas->addElement(container);
auto rect = std::make_shared<UIRectangle>(); auto rect = std::make_shared<UIRectangle>();
rect->color = COLOR_MAGENTA; rect->color = COLOR_MAGENTA;
rect->align = { 32, 32, 300, 250 }; container->appendChild(rect);
rect->alignX = UIAlignmentType::START;
rect->alignY = UIAlignmentType::START;
uiCanvas->addComponent(rect);
auto label = std::make_shared<UILabel>(); auto menu = std::make_shared<UIMenu>();
label->setText(L"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum."); menu->setSize(1, 4);
label->setFont(texture); container->appendChild(menu);
label->align = { 32, 32, 300, 250 };
label->alignX = UIAlignmentType::START; auto rowContainer = std::make_shared<UIRowContainer>();
label->alignY = UIAlignmentType::START; menu->children.push_back(rowContainer);
uiCanvas->addComponent(label);
auto label0 = std::make_shared<UILabel>();
label0->setText(L"New Game");
label0->setFont(texture);
rowContainer->appendChild(label0);
auto label1 = std::make_shared<UILabel>();
label1->setText(L"Load Game");
label1->setFont(texture);
rowContainer->appendChild(label1);
auto label2 = std::make_shared<UILabel>();
label2->setText(L"Options");
label2->setFont(texture);
rowContainer->appendChild(label2);
auto label3 = std::make_shared<UILabel>();
label3->setText(L"Exit");
label3->setFont(texture);
rowContainer->appendChild(label3);
} }