Adding bitmap fonts
This commit is contained in:
2
lib/SDL
2
lib/SDL
Submodule lib/SDL updated: c9aec268fa...87a83787a3
Submodule lib/openal-soft updated: fde74453a6...d66107e9f0
@ -13,7 +13,7 @@ set(
|
||||
add_subdirectory(dawnshared)
|
||||
|
||||
# Include tools
|
||||
# add_subdirectory(dawntools)
|
||||
add_subdirectory(dawntools)
|
||||
|
||||
# Change what we are building. Pulled from the cmake/targets dir.
|
||||
add_subdirectory(${DAWN_BUILDING})
|
||||
|
124
src/dawn/display/font/BitmapFont.cpp
Normal file
124
src/dawn/display/font/BitmapFont.cpp
Normal file
@ -0,0 +1,124 @@
|
||||
// Copyright (c) 2023 Dominic Masters
|
||||
//
|
||||
// This software is released under the MIT License.
|
||||
// https://opensource.org/licenses/MIT
|
||||
|
||||
#include "BitmapFont.hpp"
|
||||
|
||||
using namespace Dawn;
|
||||
|
||||
void BitmapFont::buffer(
|
||||
std::string text,
|
||||
float_t fontSize,
|
||||
float_t maxWidth,
|
||||
Mesh *mesh,
|
||||
struct FontMeasure *info
|
||||
) {
|
||||
assertNotNull(mesh);
|
||||
assertNotNull(info);
|
||||
assertTrue(maxWidth == -1 || maxWidth > 0);
|
||||
assertTrue(this->isReady());
|
||||
|
||||
// Initialize primitive
|
||||
mesh->createBuffers(
|
||||
QUAD_VERTICE_COUNT * text.size(),
|
||||
QUAD_INDICE_COUNT * text.size()
|
||||
);
|
||||
|
||||
// Setup Scales
|
||||
info->length = 0;
|
||||
info->realLength = 0;
|
||||
info->lines.clear();
|
||||
info->lineHeight = this->getLineHeight(fontSize);
|
||||
|
||||
// Prepare the line counters
|
||||
info->addLine(0, 0);
|
||||
|
||||
// Reset Dimensions
|
||||
char c;
|
||||
Tile tile;
|
||||
info->width = info->height = 0;
|
||||
float_t x = 0;
|
||||
float_t y = 0;
|
||||
size_t i = 0;
|
||||
size_t j = 0;
|
||||
glm::vec2 xy0(0, 0);
|
||||
glm::vec2 tileSize = glm::vec2(tileset->getTileWidth(), tileset->getTileHeight());
|
||||
|
||||
// Buffer quads
|
||||
while(c = text[i++]) {
|
||||
if(c == FONT_SPACE) {
|
||||
|
||||
// Did this space cause a newline?
|
||||
if(maxWidth != -1 && xy0.x > maxWidth) {
|
||||
info->addLine(i, 0);
|
||||
info->width = mathMax<float_t>(info->width, xy0.x);
|
||||
xy0.x = 0;
|
||||
xy0.y += tileSize.y;
|
||||
info->height = mathMax<float_t>(info->height, xy0.y);
|
||||
continue;
|
||||
}
|
||||
|
||||
xy0.x += tileSize.x;
|
||||
continue;
|
||||
}
|
||||
|
||||
if(c == FONT_NEWLINE) {
|
||||
info->addLine(i, 0);
|
||||
info->width = mathMax<float_t>(info->width, xy0.x);
|
||||
xy0.x = 0;
|
||||
xy0.y += tileSize.y;
|
||||
info->height = mathMax<float_t>(info->height, xy0.y);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check for wrapping, todo.
|
||||
if(maxWidth != -1 && (xy0.x+tileSize.x) > maxWidth) {
|
||||
// We've exceeded the edge, go back to the start of the word and newline.
|
||||
|
||||
// Go back to the previous (still current) line and remove the chars
|
||||
|
||||
// Next line begins with this word
|
||||
}
|
||||
|
||||
tile = this->tileset->getTile(c);
|
||||
QuadMesh::bufferQuadMesh(mesh,
|
||||
xy0, tile.uv0,
|
||||
xy0+tileSize, tile.uv1,
|
||||
j * QUAD_VERTICE_COUNT, j * QUAD_INDICE_COUNT
|
||||
);
|
||||
xy0.x += tileSize.x;//TODO: Spacing?
|
||||
j++;
|
||||
}
|
||||
|
||||
info->width = mathMax<float_t>(info->width, xy0.x);
|
||||
info->height = mathMax<float_t>(info->height, xy0.y + (xy0.x > 0 ? tileSize.y : 0));
|
||||
}
|
||||
|
||||
bool_t BitmapFont::isReady() {
|
||||
if(this->texture == nullptr) return false;
|
||||
if(this->tileset == nullptr) return false;
|
||||
return this->texture->isReady();
|
||||
}
|
||||
|
||||
Texture * BitmapFont::getTexture() {
|
||||
return this->texture;
|
||||
}
|
||||
|
||||
void BitmapFont::draw(Mesh *mesh, int32_t start, int32_t len) {
|
||||
assertNotNull(mesh);
|
||||
|
||||
mesh->draw(
|
||||
MESH_DRAW_MODE_TRIANGLES,
|
||||
start * QUAD_INDICE_COUNT,
|
||||
len == -1 ? len : len * QUAD_INDICE_COUNT
|
||||
);
|
||||
}
|
||||
|
||||
float_t BitmapFont::getLineHeight(float_t fontSize) {
|
||||
return 16.0f;
|
||||
}
|
||||
|
||||
float_t BitmapFont::getDefaultFontSize() {
|
||||
return 16.0f;
|
||||
}
|
31
src/dawn/display/font/BitmapFont.hpp
Normal file
31
src/dawn/display/font/BitmapFont.hpp
Normal file
@ -0,0 +1,31 @@
|
||||
// Copyright (c) 2023 Dominic Masters
|
||||
//
|
||||
// This software is released under the MIT License.
|
||||
// https://opensource.org/licenses/MIT
|
||||
|
||||
#pragma once
|
||||
#include "Font.hpp"
|
||||
#include "display/Tileset.hpp"
|
||||
|
||||
namespace Dawn {
|
||||
class BitmapFont : public Font {
|
||||
protected:
|
||||
|
||||
public:
|
||||
Texture *texture = nullptr;
|
||||
TilesetGrid *tileset = nullptr;
|
||||
|
||||
void buffer(
|
||||
std::string text,
|
||||
float_t fontSize,
|
||||
float_t maxWidth,
|
||||
Mesh *mesh,
|
||||
struct FontMeasure *info
|
||||
) override;
|
||||
bool_t isReady() override;
|
||||
Texture * getTexture() override;
|
||||
void draw(Mesh *mesh, int32_t startCharacter, int32_t length) override;
|
||||
float_t getLineHeight(float_t fontSize) override;
|
||||
float_t getDefaultFontSize() override;
|
||||
};
|
||||
}
|
@ -1,11 +1,12 @@
|
||||
# Copyright (c) 2022 Dominic Masters
|
||||
#
|
||||
# This software is released under the MIT License.
|
||||
# https://opensource.org/licenses/MIT
|
||||
|
||||
# Sources
|
||||
target_sources(${DAWN_TARGET_NAME}
|
||||
PRIVATE
|
||||
TrueTypeFont.cpp
|
||||
FontMeasure.cpp
|
||||
# Copyright (c) 2022 Dominic Masters
|
||||
#
|
||||
# This software is released under the MIT License.
|
||||
# https://opensource.org/licenses/MIT
|
||||
|
||||
# Sources
|
||||
target_sources(${DAWN_TARGET_NAME}
|
||||
PRIVATE
|
||||
BitmapFont.cpp
|
||||
TrueTypeFont.cpp
|
||||
FontMeasure.cpp
|
||||
)
|
@ -9,6 +9,7 @@ target_sources(${DAWN_TARGET_NAME}
|
||||
UICanvas.cpp
|
||||
UIComponent.cpp
|
||||
UILabel.cpp
|
||||
UIImage.cpp
|
||||
)
|
||||
|
||||
add_subdirectory(menu)
|
@ -28,8 +28,8 @@ UIComponentDimensional * UIComponent::getParentDimensional() {
|
||||
void UIComponent::updateAlignment() {
|
||||
if(!this->alignmentNeedsUpdating) return;
|
||||
|
||||
auto dimensional = this->getParentDimensional();
|
||||
auto align = (glm::vec4)this->alignment;
|
||||
auto dimensional = this->getParentDimensional();
|
||||
auto translate = this->transform->getLocalPosition();
|
||||
|
||||
UIComponent::calculateDimensions(
|
||||
@ -51,7 +51,6 @@ void UIComponent::updateAlignment() {
|
||||
|
||||
this->transform->setLocalPosition(translate);
|
||||
this->alignmentNeedsUpdating = false;
|
||||
|
||||
this->eventAlignmentUpdated.invoke();
|
||||
}
|
||||
|
||||
|
66
src/dawn/scene/components/ui/UIImage.cpp
Normal file
66
src/dawn/scene/components/ui/UIImage.cpp
Normal file
@ -0,0 +1,66 @@
|
||||
// Copyright (c) 2023 Dominic Masters
|
||||
//
|
||||
// This software is released under the MIT License.
|
||||
// https://opensource.org/licenses/MIT
|
||||
|
||||
#include "UIImage.hpp"
|
||||
#include "game/DawnGame.hpp"
|
||||
|
||||
using namespace Dawn;
|
||||
|
||||
UIImage::UIImage(SceneItem *item) :
|
||||
texture(nullptr),
|
||||
UIComponent(item)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
float_t UIImage::getContentWidth() {
|
||||
if(this->texture != nullptr) return this->texture->getWidth();
|
||||
return this->width;
|
||||
}
|
||||
|
||||
float_t UIImage::getContentHeight() {
|
||||
if(this->texture != nullptr) return this->texture->getHeight();
|
||||
return this->height;
|
||||
}
|
||||
|
||||
std::vector<struct ShaderPassItem> UIImage::getPassItems(
|
||||
glm::mat4 proj, glm::mat4 view
|
||||
) {
|
||||
struct ShaderPassItem item;
|
||||
auto shader = &getGame()->renderManager.uiShaderProgram;
|
||||
item.shaderProgram = shader;
|
||||
item.colorValues[shader->paramColor] = this->color;
|
||||
item.matrixValues[shader->paramProjection] = proj;
|
||||
item.matrixValues[shader->paramView] = view;
|
||||
item.matrixValues[shader->paramModel] = this->transform->getWorldTransform();
|
||||
if(this->texture == nullptr) {
|
||||
item.boolValues[shader->paramHasTexture] = false;
|
||||
} else {
|
||||
item.boolValues[shader->paramHasTexture] = true;
|
||||
item.textureSlots[0] = this->texture;
|
||||
item.textureValues[shader->paramTexture] = 0;
|
||||
}
|
||||
item.w = this->transform->getWorldPosition().z;
|
||||
item.renderFlags = RENDER_MANAGER_RENDER_FLAG_BLEND;
|
||||
item.mesh = &mesh;
|
||||
|
||||
return { item };
|
||||
}
|
||||
|
||||
void UIImage::onStart() {
|
||||
UIComponent::onStart();
|
||||
|
||||
useEvent([&]{
|
||||
QuadMesh::bufferPositions(&mesh,
|
||||
glm::vec2(0, 0), glm::vec2(width, height), 0
|
||||
);
|
||||
}, this->eventAlignmentUpdated);
|
||||
|
||||
QuadMesh::initQuadMesh(&mesh,
|
||||
glm::vec2(0, 0), glm::vec2(0, 0),
|
||||
glm::vec2(width, height), glm::vec2(1, 1),
|
||||
0.0f
|
||||
);
|
||||
}
|
30
src/dawn/scene/components/ui/UIImage.hpp
Normal file
30
src/dawn/scene/components/ui/UIImage.hpp
Normal file
@ -0,0 +1,30 @@
|
||||
// Copyright (c) 2023 Dominic Masters
|
||||
//
|
||||
// This software is released under the MIT License.
|
||||
// https://opensource.org/licenses/MIT
|
||||
|
||||
#pragma once
|
||||
#include "UIComponent.hpp"
|
||||
#include "display/mesh/QuadMesh.hpp"
|
||||
|
||||
namespace Dawn {
|
||||
class UIImage : public UIComponent, public UIComponentRenderable {
|
||||
private:
|
||||
Mesh mesh;
|
||||
|
||||
public:
|
||||
struct Color color = COLOR_WHITE;
|
||||
// StateProperty<float_t> width;
|
||||
// StateProperty<float_t> height;
|
||||
StateProperty<Texture*> texture;
|
||||
|
||||
UIImage(SceneItem *item);
|
||||
|
||||
float_t getContentWidth() override;
|
||||
float_t getContentHeight() override;
|
||||
std::vector<struct ShaderPassItem> getPassItems(
|
||||
glm::mat4 proj, glm::mat4 view
|
||||
) override;
|
||||
void onStart() override;
|
||||
};
|
||||
}
|
@ -82,7 +82,6 @@ std::vector<struct ShaderPassItem> UILabel::getPassItems(
|
||||
}
|
||||
|
||||
float_t UILabel::getContentWidth() {
|
||||
if(this->maxWidth > 0) return this->maxWidth;
|
||||
if(!this->hasText()) return 0;
|
||||
this->updateMesh();
|
||||
return this->measure.getWidth();
|
||||
|
@ -14,4 +14,7 @@ target_include_directories(${DAWN_TARGET_NAME}
|
||||
|
||||
# Subdirs
|
||||
add_subdirectory(game)
|
||||
add_subdirectory(save)
|
||||
add_subdirectory(save)
|
||||
|
||||
# Assets
|
||||
tool_bitmapfont(testbitmap bmfont.png 16 16)
|
@ -6,24 +6,56 @@
|
||||
#pragma once
|
||||
#include "scene/Scene.hpp"
|
||||
#include "prefabs/SimpleSpinningCubePrefab.hpp"
|
||||
#include "scene/components/ui/UILabel.hpp"
|
||||
#include "scene/components/ui/UIImage.hpp"
|
||||
#include "display/font/BitmapFont.hpp"
|
||||
|
||||
namespace Dawn {
|
||||
class HelloWorldScene : public Scene {
|
||||
protected:
|
||||
Camera *camera;
|
||||
SimpleSpinningCubePrefab *cube;
|
||||
UICanvas *canvas;
|
||||
UILabel *label;
|
||||
UIImage *image;
|
||||
BitmapFont font;
|
||||
|
||||
void stage() override {
|
||||
camera = Camera::create(this);
|
||||
camera->transform->lookAt(glm::vec3(0, 0, 8), glm::vec3(0, 0, 0));
|
||||
|
||||
cube = SimpleSpinningCubePrefab::create(this);
|
||||
|
||||
canvas = UICanvas::create(this);
|
||||
|
||||
auto imageItem = this->createSceneItem();
|
||||
image = imageItem->addComponent<UIImage>();
|
||||
image->color = COLOR_BLACK;
|
||||
imageItem->transform.setParent(canvas->transform);
|
||||
|
||||
auto labelItem = this->createSceneItem();
|
||||
label = labelItem->addComponent<UILabel>();
|
||||
labelItem->transform.setParent(canvas->transform);
|
||||
|
||||
auto assMan = &this->game->assetManager;
|
||||
auto texture = assMan->get<TextureAsset>("testbitmap_texture");
|
||||
auto tileset = assMan->get<TilesetAsset>("testbitmap_tileset");
|
||||
this->font.texture = &texture->texture;
|
||||
this->font.tileset = &tileset->tileset;
|
||||
|
||||
label->text = "Hello World, how are you today? I hope you are doing well. I really like the fact I can ramble in my text for once.";
|
||||
label->font = &font;
|
||||
label->maxWidth = 275;
|
||||
|
||||
image->alignment = glm::vec4(0, 0, label->getContentWidth(), label->getContentHeight());
|
||||
}
|
||||
|
||||
std::vector<Asset*> getRequiredAssets() override {
|
||||
auto assMan = &this->game->assetManager;
|
||||
std::vector<Asset*> assets;
|
||||
vectorAppend(&assets, SimpleSpinningCubePrefab::getRequiredAssets(assMan));
|
||||
assets.push_back(assMan->get<TextureAsset>("testbitmap_texture"));
|
||||
assets.push_back(assMan->get<TilesetAsset>("testbitmap_tileset"));
|
||||
return assets;
|
||||
}
|
||||
|
||||
|
@ -69,6 +69,11 @@ void ShaderProgram::compileShader(
|
||||
glBindAttribLocation(this->shaderProgram, 1, "aTexCoord");
|
||||
}
|
||||
|
||||
void ShaderProgram::bindAttributeLocation(std::string name, int32_t location) {
|
||||
if(this->shaderProgram == -1) throw "Shader has not yet been compiled";
|
||||
glBindAttribLocation(this->shaderProgram, location, name.c_str());
|
||||
}
|
||||
|
||||
void ShaderProgram::setTexture(shaderparameter_t param, textureslot_t slot) {
|
||||
glUniform1i(param, slot);
|
||||
}
|
||||
|
@ -25,14 +25,23 @@ namespace Dawn {
|
||||
|
||||
protected:
|
||||
/**
|
||||
* Compiles a GLSL shader and stores it on the GPU, updates the underlying
|
||||
* pointers for you.
|
||||
* Compiles a GLSL/HLSL shader and stores it on the GPU, updates the
|
||||
* underlying pointers for you.
|
||||
*
|
||||
* @param vertexShader The string source of the vertex shader.
|
||||
* @param fragmentShader The string source of the fragment shader.
|
||||
*/
|
||||
void compileShader(std::string vertexShader, std::string fragmentShader);
|
||||
|
||||
/**
|
||||
* Typically HLSL only, this method allows you to specify where vbo
|
||||
* attributes are bound. Typically 0 for positions, 1 for coordinates,
|
||||
* etc.
|
||||
*
|
||||
* @param name Attribute name in the HLSL shader.
|
||||
* @param location Index pointing to which location it is to be bound to.
|
||||
*/
|
||||
void bindAttributeLocation(std::string name, int32_t location);
|
||||
|
||||
public:
|
||||
/**
|
||||
|
@ -83,6 +83,9 @@ namespace Dawn {
|
||||
"return o_Color;\n"
|
||||
"}\n"
|
||||
);
|
||||
|
||||
this->bindAttributeLocation("aPos", 0);
|
||||
this->bindAttributeLocation("aTexCoord", 1);
|
||||
#else
|
||||
#error Shader Type must be either GLSL or HLSL
|
||||
#endif
|
||||
|
@ -36,6 +36,11 @@ function(tool_tileset targetTileset targetTexture in cols rows)
|
||||
add_dependencies(${DAWN_TARGET_NAME} ${targetTileset})
|
||||
endfunction()
|
||||
|
||||
# Bitmap Font
|
||||
function(tool_bitmapfont target in columns rows)
|
||||
tool_tileset(${target}_tileset ${target}_texture ${in} ${columns} ${rows})
|
||||
endfunction()
|
||||
|
||||
# TrueType Tool
|
||||
function(tool_truetype target in out width height fontSize)
|
||||
add_custom_target(${target}
|
||||
@ -46,7 +51,7 @@ function(tool_truetype target in out width height fontSize)
|
||||
add_dependencies(${DAWN_TARGET_NAME} ${target})
|
||||
endfunction()
|
||||
|
||||
# UI Tool
|
||||
# UI Tool
|
||||
function(tool_ui target in)
|
||||
add_custom_target(${target}
|
||||
COMMAND uigen --input="${DAWN_ASSETS_SOURCE_DIR}/${in}" --output="${DAWN_GENERATED_DIR}/prefabs/ui/${target}"
|
||||
|
Reference in New Issue
Block a user