Totally refactored UILabel
This commit is contained in:
2
lib/SDL
2
lib/SDL
Submodule lib/SDL updated: c9aec268fa...87a83787a3
Submodule lib/openal-soft updated: fde74453a6...d66107e9f0
@ -132,7 +132,6 @@ void RenderPipeline::renderSceneCamera(Scene *scene, Camera *camera) {
|
|||||||
auto itCanvas = canvases.begin();
|
auto itCanvas = canvases.begin();
|
||||||
while(itCanvas != canvases.end()) {
|
while(itCanvas != canvases.end()) {
|
||||||
auto canvas = *itCanvas;
|
auto canvas = *itCanvas;
|
||||||
glm::mat4 model;
|
|
||||||
glm::mat4 projection;
|
glm::mat4 projection;
|
||||||
glm::mat4 view;
|
glm::mat4 view;
|
||||||
|
|
||||||
@ -140,32 +139,23 @@ void RenderPipeline::renderSceneCamera(Scene *scene, Camera *camera) {
|
|||||||
case UI_DRAW_TYPE_WORLD_ABSOLUTE:
|
case UI_DRAW_TYPE_WORLD_ABSOLUTE:
|
||||||
projection = camera->getProjection();
|
projection = camera->getProjection();
|
||||||
view = camera->transform->getWorldTransform();
|
view = camera->transform->getWorldTransform();
|
||||||
model = canvas->transform->getWorldTransform();
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE:
|
case UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE:
|
||||||
projection = glm::ortho(0.0f, renderTarget->getWidth(), renderTarget->getHeight(), 0.0f);
|
projection = glm::ortho(0.0f, renderTarget->getWidth(), renderTarget->getHeight(), 0.0f);
|
||||||
view = glm::mat4(1.0f);
|
view = glm::mat4(1.0f);
|
||||||
model = canvas->transform->getWorldTransform();
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
assertUnreachable();
|
assertUnreachable();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto renderables = canvas->item->findChildrenDeep<UIComponentRendaerable>();
|
auto renderables = canvas->item->findChildrenDeep<UIComponentRenderable>();
|
||||||
auto itChild = renderables.begin();
|
auto itChild = renderables.begin();
|
||||||
while(itChild != renderables.end()) {
|
while(itChild != renderables.end()) {
|
||||||
|
vectorAppend(&shaderPassItems,(*itChild)->getPassItems(projection, view));
|
||||||
++itChild;
|
++itChild;
|
||||||
}
|
}
|
||||||
|
|
||||||
// auto itChild = canvas->children.begin();
|
|
||||||
// while(itChild != canvas->children.end()) {
|
|
||||||
// vectorAppend(&shaderPassItems, (*itChild)->getPassItems(
|
|
||||||
// projection, view, model
|
|
||||||
// ));
|
|
||||||
// ++itChild;
|
|
||||||
// }
|
|
||||||
++itCanvas;
|
++itCanvas;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -115,17 +115,24 @@ namespace Dawn {
|
|||||||
return children;
|
return children;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Finds all children, and children of children, recursively that match
|
||||||
|
* the queried component.
|
||||||
|
*
|
||||||
|
* @tparam T Component Type to find.
|
||||||
|
* @return Array of pointers to matching children components.
|
||||||
|
*/
|
||||||
template<class T>
|
template<class T>
|
||||||
std::vector<T*> findChildrenDeep() {
|
std::vector<T*> findChildrenDeep() {
|
||||||
std::vector<Transform*> itemsToCheck = this->transform.children;
|
std::vector<Transform*> transformsToCheck = this->transform.children;
|
||||||
std::vector<T*> itemsFound;
|
std::vector<T*> itemsFound;
|
||||||
|
|
||||||
while(itemsToCheck.size() > 0) {
|
while(transformsToCheck.size() > 0) {
|
||||||
auto item = itemsToCheck.begin();
|
auto tras = transformsToCheck.begin();
|
||||||
vectorAppend(&itemsToCheck, (*item)->children);
|
vectorAppend(&transformsToCheck, (*tras)->children);
|
||||||
auto component = (*item)->item->getComponent<T>();
|
auto component = (*tras)->item->getComponent<T>();
|
||||||
if(component != nullptr) itemsFound.push_back(component);
|
if(component != nullptr) itemsFound.push_back(component);
|
||||||
itemsToCheck.erase(item);
|
transformsToCheck.erase(tras);
|
||||||
}
|
}
|
||||||
|
|
||||||
return itemsFound;
|
return itemsFound;
|
||||||
|
@ -18,7 +18,7 @@ Camera::Camera(SceneItem *item) :
|
|||||||
orthoBottom(0.0f),
|
orthoBottom(0.0f),
|
||||||
orthoTop(1.0f),
|
orthoTop(1.0f),
|
||||||
clipNear(0.001f),
|
clipNear(0.001f),
|
||||||
clipFar(50.0f)
|
clipFar(1000.0f)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
|
|
||||||
#include "UICanvas.hpp"
|
#include "UICanvas.hpp"
|
||||||
#include "game/DawnGame.hpp"
|
#include "game/DawnGame.hpp"
|
||||||
|
#include "UIComponent.hpp"
|
||||||
|
|
||||||
using namespace Dawn;
|
using namespace Dawn;
|
||||||
|
|
||||||
@ -13,10 +14,47 @@ UICanvas * UICanvas::create(Scene *scene) {
|
|||||||
return item->addComponent<UICanvas>();
|
return item->addComponent<UICanvas>();
|
||||||
}
|
}
|
||||||
|
|
||||||
UICanvas::UICanvas(SceneItem *item) : SceneItemComponent(item) {
|
UICanvas::UICanvas(SceneItem *item) :
|
||||||
|
SceneItemComponent(item),
|
||||||
|
camera(nullptr)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UICanvas::getWidth() {
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UICanvas::getHeight() {
|
||||||
|
return h;
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UICanvas::getContentWidth() {
|
||||||
|
return this->getWidth();
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UICanvas::getContentHeight() {
|
||||||
|
return this->getHeight();
|
||||||
}
|
}
|
||||||
|
|
||||||
void UICanvas::onStart() {
|
void UICanvas::onStart() {
|
||||||
|
if(camera == nullptr) camera = getScene()->findComponent<Camera>();
|
||||||
|
|
||||||
|
useEffectWithTeardown([&]{
|
||||||
|
if(camera == nullptr) return evtRenderResize = [&] {};
|
||||||
|
this->w = camera->getRenderTarget()->getWidth();
|
||||||
|
this->h = camera->getRenderTarget()->getHeight();
|
||||||
|
return evtRenderResize = useEvent([&](float_t w, float_t h){
|
||||||
|
this->w = w;
|
||||||
|
this->h = h;
|
||||||
|
|
||||||
|
auto comps = this->item->findChildren<UIComponent>();
|
||||||
|
auto itComps = comps.begin();
|
||||||
|
while(itComps != comps.end()) {
|
||||||
|
(*itComps)->alignmentNeedsUpdating = true;
|
||||||
|
++itComps;
|
||||||
|
}
|
||||||
|
}, camera->eventRenderTargetResized);
|
||||||
|
}, camera)();
|
||||||
}
|
}
|
||||||
|
|
||||||
void UICanvas::onDispose() {
|
void UICanvas::onDispose() {
|
||||||
|
@ -9,13 +9,26 @@
|
|||||||
#include "scene/components/display/Camera.hpp"
|
#include "scene/components/display/Camera.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
|
class UIComponentDimensional {
|
||||||
|
public:
|
||||||
|
virtual float_t getWidth() = 0;
|
||||||
|
virtual float_t getHeight() = 0;
|
||||||
|
virtual float_t getContentWidth() = 0;
|
||||||
|
virtual float_t getContentHeight() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
enum UIDrawType {
|
enum UIDrawType {
|
||||||
UI_DRAW_TYPE_WORLD_ABSOLUTE,
|
UI_DRAW_TYPE_WORLD_ABSOLUTE,
|
||||||
UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE,
|
UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE,
|
||||||
UI_DRAW_TYPE_CAMERA_OVERLAY
|
// UI_DRAW_TYPE_CAMERA_OVERLAY
|
||||||
};
|
};
|
||||||
|
|
||||||
class UICanvas : public SceneItemComponent {
|
class UICanvas : public SceneItemComponent, public UIComponentDimensional {
|
||||||
|
private:
|
||||||
|
std::function<void()> evtRenderResize;
|
||||||
|
float_t w = 1;
|
||||||
|
float_t h = 1;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/**
|
/**
|
||||||
* Creates a UI Canvas Scene Item Element, and attaches it to the provided
|
* Creates a UI Canvas Scene Item Element, and attaches it to the provided
|
||||||
@ -27,7 +40,7 @@ namespace Dawn {
|
|||||||
static UICanvas * create(Scene *scene);
|
static UICanvas * create(Scene *scene);
|
||||||
|
|
||||||
//======================================================================//
|
//======================================================================//
|
||||||
|
StateProperty<Camera*> camera;
|
||||||
enum UIDrawType drawType = UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE;
|
enum UIDrawType drawType = UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -37,6 +50,11 @@ namespace Dawn {
|
|||||||
*/
|
*/
|
||||||
UICanvas(SceneItem *item);
|
UICanvas(SceneItem *item);
|
||||||
|
|
||||||
|
float_t getWidth() override;
|
||||||
|
float_t getHeight() override;
|
||||||
|
float_t getContentWidth() override;
|
||||||
|
float_t getContentHeight() override;
|
||||||
|
|
||||||
void onStart() override;
|
void onStart() override;
|
||||||
void onDispose() override;
|
void onDispose() override;
|
||||||
};
|
};
|
||||||
|
@ -7,6 +7,113 @@
|
|||||||
|
|
||||||
using namespace Dawn;
|
using namespace Dawn;
|
||||||
|
|
||||||
UIComponent::UIComponent(SceneItem *item) : SceneItemComponent(item) {
|
UIComponent::UIComponent(SceneItem *item) :
|
||||||
|
SceneItemComponent(item),
|
||||||
|
alignment(glm::vec4(0, 0, 128, 128)),
|
||||||
|
alignX(UI_COMPONENT_ALIGN_START),
|
||||||
|
alignY(UI_COMPONENT_ALIGN_START),
|
||||||
|
alignmentNeedsUpdating(true)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
UIComponentDimensional * UIComponent::getParentDimensional() {
|
||||||
|
auto parent = this->transform->getParent();
|
||||||
|
if(!parent) return nullptr;
|
||||||
|
auto dimensional = parent->item->getComponent<UIComponentDimensional>();
|
||||||
|
assertNotNull(dimensional);
|
||||||
|
return dimensional;
|
||||||
|
}
|
||||||
|
|
||||||
|
void UIComponent::updateAlignment() {
|
||||||
|
if(!this->alignmentNeedsUpdating) return;
|
||||||
|
|
||||||
|
auto dimensional = this->getParentDimensional();
|
||||||
|
auto align = (glm::vec4)this->alignment;
|
||||||
|
auto translate = this->transform->getLocalPosition();
|
||||||
|
|
||||||
|
UIComponent::calculateDimensions(
|
||||||
|
this->alignX,
|
||||||
|
&translate.x,
|
||||||
|
&this->width,
|
||||||
|
dimensional->getWidth(),
|
||||||
|
this->getContentWidth(),
|
||||||
|
glm::vec2(align[0], align[2])
|
||||||
|
);
|
||||||
|
UIComponent::calculateDimensions(
|
||||||
|
this->alignY,
|
||||||
|
&translate.y,
|
||||||
|
&this->height,
|
||||||
|
dimensional->getHeight(),
|
||||||
|
this->getContentHeight(),
|
||||||
|
glm::vec2(align[1], align[3])
|
||||||
|
);
|
||||||
|
|
||||||
|
this->transform->setLocalPosition(translate);
|
||||||
|
this->alignmentNeedsUpdating = false;
|
||||||
|
|
||||||
|
this->eventAlignmentUpdated.invoke();
|
||||||
|
}
|
||||||
|
|
||||||
|
void UIComponent::calculateDimensions(
|
||||||
|
enum UIComponentAlign align,
|
||||||
|
float_t *position,
|
||||||
|
float_t *size,
|
||||||
|
float_t outerSize,
|
||||||
|
float_t innerSize,
|
||||||
|
glm::vec2 alignment
|
||||||
|
) {
|
||||||
|
assertNotNull(position);
|
||||||
|
assertNotNull(size);
|
||||||
|
|
||||||
|
switch(align) {
|
||||||
|
case UI_COMPONENT_ALIGN_STRETCH:
|
||||||
|
*position = alignment[0];
|
||||||
|
*size = outerSize + (alignment[0] + alignment[1]);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case UI_COMPONENT_ALIGN_START:
|
||||||
|
*position = alignment[0];
|
||||||
|
*size = alignment[1];
|
||||||
|
break;
|
||||||
|
|
||||||
|
case UI_COMPONENT_ALIGN_MIDDLE:
|
||||||
|
*size = mathMax<float_t>(innerSize, alignment[1]);
|
||||||
|
*position = (outerSize / 2.0f) - (innerSize / 2.0f) + alignment[0];
|
||||||
|
break;
|
||||||
|
|
||||||
|
case UI_COMPONENT_ALIGN_END:
|
||||||
|
*size = alignment[0];
|
||||||
|
*position = outerSize - innerSize - alignment[1];
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
assertUnreachable();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UIComponent::getWidth() {
|
||||||
|
return this->width;
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UIComponent::getHeight() {
|
||||||
|
return this->height;
|
||||||
|
}
|
||||||
|
|
||||||
|
void UIComponent::onStart() {
|
||||||
|
useEffect([&]{
|
||||||
|
this->alignmentNeedsUpdating = true;
|
||||||
|
}, { &alignment, &alignX, &alignY });
|
||||||
|
|
||||||
|
useEffect([&]{
|
||||||
|
this->updateAlignment();
|
||||||
|
|
||||||
|
auto child = this->item->findChildren<UIComponent>();
|
||||||
|
auto itChild = child.begin();
|
||||||
|
while(itChild != child.end()) {
|
||||||
|
(*itChild)->alignmentNeedsUpdating = true;
|
||||||
|
++itChild;
|
||||||
|
}
|
||||||
|
}, alignmentNeedsUpdating)();
|
||||||
}
|
}
|
@ -6,20 +6,71 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include "scene/SceneItemComponent.hpp"
|
#include "scene/SceneItemComponent.hpp"
|
||||||
#include "UICanvas.hpp"
|
#include "UICanvas.hpp"
|
||||||
|
#include "display/shader/Shader.hpp"
|
||||||
|
#include "util/mathutils.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
class UIComponentRendaerable {
|
enum UIComponentAlign {
|
||||||
public:
|
UI_COMPONENT_ALIGN_START,
|
||||||
int32_t cum;
|
UI_COMPONENT_ALIGN_MIDDLE,
|
||||||
|
UI_COMPONENT_ALIGN_END,
|
||||||
|
UI_COMPONENT_ALIGN_STRETCH
|
||||||
};
|
};
|
||||||
|
|
||||||
class UIComponent : public SceneItemComponent {
|
class UIComponentRenderable {
|
||||||
private:
|
public:
|
||||||
UICanvas *canvas = nullptr;
|
virtual std::vector<struct ShaderPassItem> getPassItems(
|
||||||
|
glm::mat4 projection,
|
||||||
|
glm::mat4 view
|
||||||
|
) = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class UIComponent : public SceneItemComponent, public UIComponentDimensional {
|
||||||
|
protected:
|
||||||
|
// Calculated (and cached) values
|
||||||
|
float_t width = 1;
|
||||||
|
float_t height = 1;
|
||||||
|
|
||||||
|
StateEvent<> eventAlignmentUpdated;
|
||||||
|
UIComponentDimensional * getParentDimensional();
|
||||||
|
|
||||||
|
void updateAlignment();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
StateProperty<bool_t> alignmentNeedsUpdating;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method used to calculate alignment dimensions.
|
||||||
|
*
|
||||||
|
* @param align Alignment value enumator.
|
||||||
|
* @param position Output position floating point.
|
||||||
|
* @param size Output size floating point.
|
||||||
|
* @param outerSize Outer size (of the parent).
|
||||||
|
* @param innerSize Inner size (of this element's content).
|
||||||
|
* @param alignment Alignment settings.
|
||||||
|
*/
|
||||||
|
static void calculateDimensions(
|
||||||
|
enum UIComponentAlign align,
|
||||||
|
float_t *position,
|
||||||
|
float_t *size,
|
||||||
|
float_t outerSize,
|
||||||
|
float_t innerSize,
|
||||||
|
glm::vec2 alignment
|
||||||
|
);
|
||||||
|
|
||||||
|
//
|
||||||
|
|
||||||
|
StateProperty<UIComponentAlign> alignX;
|
||||||
|
StateProperty<UIComponentAlign> alignY;
|
||||||
|
StateProperty<glm::vec4> alignment;
|
||||||
|
|
||||||
UIComponent(SceneItem *item);
|
UIComponent(SceneItem *item);
|
||||||
|
|
||||||
|
float_t getWidth() override;
|
||||||
|
float_t getHeight() override;
|
||||||
|
|
||||||
|
void onStart() override;
|
||||||
|
|
||||||
friend class UICanvas;
|
friend class UICanvas;
|
||||||
};
|
};
|
||||||
}
|
}
|
@ -4,6 +4,7 @@
|
|||||||
// https://opensource.org/licenses/MIT
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
#include "UILabel.hpp"
|
#include "UILabel.hpp"
|
||||||
|
#include "game/DawnGame.hpp"
|
||||||
|
|
||||||
using namespace Dawn;
|
using namespace Dawn;
|
||||||
|
|
||||||
@ -11,39 +12,100 @@ UILabel::UILabel(SceneItem *item) :
|
|||||||
UIComponent(item),
|
UIComponent(item),
|
||||||
text(""),
|
text(""),
|
||||||
fontSize(10.0f),
|
fontSize(10.0f),
|
||||||
font(nullptr)
|
font(nullptr),
|
||||||
|
maxWidth(UI_LABEL_MAX_WIDTH_NONE)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool_t UILabel::hasText() {
|
||||||
|
return (
|
||||||
|
this->font != nullptr &&
|
||||||
|
this->font->isReady() &&
|
||||||
|
((std::string)this->text).size() > 0
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
void UILabel::updateMesh() {
|
void UILabel::updateMesh() {
|
||||||
if(!this->needsRebuffering || !this->hasText) return;
|
if(!this->needsRebuffering) return;
|
||||||
if(((Font*)this->font) == nullptr || !this->font->isReady()) return;
|
if(!this->hasText()) return;
|
||||||
|
|
||||||
// float_t width = this->width;
|
float_t width = this->maxWidth;
|
||||||
// if(width == 0) width = -1;
|
assertTrue(width == 0 || width == -1 || width > 0);
|
||||||
|
if(width == 0) {
|
||||||
// std::string text = this->getGame()->localeManager.getString(key);
|
auto dimensional = this->getParentDimensional();
|
||||||
|
auto align = (glm::vec4)this->alignment;
|
||||||
|
float_t x;
|
||||||
|
UIComponent::calculateDimensions(
|
||||||
|
this->alignX,
|
||||||
|
&x,
|
||||||
|
&width,
|
||||||
|
dimensional->getWidth(),
|
||||||
|
0,
|
||||||
|
glm::vec2(align[0], align[2])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this->font->buffer(
|
this->font->buffer(
|
||||||
text,
|
text,
|
||||||
fontSize,
|
fontSize,
|
||||||
-1,
|
width,
|
||||||
&this->mesh,
|
&this->mesh,
|
||||||
&this->measure
|
&this->measure
|
||||||
);
|
);
|
||||||
|
|
||||||
this->needsRebuffering = false;
|
this->needsRebuffering = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<struct ShaderPassItem> UILabel::getPassItems(
|
||||||
|
glm::mat4 proj, glm::mat4 view
|
||||||
|
) {
|
||||||
|
if(!this->hasText()) return {};
|
||||||
|
this->updateMesh();
|
||||||
|
|
||||||
|
struct ShaderPassItem item;
|
||||||
|
auto shader = &getGame()->renderManager.uiShaderProgram;
|
||||||
|
item.shaderProgram = shader;
|
||||||
|
item.colorValues[shader->paramColor] = textColor;
|
||||||
|
item.matrixValues[shader->paramProjection] = proj;
|
||||||
|
item.matrixValues[shader->paramView] = view;
|
||||||
|
item.matrixValues[shader->paramModel] = this->transform->getWorldTransform();
|
||||||
|
item.boolValues[shader->paramHasTexture] = true;
|
||||||
|
item.textureSlots[0] = this->font->getTexture();
|
||||||
|
item.textureValues[shader->paramTexture] = 0;
|
||||||
|
item.start = this->startQuad * QUAD_INDICE_COUNT;
|
||||||
|
item.count = this->quadCount == -1 ? -1 : this->quadCount * QUAD_INDICE_COUNT;
|
||||||
|
item.w = this->transform->getWorldPosition().z;
|
||||||
|
item.renderFlags = RENDER_MANAGER_RENDER_FLAG_BLEND;
|
||||||
|
item.mesh = &mesh;
|
||||||
|
|
||||||
|
return { item };
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UILabel::getContentWidth() {
|
||||||
|
if(this->maxWidth > 0) return this->maxWidth;
|
||||||
|
if(!this->hasText()) return 0;
|
||||||
|
this->updateMesh();
|
||||||
|
return this->measure.getWidth();
|
||||||
|
}
|
||||||
|
|
||||||
|
float_t UILabel::getContentHeight() {
|
||||||
|
if(!this->hasText()) return 0;
|
||||||
|
this->updateMesh();
|
||||||
|
return this->measure.getHeight();
|
||||||
|
}
|
||||||
|
|
||||||
void UILabel::onStart() {
|
void UILabel::onStart() {
|
||||||
|
UIComponent::onStart();
|
||||||
|
|
||||||
useEffect([&]{
|
useEffect([&]{
|
||||||
hasText = ((std::string)text).size() > 0;
|
alignmentNeedsUpdating = true;
|
||||||
needsRebuffering = true;
|
}, { &fontSize, &font, &text, &maxWidth });
|
||||||
}, text);
|
|
||||||
|
|
||||||
useEffect([&]{
|
useEffect([&]{
|
||||||
needsRebuffering = true;
|
needsRebuffering = true;
|
||||||
}, fontSize);
|
}, alignmentNeedsUpdating);
|
||||||
|
|
||||||
useEffect([&]{
|
useEvent([&]{
|
||||||
needsRebuffering = true;
|
needsRebuffering = true;
|
||||||
}, font);
|
}, eventAlignmentUpdated);
|
||||||
}
|
}
|
@ -7,20 +7,23 @@
|
|||||||
#include "UIComponent.hpp"
|
#include "UIComponent.hpp"
|
||||||
#include "display/font/Font.hpp"
|
#include "display/font/Font.hpp"
|
||||||
|
|
||||||
|
#define UI_LABEL_MAX_WIDTH_NONE -1
|
||||||
|
#define UI_LABEL_MAX_WIDTH_ALIGN 0
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
class UILabel : public UIComponent {
|
class UILabel : public UIComponent, public UIComponentRenderable {
|
||||||
private:
|
private:
|
||||||
bool_t needsRebuffering = true;
|
bool_t needsRebuffering = true;
|
||||||
bool_t hasText = false;
|
|
||||||
|
|
||||||
Mesh mesh;
|
Mesh mesh;
|
||||||
|
|
||||||
|
bool_t hasText();
|
||||||
void updateMesh();
|
void updateMesh();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
StateProperty<std::string> text;
|
StateProperty<std::string> text;
|
||||||
StateProperty<float_t> fontSize;
|
StateProperty<float_t> fontSize;
|
||||||
StateProperty<Font*> font;
|
StateProperty<Font*> font;
|
||||||
|
StateProperty<float_t> maxWidth;
|
||||||
struct Color textColor = COLOR_MAGENTA;
|
struct Color textColor = COLOR_MAGENTA;
|
||||||
struct FontMeasure measure;
|
struct FontMeasure measure;
|
||||||
int32_t startQuad = 0;
|
int32_t startQuad = 0;
|
||||||
@ -28,6 +31,11 @@ namespace Dawn {
|
|||||||
|
|
||||||
UILabel(SceneItem *item);
|
UILabel(SceneItem *item);
|
||||||
|
|
||||||
|
float_t getContentWidth() override;
|
||||||
|
float_t getContentHeight() override;
|
||||||
|
std::vector<struct ShaderPassItem> getPassItems(
|
||||||
|
glm::mat4 projection, glm::mat4 view
|
||||||
|
) override;
|
||||||
void onStart() override;
|
void onStart() override;
|
||||||
};
|
};
|
||||||
}
|
}
|
@ -1,98 +1,98 @@
|
|||||||
// Copyright (c) 2023 Dominic Masters
|
// Copyright (c) 2023 Dominic Masters
|
||||||
//
|
//
|
||||||
// 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
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "StateInterfaces.hpp"
|
#include "StateInterfaces.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
template<class V>
|
template<class V>
|
||||||
class StateProperty : public IStateProperty {
|
class StateProperty : public IStateProperty {
|
||||||
private:
|
private:
|
||||||
/**
|
/**
|
||||||
* Method that is invoked every time that the value of this state property
|
* Method that is invoked every time that the value of this state property
|
||||||
* is updated.
|
* is updated.
|
||||||
*
|
*
|
||||||
* @param val Value that is to be used for this property.
|
* @param val Value that is to be used for this property.
|
||||||
*/
|
*/
|
||||||
void setInternal(V val) {
|
void setInternal(V val) {
|
||||||
if(val == this->_realValue) return;// TODO: can I omit this? kinda bad tbh.
|
if(val == this->_realValue) return;// TODO: can I omit this? kinda bad tbh.
|
||||||
|
|
||||||
// Run the teardowns
|
// Run the teardowns
|
||||||
auto itTeardown = this->_effectTeardowns.begin();
|
auto itTeardown = this->_effectTeardowns.begin();
|
||||||
while(itTeardown != this->_effectTeardowns.end()) {
|
while(itTeardown != this->_effectTeardowns.end()) {
|
||||||
(*itTeardown)();
|
(*itTeardown)();
|
||||||
++itTeardown;
|
++itTeardown;
|
||||||
}
|
}
|
||||||
this->_effectTeardowns.clear();
|
this->_effectTeardowns.clear();
|
||||||
|
|
||||||
// Update the values
|
// Update the values
|
||||||
this->previous = this->_realValue;
|
this->previous = this->_realValue;
|
||||||
this->_realValue = val;
|
this->_realValue = val;
|
||||||
|
|
||||||
// Notify the effect listeners
|
// Notify the effect listeners
|
||||||
auto itEffect = this->_effectListners.begin();
|
auto itEffect = this->_effectListners.begin();
|
||||||
while(itEffect != this->_effectListners.end()) {
|
while(itEffect != this->_effectListners.end()) {
|
||||||
(*itEffect)();
|
(*itEffect)();
|
||||||
++itEffect;
|
++itEffect;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Notify the teardown effect listeners
|
// Notify the teardown effect listeners
|
||||||
auto itWithTeardown = this->_effectListnersWithTeardown.begin();
|
auto itWithTeardown = this->_effectListnersWithTeardown.begin();
|
||||||
while(itWithTeardown != this->_effectListnersWithTeardown.end()) {
|
while(itWithTeardown != this->_effectListnersWithTeardown.end()) {
|
||||||
auto teardown = (*itWithTeardown)();
|
auto teardown = (*itWithTeardown)();
|
||||||
this->_effectTeardowns.push_back(teardown);
|
this->_effectTeardowns.push_back(teardown);
|
||||||
++itWithTeardown;
|
++itWithTeardown;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
V _realValue;
|
V _realValue;
|
||||||
V previous;
|
V previous;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new state property and listens for its change.
|
* Creates a new state property and listens for its change.
|
||||||
*/
|
*/
|
||||||
StateProperty() {}
|
StateProperty() {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new state property and listens for its change.
|
* Creates a new state property and listens for its change.
|
||||||
* @param initial The initial value of this state.
|
* @param initial The initial value of this state.
|
||||||
*/
|
*/
|
||||||
StateProperty(V initial) : _realValue(initial), previous(initial) {}
|
StateProperty(V initial) : _realValue(initial), previous(initial) {}
|
||||||
|
|
||||||
const StateProperty& operator += (const V &value) {
|
const StateProperty& operator += (const V &value) {
|
||||||
this->setInternal(this->_realValue + value);
|
this->setInternal(this->_realValue + value);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
const bool_t operator != (const V &value) {
|
const bool_t operator != (const V &value) {
|
||||||
return value != this->_realValue;
|
return value != this->_realValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const bool_t operator == (const V &value) {
|
const bool_t operator == (const V &value) {
|
||||||
return value == this->_realValue;
|
return value == this->_realValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const StateProperty& operator = (const V &val) {
|
const StateProperty& operator = (const V &val) {
|
||||||
this->setInternal(val);
|
this->setInternal(val);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
const V operator->() const {
|
const V operator->() const {
|
||||||
return this->_realValue;
|
return this->_realValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
operator V() const {
|
operator V() const {
|
||||||
return this->_realValue;
|
return this->_realValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Destructor for StateProperty.
|
* Destructor for StateProperty.
|
||||||
*/
|
*/
|
||||||
~StateProperty() {}
|
~StateProperty() {}
|
||||||
|
|
||||||
friend class StateOwner;
|
friend class StateOwner;
|
||||||
};
|
};
|
||||||
}
|
}
|
@ -1,43 +1,14 @@
|
|||||||
// Copyright (c) 2022 Dominic Masters
|
// Copyright (c) 2022 Dominic Masters
|
||||||
//
|
//
|
||||||
// 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
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "SimpleTexturedShaderProgram.hpp"
|
#include "SimpleTexturedShaderProgram.hpp"
|
||||||
|
|
||||||
#define UI_SHADER_PROGRAM_PRIORITY 100
|
#define UI_SHADER_PROGRAM_PRIORITY 100
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
class UIShaderProgram : public SimpleTexturedShaderProgram {
|
class UIShaderProgram : public SimpleTexturedShaderProgram {
|
||||||
public:
|
};
|
||||||
struct ShaderPassItem getUIPassItem(
|
|
||||||
glm::mat4 projection,
|
|
||||||
glm::mat4 view,
|
|
||||||
glm::mat4 transform,
|
|
||||||
Texture *texture,
|
|
||||||
struct Color color,
|
|
||||||
Mesh *mesh,
|
|
||||||
float_t z
|
|
||||||
) {
|
|
||||||
struct ShaderPassItem item;
|
|
||||||
item.shaderProgram = this;
|
|
||||||
item.colorValues[this->paramColor] = color;
|
|
||||||
if(texture == nullptr) {
|
|
||||||
item.boolValues[this->paramHasTexture] = false;
|
|
||||||
} else {
|
|
||||||
item.textureSlots[0] = texture;
|
|
||||||
item.textureValues[this->paramTexture] = 0;
|
|
||||||
item.boolValues[this->paramHasTexture] = true;
|
|
||||||
}
|
|
||||||
item.matrixValues[this->paramProjection] = projection;
|
|
||||||
item.matrixValues[this->paramView] = view;
|
|
||||||
item.matrixValues[this->paramModel] = transform;
|
|
||||||
item.priority = UI_SHADER_PROGRAM_PRIORITY;
|
|
||||||
item.w = z;
|
|
||||||
item.renderFlags = RENDER_MANAGER_RENDER_FLAG_BLEND;
|
|
||||||
item.mesh = mesh;
|
|
||||||
return item;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
@ -86,7 +86,7 @@ void TicTacToeGame::onStart() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(isPlayerMove) {
|
if(isPlayerMove) {
|
||||||
if(getGame()->inputManager.isPressed(INPUT_BIND_MOUSE_CLICK)) {
|
if(getGame()->inputManager.isPressed(INPUT_BIND_MOUSE_CLICK) && hovered != nullptr) {
|
||||||
this->makeMove(hovered->tile, nextMove);
|
this->makeMove(hovered->tile, nextMove);
|
||||||
}
|
}
|
||||||
} else if(nextMove != TIC_TAC_TOE_NOUGHT) {
|
} else if(nextMove != TIC_TAC_TOE_NOUGHT) {
|
||||||
|
@ -15,15 +15,18 @@
|
|||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
|
|
||||||
class TicTacToeScene : public Scene {
|
class TicTacToeScene : public Scene, public StateOwner {
|
||||||
protected:
|
protected:
|
||||||
Camera *camera;
|
Camera *camera;
|
||||||
int32_t age = 0;
|
|
||||||
std::function<void()> evtUnsub;
|
std::function<void()> evtUnsub;
|
||||||
|
|
||||||
|
UILabel *label;
|
||||||
|
StateProperty<int32_t> test;
|
||||||
|
|
||||||
void stage() override {
|
void stage() override {
|
||||||
camera = Camera::create(this);
|
camera = Camera::create(this);
|
||||||
camera->transform->lookAt(glm::vec3(0, 0, 8), glm::vec3(0, 0, 0));
|
// camera->transform->lookAt(glm::vec3(0, 0, 8), glm::vec3(0, 0, 0));
|
||||||
|
camera->transform->lookAt(glm::vec3(32, 32, 32), glm::vec3(0, 0, 0));
|
||||||
|
|
||||||
float_t s = 2.0f;
|
float_t s = 2.0f;
|
||||||
camera->orthoTop = s;
|
camera->orthoTop = s;
|
||||||
@ -49,10 +52,24 @@ namespace Dawn {
|
|||||||
auto canvas = canvasItem->addComponent<UICanvas>();
|
auto canvas = canvasItem->addComponent<UICanvas>();
|
||||||
|
|
||||||
auto labelItem = this->createSceneItem();
|
auto labelItem = this->createSceneItem();
|
||||||
auto label = labelItem->addComponent<UILabel>();
|
label = labelItem->addComponent<UILabel>();
|
||||||
label->font = &this->game->assetManager.get<TrueTypeAsset>("truetype_bizudp")->font;
|
label->font = &this->game->assetManager.get<TrueTypeAsset>("truetype_bizudp")->font;
|
||||||
label->text = "Hello World";
|
label->text = "Hello World";
|
||||||
|
label->fontSize = 36.0f;
|
||||||
labelItem->transform.setParent(canvas->transform);
|
labelItem->transform.setParent(canvas->transform);
|
||||||
|
|
||||||
|
label->alignX = UI_COMPONENT_ALIGN_MIDDLE;
|
||||||
|
label->alignY = UI_COMPONENT_ALIGN_MIDDLE;
|
||||||
|
label->alignment = glm::vec4(0, 0, 350, 100);
|
||||||
|
label->maxWidth = 0;
|
||||||
|
|
||||||
|
useEffect([&]{
|
||||||
|
label->text = "Hello " + std::to_string(test) + " world";
|
||||||
|
}, test);
|
||||||
|
|
||||||
|
useEvent([&](float_t delta){
|
||||||
|
test = test + 1;
|
||||||
|
}, this->eventSceneUpdate);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<Asset*> getRequiredAssets() override {
|
std::vector<Asset*> getRequiredAssets() override {
|
||||||
@ -65,6 +82,6 @@ namespace Dawn {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TicTacToeScene(DawnGame *game) : Scene(game) {}
|
TicTacToeScene(DawnGame *game) : Scene(game), test(0) {}
|
||||||
};
|
};
|
||||||
}
|
}
|
Reference in New Issue
Block a user