Added UI Scaling support
This commit is contained in:
@ -30,6 +30,14 @@ namespace Dawn {
|
|||||||
*/
|
*/
|
||||||
virtual float_t getHeight() = 0;
|
virtual float_t getHeight() = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the scale (as in pixel density) of the render target. This is
|
||||||
|
* typically 1.0f, but on high DPI displays this may be 2.0f or higher.
|
||||||
|
*
|
||||||
|
* @return The scale of the render target.
|
||||||
|
*/
|
||||||
|
virtual float_t getScale() = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the clear color of the render target when the clear method for
|
* Sets the clear color of the render target when the clear method for
|
||||||
* the color buffer is requested.
|
* the color buffer is requested.
|
||||||
|
@ -32,8 +32,8 @@ void UICanvas::rebufferShaderParameters() {
|
|||||||
case UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE:
|
case UI_DRAW_TYPE_WORLD_CAMERA_RELATIVE:
|
||||||
data.projection = glm::ortho(
|
data.projection = glm::ortho(
|
||||||
0.0f,
|
0.0f,
|
||||||
camera->getRenderTarget()->getWidth(),
|
camera->getRenderTarget()->getWidth() / this->getScale(),
|
||||||
camera->getRenderTarget()->getHeight(),
|
camera->getRenderTarget()->getHeight() / this->getScale(),
|
||||||
0.0f
|
0.0f
|
||||||
);
|
);
|
||||||
data.view = glm::mat4(1.0f);
|
data.view = glm::mat4(1.0f);
|
||||||
@ -46,6 +46,10 @@ void UICanvas::rebufferShaderParameters() {
|
|||||||
this->shaderBuffer.buffer(&data);
|
this->shaderBuffer.buffer(&data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float_t UICanvas::getScale() {
|
||||||
|
return this->camera->getRenderTarget()->getScale();
|
||||||
|
}
|
||||||
|
|
||||||
float_t UICanvas::getWidth() {
|
float_t UICanvas::getWidth() {
|
||||||
return w;
|
return w;
|
||||||
}
|
}
|
||||||
@ -78,13 +82,13 @@ void UICanvas::onStart() {
|
|||||||
|
|
||||||
useEffectWithTeardown([&]{
|
useEffectWithTeardown([&]{
|
||||||
if(camera == nullptr) return evtRenderResize = [&] {};
|
if(camera == nullptr) return evtRenderResize = [&] {};
|
||||||
this->w = camera->getRenderTarget()->getWidth();
|
this->w = camera->getRenderTarget()->getWidth() / this->getScale();
|
||||||
this->h = camera->getRenderTarget()->getHeight();
|
this->h = camera->getRenderTarget()->getHeight() / this->getScale();
|
||||||
this->rebufferShaderParameters();
|
this->rebufferShaderParameters();
|
||||||
|
|
||||||
return evtRenderResize = useEvent([&](float_t w, float_t h){
|
return evtRenderResize = useEvent([&](float_t w, float_t h){
|
||||||
this->w = w;
|
this->w = w / this->getScale();
|
||||||
this->h = h;
|
this->h = h / this->getScale();
|
||||||
this->rebufferShaderParameters();
|
this->rebufferShaderParameters();
|
||||||
|
|
||||||
auto comps = this->item->findChildren<UIComponent>();
|
auto comps = this->item->findChildren<UIComponent>();
|
||||||
|
@ -96,6 +96,13 @@ namespace Dawn {
|
|||||||
*/
|
*/
|
||||||
UICanvas(SceneItem *item);
|
UICanvas(SceneItem *item);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the scale of this canvas.
|
||||||
|
*
|
||||||
|
* @return The scale of the canvas, where 1 is default scaling.
|
||||||
|
*/
|
||||||
|
float_t getScale();
|
||||||
|
|
||||||
float_t getWidth() override;
|
float_t getWidth() override;
|
||||||
float_t getHeight() override;
|
float_t getHeight() override;
|
||||||
float_t getContentWidth() override;
|
float_t getContentWidth() override;
|
||||||
|
@ -130,8 +130,8 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
quadCountTotal = 0;
|
quadCountTotal = 0;
|
||||||
quadCount = -1;
|
quadCount = -1;
|
||||||
std::vector<struct UILabelText> realNewTexts;
|
std::vector<struct UILabelText> realNewTexts;
|
||||||
|
|
||||||
float_t maxWidth = this->width;
|
float_t maxWidth = this->width;
|
||||||
|
float_t canvasScale = this->getCanvas()->getScale();
|
||||||
|
|
||||||
// Reset
|
// Reset
|
||||||
lines.clear();
|
lines.clear();
|
||||||
@ -141,7 +141,7 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
// Determine font dimensions.
|
// Determine font dimensions.
|
||||||
auto itText = newTexts.begin();
|
auto itText = newTexts.begin();
|
||||||
while(itText != newTexts.end()) {
|
while(itText != newTexts.end()) {
|
||||||
position.y = mathMax<float_t>(position.y, itText->style.size/* this->lineHeight - THIS PART WOULD TAKE THE LINE HEIGHT INTO CONSIDERATION ON THE FIRST/INITIAL LINE */);
|
position.y = mathMax<float_t>(position.y, itText->style.size);
|
||||||
++itText;
|
++itText;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,7 +163,7 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
// Lock the font
|
// Lock the font
|
||||||
assertNotNull(text.style.font, "UILabel::rebufferQuads: Font cannot be null");
|
assertNotNull(text.style.font, "UILabel::rebufferQuads: Font cannot be null");
|
||||||
realText.lockId = text.style.font->lock(TrueTypeFaceTextureStyle{
|
realText.lockId = text.style.font->lock(TrueTypeFaceTextureStyle{
|
||||||
text.style.size,
|
(uint32_t)(text.style.size * canvasScale),// Scale for resolution.
|
||||||
text.style.style
|
text.style.style
|
||||||
});
|
});
|
||||||
assertTrue(realText.lockId != -1, "UILabel::rebufferQuads: Failed to lock font");
|
assertTrue(realText.lockId != -1, "UILabel::rebufferQuads: Failed to lock font");
|
||||||
@ -259,13 +259,27 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
|
|
||||||
// Get font data.
|
// Get font data.
|
||||||
auto charInfo = realText.texture->getCharacterData(ch);
|
auto charInfo = realText.texture->getCharacterData(ch);
|
||||||
|
|
||||||
|
// Now we scale down the char info here. This is because we fetch the
|
||||||
|
// texture of the font based on the canvas scale, but the sizes that we
|
||||||
|
// render out need to be shrunk to match the original sizes.
|
||||||
|
glm::vec2 charSize = glm::vec2(
|
||||||
|
charInfo.bitmapSize.x / canvasScale, charInfo.bitmapSize.y / canvasScale
|
||||||
|
);
|
||||||
|
glm::vec2 charAdvance = glm::vec2(
|
||||||
|
charInfo.advanceX / canvasScale, charInfo.advanceY / canvasScale
|
||||||
|
);
|
||||||
|
glm::vec2 charPos = glm::vec2(
|
||||||
|
charInfo.bitmapPosition.x / canvasScale,
|
||||||
|
charInfo.bitmapPosition.y / canvasScale
|
||||||
|
);
|
||||||
|
|
||||||
// Word wrapping
|
// Word wrapping
|
||||||
if(
|
if(
|
||||||
ch != ' ' &&
|
ch != ' ' &&
|
||||||
lastSpaceCharacter != -1 &&
|
lastSpaceCharacter != -1 &&
|
||||||
maxWidth > charInfo.bitmapSize.x &&
|
maxWidth > charSize.x &&
|
||||||
(position.x + charInfo.advanceX) > maxWidth
|
(position.x + charAdvance.x) > maxWidth
|
||||||
) {
|
) {
|
||||||
// Basically this rewinds everything we've done to the last space char,
|
// Basically this rewinds everything we've done to the last space char,
|
||||||
// changes it to a newline, and then moves the position along.
|
// changes it to a newline, and then moves the position along.
|
||||||
@ -284,7 +298,7 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Buffer coordinates
|
// Buffer coordinates, use original (non scaled) values.
|
||||||
glm::vec4 uvs;
|
glm::vec4 uvs;
|
||||||
uvs.x = 0.0f;
|
uvs.x = 0.0f;
|
||||||
uvs.y = charInfo.textureY / wh.y;
|
uvs.y = charInfo.textureY / wh.y;
|
||||||
@ -292,10 +306,10 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
uvs.z = uvs.y + (charInfo.bitmapSize.y / wh.y);
|
uvs.z = uvs.y + (charInfo.bitmapSize.y / wh.y);
|
||||||
|
|
||||||
glm::vec4 vert;
|
glm::vec4 vert;
|
||||||
vert.x = position.x + charInfo.bitmapPosition.x;
|
vert.x = position.x + charPos.x;
|
||||||
vert.y = position.y + charInfo.bitmapPosition.y;
|
vert.y = position.y + charPos.y;
|
||||||
vert.w = vert.x + charInfo.bitmapSize.x;
|
vert.w = vert.x + charSize.x;
|
||||||
vert.z = vert.y + charInfo.bitmapSize.y;
|
vert.z = vert.y + charSize.y;
|
||||||
vertices.push_back(std::make_pair(vert, uvs));
|
vertices.push_back(std::make_pair(vert, uvs));
|
||||||
|
|
||||||
// Decorations
|
// Decorations
|
||||||
@ -318,16 +332,16 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Move the current position along.
|
// Move the current position along.
|
||||||
position.x += charInfo.advanceX;
|
position.x += charAdvance.x;
|
||||||
position.y += charInfo.advanceY;
|
position.y += charAdvance.y;
|
||||||
|
|
||||||
// Update the continuous dimensions
|
// Update the continuous dimensions
|
||||||
if(ch == ' ') {
|
if(ch == ' ') {
|
||||||
lineWidth += wordWidth;
|
lineWidth += wordWidth;
|
||||||
lineWidth += charInfo.advanceX;
|
lineWidth += charAdvance.x;
|
||||||
wordWidth = 0.0f;
|
wordWidth = 0.0f;
|
||||||
} else {
|
} else {
|
||||||
wordWidth += charInfo.advanceX;
|
wordWidth += charAdvance.x;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set the part index to the quad mappings
|
// Set the part index to the quad mappings
|
||||||
|
@ -8,4 +8,5 @@ if(DAWN_TARGET_WIN32 AND DAWN_TARGET_GLFW)
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(DAWN_TARGET_OSX AND DAWN_TARGET_GLFW)
|
if(DAWN_TARGET_OSX AND DAWN_TARGET_GLFW)
|
||||||
|
add_subdirectory(opengl-osx)
|
||||||
endif()
|
endif()
|
@ -57,10 +57,17 @@ int32_t DawnHost::init(DawnGame *game) {
|
|||||||
|
|
||||||
// Override the defaults
|
// Override the defaults
|
||||||
int32_t fbWidth, fbHeight;
|
int32_t fbWidth, fbHeight;
|
||||||
|
int32_t windowWidth, windowHeight;
|
||||||
glfwGetFramebufferSize(this->data->window, &fbWidth, &fbHeight);
|
glfwGetFramebufferSize(this->data->window, &fbWidth, &fbHeight);
|
||||||
|
glfwGetWindowSize(this->data->window, &windowWidth, &windowHeight);
|
||||||
|
|
||||||
assertTrue(fbWidth > 0, "Detected framebuffer width is too small?");
|
assertTrue(fbWidth > 0, "Detected framebuffer width is too small?");
|
||||||
assertTrue(fbWidth > 0, "Detected framebuffer height is too small?");
|
assertTrue(fbWidth > 0, "Detected framebuffer height is too small?");
|
||||||
|
assertTrue(windowWidth > 0, "Detected window width is too small?");
|
||||||
|
assertTrue(windowHeight > 0, "Detected window height is too small?");
|
||||||
game->renderManager.backBuffer.setSize(fbWidth, fbHeight);
|
game->renderManager.backBuffer.setSize(fbWidth, fbHeight);
|
||||||
|
game->renderManager.backBuffer.scale = ((float_t)fbWidth) / ((float_t)windowWidth);
|
||||||
|
assertTrue(game->renderManager.backBuffer.scale > 0, "Back buffer scale is invalid");
|
||||||
assertNoGLError();
|
assertNoGLError();
|
||||||
|
|
||||||
// Default keybinds
|
// Default keybinds
|
||||||
@ -166,9 +173,18 @@ void glfwOnError(int error, const char* description) {
|
|||||||
|
|
||||||
void glfwOnResize(GLFWwindow *window, int32_t w, int32_t h) {
|
void glfwOnResize(GLFWwindow *window, int32_t w, int32_t h) {
|
||||||
if(DAWN_HOST == nullptr) return;
|
if(DAWN_HOST == nullptr) return;
|
||||||
|
assertTrue(window == DAWN_HOST->data->window, "glfwOnResize: Window mismatch");
|
||||||
|
|
||||||
|
auto backBuffer = ((BackBufferRenderTarget*)&DAWN_HOST->game->renderManager.backBuffer);
|
||||||
|
assertNotNull(backBuffer, "glfwOnResize: Back buffer is not valid");
|
||||||
|
|
||||||
|
int32_t windowWidth, windowHeight;
|
||||||
|
glfwGetWindowSize(window, &windowWidth, &windowHeight);
|
||||||
|
|
||||||
// TODO: I may throttle this, it calls for every frame the window's resized.
|
// TODO: I may throttle this, it calls for every frame the window's resized.
|
||||||
DAWN_HOST->game->renderManager.backBuffer.setSize((float_t)w, (float_t)h);
|
backBuffer->setSize((float_t)w, (float_t)h);
|
||||||
|
backBuffer->scale = ((float_t)w) / ((float_t)windowWidth);
|
||||||
|
assertTrue(backBuffer->scale > 0, "glfwOnResize: Back buffer scale is invalid");
|
||||||
}
|
}
|
||||||
|
|
||||||
void glfwOnKey(
|
void glfwOnKey(
|
||||||
|
@ -12,6 +12,10 @@ BackBufferRenderTarget::BackBufferRenderTarget(RenderManager &renderManager) :
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float_t BackBufferRenderTarget::getScale() {
|
||||||
|
return this->scale;
|
||||||
|
}
|
||||||
|
|
||||||
float_t BackBufferRenderTarget::getWidth() {
|
float_t BackBufferRenderTarget::getWidth() {
|
||||||
return this->width;
|
return this->width;
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,8 @@ namespace Dawn {
|
|||||||
struct Color clearColor = COLOR_CORNFLOWER_BLUE;
|
struct Color clearColor = COLOR_CORNFLOWER_BLUE;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
float_t scale = 1.0f;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct the back buffer render target.
|
* Construct the back buffer render target.
|
||||||
*
|
*
|
||||||
@ -38,6 +40,7 @@ namespace Dawn {
|
|||||||
*/
|
*/
|
||||||
void setSize(float_t width, float_t height);
|
void setSize(float_t width, float_t height);
|
||||||
|
|
||||||
|
float_t getScale() override;
|
||||||
float_t getWidth() override;
|
float_t getWidth() override;
|
||||||
float_t getHeight() override;
|
float_t getHeight() override;
|
||||||
void setClearColor(struct Color color) override;
|
void setClearColor(struct Color color) override;
|
||||||
|
@ -64,6 +64,10 @@ void TextureRenderTarget::setSize(float_t width, float_t height) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float_t TextureRenderTarget::getScale() {
|
||||||
|
return 1.0f;
|
||||||
|
}
|
||||||
|
|
||||||
float_t TextureRenderTarget::getWidth() {
|
float_t TextureRenderTarget::getWidth() {
|
||||||
return (float_t)this->texture.getWidth();
|
return (float_t)this->texture.getWidth();
|
||||||
}
|
}
|
||||||
|
@ -26,6 +26,7 @@ namespace Dawn {
|
|||||||
|
|
||||||
void setSize(float_t width, float_t height);
|
void setSize(float_t width, float_t height);
|
||||||
|
|
||||||
|
float_t getScale() override;
|
||||||
float_t getWidth() override;
|
float_t getWidth() override;
|
||||||
float_t getHeight() override;
|
float_t getHeight() override;
|
||||||
void setClearColor(struct Color color) override;
|
void setClearColor(struct Color color) override;
|
||||||
|
Reference in New Issue
Block a user