Prepping label statistics.
This commit is contained in:
@ -89,7 +89,7 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
struct FontShaderBufferData fontData;
|
struct FontShaderBufferData fontData;
|
||||||
quadCountTotal = 0;
|
quadCountTotal = 0;
|
||||||
std::vector<struct UILabelText> realNewTexts;
|
std::vector<struct UILabelText> realNewTexts;
|
||||||
|
|
||||||
// Determine font dimensions.
|
// Determine font dimensions.
|
||||||
auto itText = newTexts.begin();
|
auto itText = newTexts.begin();
|
||||||
while(itText != newTexts.end()) {
|
while(itText != newTexts.end()) {
|
||||||
@ -131,24 +131,54 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
realText.texture->texture.getHeight()
|
realText.texture->texture.getHeight()
|
||||||
);
|
);
|
||||||
|
|
||||||
// Now, iterate each character
|
// Prepare loop properties and shorthands
|
||||||
auto len = text.text.length();
|
auto len = text.text.length();
|
||||||
|
float_t lineWidth = 0;
|
||||||
|
float_t wordWidth = 0;
|
||||||
int32_t lastSpaceCharacter = -1;
|
int32_t lastSpaceCharacter = -1;
|
||||||
for(int32_t i = 0; i < len; i++) {
|
|
||||||
char ch = text.text[i];
|
|
||||||
|
|
||||||
// Handle newline
|
struct UILabelLine currentLine;
|
||||||
if(ch == '\n') {
|
|
||||||
position.x = 0;
|
|
||||||
position.y += realText.style.size;
|
|
||||||
|
|
||||||
|
std::function<void(int32_t)> fnInsertNewline = [&](int32_t i){
|
||||||
|
if(i != len) {
|
||||||
|
// Update text.
|
||||||
|
realText.text += '\n';
|
||||||
|
|
||||||
|
// Insert dummy quad
|
||||||
glm::vec4 uvs(0, 0, 1, 1);
|
glm::vec4 uvs(0, 0, 1, 1);
|
||||||
glm::vec4 vert(0, 0, 0, 0);
|
glm::vec4 vert(0, 0, 0, 0);
|
||||||
vertices.push_back(std::make_pair(vert, uvs));
|
vertices.push_back(std::make_pair(vert, uvs));
|
||||||
fontData.fontQuadMappings[quadCountTotal] = partIndex;
|
fontData.fontQuadMappings[quadCountTotal] = partIndex;
|
||||||
|
|
||||||
|
currentLine.quadCount++;
|
||||||
quadCountTotal++;
|
quadCountTotal++;
|
||||||
realText.text += ch;
|
}
|
||||||
lastSpaceCharacter = i;
|
|
||||||
|
// Finalize current line
|
||||||
|
lineWidth += wordWidth;
|
||||||
|
currentLine.width = lineWidth;
|
||||||
|
|
||||||
|
// Move to next line
|
||||||
|
position.x = 0;
|
||||||
|
position.y += realText.style.size;
|
||||||
|
lines.push_back(currentLine);
|
||||||
|
|
||||||
|
// Reset line
|
||||||
|
lastSpaceCharacter = i;
|
||||||
|
wordWidth = 0.0f;
|
||||||
|
lineWidth = 0.0f;
|
||||||
|
|
||||||
|
currentLine = UILabelLine();
|
||||||
|
currentLine.quadStart = quadCountTotal;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Now, iterate each character
|
||||||
|
for(int32_t i = 0; i < len; i++) {
|
||||||
|
char ch = text.text[i];
|
||||||
|
|
||||||
|
// Handle special characters
|
||||||
|
if(ch == '\n') {
|
||||||
|
fnInsertNewline(i);
|
||||||
continue;
|
continue;
|
||||||
} else if(ch == ' ') {
|
} else if(ch == ' ') {
|
||||||
lastSpaceCharacter = i;
|
lastSpaceCharacter = i;
|
||||||
@ -160,6 +190,7 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
assertTrue(ch != '\t');
|
assertTrue(ch != '\t');
|
||||||
assertTrue(ch != '\n');
|
assertTrue(ch != '\n');
|
||||||
|
|
||||||
|
// Get font data.
|
||||||
FT_ULong c = ch;
|
FT_ULong c = ch;
|
||||||
auto charInfo = realText.texture->getCharacterData(c);
|
auto charInfo = realText.texture->getCharacterData(c);
|
||||||
|
|
||||||
@ -170,12 +201,17 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
this->width > charInfo.bitmapSize.x &&
|
this->width > charInfo.bitmapSize.x &&
|
||||||
(position.x + charInfo.advanceX) > this->width
|
(position.x + charInfo.advanceX) > this->width
|
||||||
) {
|
) {
|
||||||
|
// Basically this rewinds everything we've done to the last space char,
|
||||||
|
// changes it to a newline, and then moves the position along.
|
||||||
int32_t diff = i - lastSpaceCharacter;
|
int32_t diff = i - lastSpaceCharacter;
|
||||||
for(int32_t k = 0; k < diff; k++) vertices.pop_back();
|
for(int32_t k = 0; k < diff; k++) vertices.pop_back();
|
||||||
text.text[lastSpaceCharacter] = '\n';
|
text.text[lastSpaceCharacter] = '\n';
|
||||||
i = lastSpaceCharacter - 1;
|
i = lastSpaceCharacter;
|
||||||
lastSpaceCharacter = -1;
|
lastSpaceCharacter = -1;
|
||||||
quadCountTotal -= diff;
|
quadCountTotal -= diff;
|
||||||
|
|
||||||
|
// Now we've rewound to the space, treat it like a newline instead.
|
||||||
|
fnInsertNewline(i);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -192,19 +228,31 @@ void UILabel::rebufferQuads(const std::vector<struct UILabelText> newTexts) {
|
|||||||
vert.w = vert.x + charInfo.bitmapSize.x;
|
vert.w = vert.x + charInfo.bitmapSize.x;
|
||||||
vert.z = vert.y + charInfo.bitmapSize.y;
|
vert.z = vert.y + charInfo.bitmapSize.y;
|
||||||
|
|
||||||
// TODO:Check wordwrap here
|
|
||||||
vertices.push_back(std::make_pair(vert, uvs));
|
vertices.push_back(std::make_pair(vert, uvs));
|
||||||
|
|
||||||
// Move the current position along.
|
// Move the current position along.
|
||||||
position.x += charInfo.advanceX;
|
position.x += charInfo.advanceX;
|
||||||
position.y += charInfo.advanceY;
|
position.y += charInfo.advanceY;
|
||||||
|
|
||||||
|
// Update the continuous dimensions
|
||||||
|
if(ch == ' ') {
|
||||||
|
lineWidth += wordWidth;
|
||||||
|
lineWidth += charInfo.advanceX;
|
||||||
|
wordWidth = 0.0f;
|
||||||
|
} else {
|
||||||
|
wordWidth += charInfo.advanceX;
|
||||||
|
}
|
||||||
|
|
||||||
// Set the part index to the quad mappings
|
// Set the part index to the quad mappings
|
||||||
fontData.fontQuadMappings[quadCountTotal] = partIndex;
|
fontData.fontQuadMappings[quadCountTotal] = partIndex;
|
||||||
quadCountTotal++;
|
quadCountTotal++;
|
||||||
realText.text += ch;
|
realText.text += ch;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Now we insert a line. We do this because there is no newline at the end
|
||||||
|
// of the text, so we need to insert the last line manually.
|
||||||
|
fnInsertNewline(len);
|
||||||
|
|
||||||
// Next
|
// Next
|
||||||
++partIndex;
|
++partIndex;
|
||||||
++itText;
|
++itText;
|
||||||
|
@ -22,27 +22,27 @@ namespace Dawn {
|
|||||||
struct UILabelText {
|
struct UILabelText {
|
||||||
std::string text;
|
std::string text;
|
||||||
struct UILabelStyle style;
|
struct UILabelStyle style;
|
||||||
|
int32_t lineStart = 0;
|
||||||
glm::vec2 position;
|
int32_t lineCount = 0;
|
||||||
glm::vec2 size;
|
|
||||||
|
|
||||||
// Part index?
|
|
||||||
// Quad start?
|
|
||||||
// quad count?
|
|
||||||
// position?
|
|
||||||
// size?
|
|
||||||
// some kind of custom data e.g. wobble or shake?
|
|
||||||
|
|
||||||
usagelockid_t lockId = -1;
|
usagelockid_t lockId = -1;
|
||||||
struct NewTrueTypeFaceTexture *texture = nullptr;
|
struct NewTrueTypeFaceTexture *texture = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct UILabelLine {
|
||||||
|
float_t width = 0.0f;
|
||||||
|
int32_t quadStart = -1;
|
||||||
|
int32_t quadCount = 0;
|
||||||
|
|
||||||
|
std::vector<struct UILabelText*> texts;
|
||||||
|
};
|
||||||
|
|
||||||
class UILabel : public UIComponentRenderable {
|
class UILabel : public UIComponentRenderable {
|
||||||
private:
|
private:
|
||||||
Mesh mesh;
|
Mesh mesh;
|
||||||
FontShaderBuffer shaderBuffer;
|
FontShaderBuffer shaderBuffer;
|
||||||
std::vector<struct UILabelText> texts;
|
std::vector<struct UILabelText> texts;
|
||||||
std::vector<struct UILabelText> textsBuffered;
|
std::vector<struct UILabelText> textsBuffered;
|
||||||
|
std::vector<struct UILabelLine> lines;
|
||||||
std::map<NewTrueTypeFaceTexture*, int32_t> textureMap;
|
std::map<NewTrueTypeFaceTexture*, int32_t> textureMap;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
@ -30,21 +30,23 @@ namespace Dawn {
|
|||||||
newLabel->alignment = glm::vec4(0, 0, 0, 0);
|
newLabel->alignment = glm::vec4(0, 0, 0, 0);
|
||||||
newLabel->alignX = UI_COMPONENT_ALIGN_STRETCH;
|
newLabel->alignX = UI_COMPONENT_ALIGN_STRETCH;
|
||||||
newLabel->alignY = UI_COMPONENT_ALIGN_STRETCH;
|
newLabel->alignY = UI_COMPONENT_ALIGN_STRETCH;
|
||||||
|
|
||||||
// newLabel->maxWidth = 300.0f;
|
// newLabel->maxWidth = 300.0f;
|
||||||
newLabel->richText = std::string(
|
newLabel->richText = std::string(
|
||||||
"<font font=\"font_arial\" size=\"16\" color=\"COLOR_BLUE\">"
|
"<font font=\"font_arial\" size=\"32\" color=\"COLOR_BLUE\">"
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
"Hello\nWorld"
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
|
// "Lorem ipsum dolor sit amet, consectetur adipiscing elit. "
|
||||||
"</font>"
|
"</font>"
|
||||||
);
|
);
|
||||||
newLabelItem->transform.setParent(canvas->transform);
|
newLabelItem->transform.setParent(canvas->transform);
|
||||||
|
Reference in New Issue
Block a user