TDME2 1.9.121
GUIStyledTextNode.cpp
Go to the documentation of this file.
2
3#include <list>
4#include <string>
5#include <string_view>
6
7#include <tdme/tdme.h>
10#include <tdme/engine/Engine.h>
22#include <tdme/gui/GUI.h>
23#include <tdme/math/Math.h>
29
30using std::list;
31using std::string;
32using std::string_view;
33using std::to_string;
34
50using tdme::gui::GUI;
57
58GUIStyledTextNode::GUIStyledTextNode(
59 GUIScreenNode* screenNode,
60 GUIParentNode* parentNode,
61 const string& id,
62 GUINode_Flow* flow,
63 const GUINode_Alignments& alignments,
64 const GUINode_RequestedConstraints& requestedConstraints,
65 const GUIColor& backgroundColor,
66 const string& backgroundImage,
67 const GUINode_Scale9Grid& backgroundImageScale9Grid,
68 const GUIColor& backgroundImageEffectColorMul,
69 const GUIColor& backgroundImageEffectColorAdd,
70 const GUINode_Border& border,
71 const GUINode_Padding& padding,
72 const GUINodeConditions& showOn,
73 const GUINodeConditions& hideOn,
74 bool preformatted,
75 const string& font,
76 const string& color,
77 const MutableString& text
78):
79 GUINode(screenNode, parentNode, id, flow, alignments, requestedConstraints, backgroundColor, backgroundImage, backgroundImageScale9Grid, backgroundImageEffectColorMul, backgroundImageEffectColorAdd, border, padding, showOn, hideOn)
80{
81 this->font = font.empty() == true?nullptr:GUI::getFont(screenNode->getApplicationRootPathName(), font);
82 this->color = color.empty() == true || color.length() == 0?GUIColor():GUIColor(color);
83 this->autoWidth = 0;
84 this->autoHeight = 0;
85 this->parentOffsetsChanged = true;
86 this->parentXOffsetLast = 0.0f;
87 this->parentYOffsetLast = 0.0f;
88 this->startRenderY = 0;
89 this->charStartIdx = 0;
90 this->charEndIdx = text.size() - 1;
91 this->widthLast = -1;
92 this->heightLast = -1;
93 this->startTextStyleIdx = -1;
94 this->preformatted = preformatted;
95 if (this->font != nullptr) this->font->initialize();
97}
98
100{
101 return "multiline-text";
102}
103
105{
106 return true;
107}
108
110{
112 return font != nullptr?autoWidth + border.left + border.right + padding.left + padding.right:0;
113 } else {
115 }
116}
117
120 return font != nullptr?autoHeight + border.top + border.bottom + padding.top + padding.bottom:0;
121 } else {
123 }
124}
125
127 // If fixed width requested and no computed constraints yet, abort
129 // width did not change, but relayout has been requested
131 // no font, exit
132 if (font == nullptr) return;
133
134 //
135 auto maxLineWidth = getAutoWidth();
136
137 //
138 autoWidth = 0;
139 autoHeight = 0;
140
141 //
142 auto textStyleIdx = 0;
143 for (auto i = 0; i < text.size(); ) {
144 //
145 determineNextLineConstraints(i, text.size(), textStyleIdx);
146
147 /*
148 {
149 auto l = 0;
150 for (auto k = 0; k < lineConstraints.size(); k++) {
151 string linePart;
152 for (auto j = l; j < lineConstraints[k].idx; j++) {
153 if (line[j] == 0) {
154 linePart += "[image]";
155 } else {
156 linePart += line[j];
157 }
158 }
159 Console::println(
160 "auto line@" + to_string(k) + ": '" + line + "': '" + linePart
161 + "': " + to_string(lineConstraints[k].idx) + "; width = "
162 + to_string(lineConstraints[k].width) + " / "
163 + to_string(maxLineWidth) + ", line height = "
164 + to_string(lineConstraints[k].lineHeight) + ", height "
165 + to_string(lineConstraints[k].height) + ", base line: "
166 + to_string(lineConstraints[k].baseLine) + ": Y = " + to_string(autoHeight));
167 l = lineConstraints[k].idx;
168 }
169 }
170 */
171
172 //
173 for (auto& lineConstraintsEntity: lineConstraints) {
174 if (lineConstraintsEntity.width > autoWidth) autoWidth = lineConstraintsEntity.width;
175 autoHeight+= lineConstraintsEntity.height;
176 }
177
178 //
179 line.clear();
180 lineCharIdxs.clear();
181 lineConstraints.clear();
182 }
183
184 //
185 this->parentOffsetsChanged = true;
188}
189
191 this->parentOffsetsChanged = true;
192 this->parentXOffsetLast = 0.0f;
193 this->parentYOffsetLast = 0.0f;
194 this->charStartIdx = 0;
195 this->charEndIdx = text.size() - 1;
196 this->startRenderY = 0;
197 this->widthLast = -1;
198 this->heightLast = -1;
199 this->startTextStyleIdx = -1;
202 this->text.reset();
203 /*
204 // Currently supported BBCode like tags
205 [font=schriftart]Text[/font]
206 [color=farbe]farbiger Text[/color]
207 [url=http://example.com/]Linktext[/url]
208 [image]example.com/bild.jpg[/image] (Bearbeitet)
209 */
210 string styleFont;
211 string styleColor;
212 string styleUrl;
213 string styleImage;
214 //
215 auto parseStyle = false;
216 auto parseImage = false;
217 string currentStyle;
218 int styleStartIdx = -1;
219 char lc = 0;
220 for (auto i = 0; i < text.size(); i++) {
221 auto c = text.charAt(i);
222 if (parseStyle == true) {
223 // end of style
224 if (c == ']') {
225 if (lc != '\\') {
226 auto styleTokenized = StringTools::tokenize(currentStyle, "=");
227 // apply style until current text size
228 if (styleStartIdx != -1 &&
229 (styleFont.empty() == false ||
230 styleColor.empty() == false ||
231 styleUrl.empty() == false)) {
232 if (styleColor.empty() == false) {
233 if (this->text.size() > styleStartIdx) setTextStyle(styleStartIdx, this->text.size() - 1, GUIColor(styleColor), styleFont, styleUrl);
234 } else {
235 if (this->text.size() > styleStartIdx) setTextStyle(styleStartIdx, this->text.size() - 1, styleFont, styleUrl);
236 }
237 }
238 if (styleTokenized.size() == 2) {
239 auto command = StringTools::toLowerCase(StringTools::trim(styleTokenized[0]));
240 auto argument = StringTools::trim(styleTokenized[1]);
241 if (command == "font") {
242 styleFont = argument;
243 styleStartIdx = this->text.size();
244 } else
245 if (command == "color") {
246 styleColor = argument;
247 styleStartIdx = this->text.size();
248 } else
249 if (command == "url") {
250 styleUrl = argument;
251 styleStartIdx = this->text.size();
252 } else {
253 Console::println("GUIStyledTextNode::setText(): unknown style command: " + currentStyle);
254 }
255 } else
256 if (styleTokenized.size() == 1) {
257 auto command = StringTools::toLowerCase(StringTools::trim(styleTokenized[0]));
258 if (command == "/font") {
259 styleFont.clear();
260 } else
261 if (command == "/color") {
262 styleColor.clear();
263 } else
264 if (command == "/url") {
265 styleUrl.clear();
266 } else
267 if (command == "image") {
268 parseImage = true;
269 } else
270 if (command == "/image") {
271 parseImage = false;
272 this->text.append(static_cast<char>(0));
273 setImage(this->text.size() - 1, styleImage, styleUrl, -1, -1);
274 styleImage.clear();
275 } else {
276 Console::println("GUIStyledTextNode::setText(): unknown style command: " + currentStyle);
277 }
278 } else {
279 Console::println("GUIStyledTextNode::setText(): unknown style command: " + currentStyle);
280 }
281 //
282 currentStyle.clear();
283 parseStyle = false;
284 if (styleFont.empty() == false ||
285 styleColor.empty() == false ||
286 styleUrl.empty() == false) {
287 styleStartIdx = this->text.size();
288 }
289 } else {
290 this->text.remove(this->text.size() - 1, 1);
291 currentStyle+= c;
292 }
293 } else {
294 // style command
295 currentStyle+= c;
296 }
297 } else
298 // start of style
299 if (c == '[') {
300 if (lc != '\\') {
301 parseStyle = true;
302 } else {
303 this->text.remove(this->text.size() - 1, 1);
304 this->text.append(c);
305 }
306 } else {
307 if (c == ']' && lc == '\\') {
308 this->text.remove(this->text.size() - 1, 1);
309 }
310 if (parseImage == true) {
311 // image
312 styleImage+= c;
313 } else {
314 // ordinary text
315 this->text.append(c);
316 }
317 }
318 //
319 lc = c;
320 }
321 // apply style until current text size
322 if (styleStartIdx != -1 &&
323 (styleFont.empty() == false ||
324 styleColor.empty() == false ||
325 styleUrl.empty() == false)) {
326 if (styleColor.empty() == false) {
327 if (this->text.size() > styleStartIdx) setTextStyle(styleStartIdx, this->text.size() - 1, GUIColor(styleColor), styleFont, styleUrl);
328 } else {
329 if (this->text.size() > styleStartIdx) setTextStyle(styleStartIdx, this->text.size() - 1, styleFont, styleUrl);
330 }
331 }
332 //
333 Console::println("'" + this->text.getString() + "'");
334}
335
337{
339 if (font != nullptr) font->dispose();
341}
342
343void GUIStyledTextNode::determineNextLineConstraints(int& i, int charEndIdx, int textStyleIdx) {
344 //
346
347 // determine line to render
348 if (preformatted == true) {
349 line.clear();
350 lineCharIdxs.clear();
351 auto k = i;
352 for (; k < charEndIdx; k++) {
353 auto c = text.charAt(k);
354 // line finished?
355 if (c == '\n') {
356 break;
357 } else
358 if (c == '\t') {
359 // extend tab to 4 spaces if line is not empty
361 lineCharIdxs.push_back(k);
362 lineCharIdxs.push_back(k);
363 lineCharIdxs.push_back(k);
364 lineCharIdxs.push_back(k);
365 } else {
366 line+= c;
367 lineCharIdxs.push_back(k);
368 }
369 }
370 i = k + 1;
371 } else {
372 line.clear();
373 lineCharIdxs.clear();
374 auto k = i;
375 for (; k < charEndIdx; k++) {
376 auto c = text.charAt(k);
377 // line finished?
378 if (c == '\n') {
379 break;
380 } else
381 if (line.empty() == false && c == ' ' && StringTools::endsWith(line, spaceString) == true) {
382 // no op as we have a line which already has a space at the end
383 } else
384 if (line.empty() == true && (c == ' ' || c == '\t')) {
385 // no op
386 } else
387 if (c == '\t') {
388 // extend tab to 4 spaces if line is not empty
389 if (line.empty() == false) {
390 if (StringTools::endsWith(line, spaceString) == true) {
392 } else {
394 }
395 lineCharIdxs.push_back(k);
396 lineCharIdxs.push_back(k);
397 lineCharIdxs.push_back(k);
398 lineCharIdxs.push_back(k);
399 }
400 } else {
401 line+= c;
402 lineCharIdxs.push_back(k);
403 }
404 }
405 i = k + 1;
406 }
407 // remove trailing space
408 while (StringTools::endsWith(line, spaceString) == true) line.erase(line.begin() + line.size() - 1);
409
410 if (line.empty() == true) {
411 //
412 lineConstraints.push_back(
413 {
414 .idx = 0,
415 .width = 0.0f,
416 .height = font->getLineHeight(),
417 .lineHeight = font->getLineHeight(),
418 .baseLine = font->getBaseLine(),
419 .spaceWrap = true
420 }
421 );
422 } else {
423 // determine baseline and part of line to render
424 auto baseLine = 0.0f;
425 auto lineHeight = 0.0f;
426 auto lineWidth = 0.0f;
427 auto lineWidthSpaceWrap = 0.0f;
428 auto lineHeightSpaceWrap = 0.0f;
429 auto baseLineSpaceWrap = 0.0f;
430 auto imageHeight = 0.0f;
431
432 //
433 lineConstraints.clear();
434 lineConstraints.push_back(
435 {
436 .idx = -1,
437 .width = 0.0f,
438 .height = 0.0f,
439 .lineHeight = 0.0f,
440 .baseLine = 0.0f,
441 .spaceWrap = false
442 }
443 );
444 {
445 auto currentTextStyleIdx = textStyleIdx;
446 for (auto k = 0; k < line.size(); k++) {
447 auto textStyle = getTextStyle(lineCharIdxs, k, currentTextStyleIdx);
448 auto currentFont = textStyle != nullptr && textStyle->font != nullptr?textStyle->font:font;
449 baseLine = Math::max(baseLine, currentFont->getBaseLine());
450 baseLineSpaceWrap = Math::max(baseLineSpaceWrap, currentFont->getBaseLine());
451 lineHeight = Math::max(lineHeight, currentFont->getLineHeight());
452 lineHeightSpaceWrap = Math::max(lineHeightSpaceWrap, currentFont->getLineHeight());
453 // render a image
454 if (textStyle != nullptr && textStyle->image != nullptr) {
455 if (lineConstraints[lineConstraints.size() - 1].spaceWrap == false) {
456 lineConstraints[lineConstraints.size() - 1] = {
457 .idx = k,
458 .width = lineWidth,
459 .height = Math::max(lineHeight, baseLine + imageHeight),
460 .lineHeight = lineHeight,
461 .baseLine = baseLine,
462 .spaceWrap = false
463 };
464 lineWidthSpaceWrap = 0.0f;
465 lineHeightSpaceWrap = 0.0f;
466 baseLineSpaceWrap = 0.0f;
467 }
468 if (lineWidth > maxLineWidth) {
469 imageHeight = 0.0f;
470 lineWidth = lineWidthSpaceWrap;
471 lineHeight = lineHeightSpaceWrap;
472 baseLine = baseLineSpaceWrap;
473 lineConstraints.push_back(
474 {
475 .idx = -1,
476 .width = 0.0f,
477 .height = 0.0f,
478 .lineHeight = 0.0f,
479 .baseLine = 0.0f,
480 .spaceWrap = false
481 }
482 );
483 }
484 lineWidth+= textStyle->width;
485 lineWidthSpaceWrap+= textStyle->width;
486 imageHeight = Math::max(imageHeight, static_cast<float>(textStyle->height));
487 } else {
488 // render text
489 if (line[k] == ' ') {
490 lineConstraints[lineConstraints.size() - 1] = {
491 .idx = k,
492 .width = lineWidth,
493 .height = Math::max(lineHeight, baseLine + imageHeight),
494 .lineHeight = lineHeight,
495 .baseLine = baseLine,
496 .spaceWrap = true
497 };
498 lineWidthSpaceWrap = 0.0f;
499 lineHeightSpaceWrap = 0.0f;
500 baseLineSpaceWrap = 0.0f;
501 }
502 auto character = currentFont->getCharacter(line[k]);
503 if (character != nullptr) {
504 if (lineConstraints[lineConstraints.size() - 1].spaceWrap == false) {
505 lineConstraints[lineConstraints.size() - 1] = {
506 .idx = k,
507 .width = lineWidth,
508 .height = Math::max(lineHeight, baseLine + imageHeight),
509 .lineHeight = lineHeight,
510 .baseLine = baseLine,
511 .spaceWrap = false
512 };
513 lineWidthSpaceWrap = 0.0f;
514 lineHeightSpaceWrap = 0.0f;
515 baseLineSpaceWrap = 0.0f;
516 }
517 if (lineWidth > maxLineWidth) {
518 lineWidth = lineWidthSpaceWrap;
519 if (k != line.size() - 1) {
520 imageHeight = 0.0f;
521 lineHeight = lineHeightSpaceWrap;
522 baseLine = baseLineSpaceWrap;
523 }
524 lineConstraints.push_back(
525 {
526 .idx = -1,
527 .width = 0.0f,
528 .height = 0.0f,
529 .lineHeight = 0.0f,
530 .baseLine = 0.0f,
531 .spaceWrap = false
532 }
533 );
534 }
535 lineWidth+= character->getXAdvance();
536 lineWidthSpaceWrap+= lineWidthSpaceWrap < Math::EPSILON && line[k] == ' '?0.0f:character->getXAdvance();
537 }
538 }
539 }
540 }
541
542 //
543 lineConstraints[lineConstraints.size() - 1] = {
544 .idx = static_cast<int>(line.size()),
545 .width = lineWidth,
546 .height = Math::max(lineHeight, baseLine + imageHeight),
547 .lineHeight = lineHeight,
548 .baseLine = baseLine,
549 .spaceWrap = false
550 };
551 }
552}
553
555{
556 if (shouldRender() == false) return;
557
558 //
559 GUINode::render(guiRenderer);
560
561 //
562 if (font == nullptr) return;
563
564 // screen dimensions
565 auto screenWidth = screenNode->getScreenWidth();
566 auto screenHeight = screenNode->getScreenHeight();
567
568 // indents
571
572 // vertical alignment
573 auto y = 0;
575 // no op
576 } else
579 } else
582 }
583
584 //
585 auto parentXOffset = computeParentChildrenRenderOffsetXTotal();
586 auto parentYOffset = computeParentChildrenRenderOffsetYTotal();
587 bool visible = false;
588
589 // did a scrolling appear, then reset bounds to work with
590 if (parentOffsetsChanged == true ||
591 Math::abs(parentXOffset - parentXOffsetLast) > Math::EPSILON ||
592 Math::abs(parentYOffset - parentYOffsetLast) > Math::EPSILON) {
593 parentXOffsetLast = parentXOffset;
594 parentYOffsetLast = parentYOffset;
595 charStartIdx = 0;
596 charEndIdx = text.size();
598 parentOffsetsChanged = false;
599 startRenderY = 0;
600 } else {
601 y = startRenderY;
602 visible = true;
603 }
604
605 // Console::println("char start idx: " + to_string(charStartIdx) + ", char end idx: " + to_string(charEndIdx) + ", chars: " + to_string(text.size()) + ", start text style idx: " + to_string(startTextStyleIdx) + ", start render y: " + to_string(startRenderY) + ", auto width: " + to_string(autoWidth) + ", auto height = " + to_string(autoHeight))
606
607 //
608 auto maxLineWidth = getAutoWidth();
609 auto textStyleIdx = startTextStyleIdx;
610 auto currentCharStartIdx = charStartIdx;
611 auto j = charStartIdx;
612 auto boundTexture = -1;
613 GUIColor lastColor = color;
614 for (auto i = charStartIdx; i < charEndIdx;) {
615 //
616 currentCharStartIdx = i;
618
619 // empty lines are cheap, handle them immediatly
620 if (line.empty() == true) {
621 for (auto& lineConstraint: lineConstraints) {
622 y+= lineConstraint.height;
623 }
624 //
625 line.clear();
626 lineCharIdxs.clear();
627 lineConstraints.clear();
628 //
629 continue;
630 }
631
632 //
633 /*
634 {
635 auto l = 0;
636 for (auto k = 0; k < lineConstraints.size(); k++) {
637 string linePart;
638 for (auto j = l; j < lineConstraints[k].idx; j++) {
639 if (line[j] == 0) {
640 linePart += "[image]";
641 } else {
642 linePart += line[j];
643 }
644 }
645 Console::println(
646 "render line@" + to_string(k) + ": '" + line + "': '" + linePart
647 + "': " + to_string(lineConstraints[k].idx) + "; width = "
648 + to_string(lineConstraints[k].width) + " / "
649 + to_string(maxLineWidth) + ", line height = "
650 + to_string(lineConstraints[k].lineHeight) + ", height "
651 + to_string(lineConstraints[k].height) + ", base line: "
652 + to_string(lineConstraints[k].baseLine) + ": Y = " + to_string(y));
653 l = lineConstraints[k].idx;
654 }
655 }
656 */
657
658 // render
659 auto lineIdx = 0;
660 {
661 //
662 auto skipSpaces = false;
663 auto& currentTextStyleIdx = textStyleIdx;
664 auto x = 0;
665 // determine visibility of (sub) lines
666 for (lineIdx = 0; lineIdx < lineConstraints.size(); lineIdx++) {
667 // x alignment
669 x = 0;
670 } else
672 x = (maxLineWidth - lineConstraints[lineIdx].width) / 2;
673 } else
675 x = maxLineWidth - lineConstraints[lineIdx].width;
676 }
677 //
678 float left = x + xIndentLeft;
679 float top = y + yIndentTop;
680 float width = lineConstraints[lineIdx].width;
681 float height = lineConstraints[lineIdx].height;
682 if (guiRenderer->isQuadVisible(
683 ((left) / (screenWidth / 2.0f)) - 1.0f,
684 ((screenHeight - top) / (screenHeight / 2.0f)) - 1.0f,
685 ((left + width) / (screenWidth / 2.0f)) - 1.0f,
686 ((screenHeight - top) / (screenHeight / 2.0f)) - 1.0f,
687 ((left + width) / (screenWidth / 2.0f)) - 1.0f,
688 ((screenHeight - top - height) / (screenHeight / 2.0f)) - 1.0f,
689 ((left) / (screenWidth / 2.0f)) - 1.0f, ((screenHeight - top - height) / (screenHeight / 2.0f)) - 1.0f) == true) {
690 // break if visible
691 break;
692 }
693 // increment y by line height
694 y+= lineConstraints[lineIdx].height;
695 // iterate text style
696 for (auto k = lineIdx == 0?0:lineConstraints[lineIdx - 1].idx; k < lineConstraints[lineIdx].idx; k++)
697 getTextStyle(lineCharIdxs, k, currentTextStyleIdx);
698 }
699 // render
700 if (lineIdx == lineConstraints.size()) {
701 // was visible, then store text render end values
702 if (visible == true) {
703 visible = false;
705 break;
706 }
707 } else {
708 // if text was not visible before store text render start values
709 if (visible == false) {
710 visible = true;
711 charStartIdx = lineCharIdxs[lineIdx == 0?0:lineConstraints[lineIdx - 1].idx];
712 startTextStyleIdx = currentTextStyleIdx;
713 startRenderY = y;
714 }
715 // x alignment
717 // no op
718 x = 0;
719 } else
721 x = (maxLineWidth - lineConstraints[lineIdx].width) / 2;
722 } else
724 x = maxLineWidth - lineConstraints[lineIdx].width;
725 }
726 // render
727 for (auto k = lineIdx == 0?0:lineConstraints[lineIdx - 1].idx; k < line.size(); k++) {
728 auto textStyle = getTextStyle(lineCharIdxs, k, currentTextStyleIdx);
729 Color4 currentColor = color;
730 GUIFont* currentFont = font;
731 // apply text style or defaults
732 if (textStyle != nullptr) {
733 currentFont = textStyle->font != nullptr?textStyle->font:font;
734 currentColor = textStyle->color;
735 }
736 if (textStyle != nullptr && textStyle->image != nullptr) {
737 guiRenderer->render();
738 guiRenderer->bindTexture(textStyle->textureId);
739 float left = x + xIndentLeft;
740 float top = y + yIndentTop + (lineConstraints[lineIdx].baseLine - textStyle->height) + (lineConstraints[lineIdx].height - lineConstraints[lineIdx].lineHeight);
741 float width = textStyle->width;
742 float height = textStyle->height;
743 guiRenderer->addQuad(
744 ((left) / (screenWidth / 2.0f)) - 1.0f,
745 ((screenHeight - top) / (screenHeight / 2.0f)) - 1.0f,
746 1.0f, 1.0f, 1.0f, 1.0f,
747 0.0f,
748 0.0f,
749 ((left + width) / (screenWidth / 2.0f)) - 1.0f,
750 ((screenHeight - top) / (screenHeight / 2.0f)) - 1.0f,
751 1.0f, 1.0f, 1.0f, 1.0f,
752 1.0f,
753 0.0f,
754 ((left + width) / (screenWidth / 2.0f)) - 1.0f,
755 ((screenHeight - top - height) / (screenHeight / 2.0f)) - 1.0f,
756 1.0f, 1.0f, 1.0f, 1.0f,
757 1.0f,
758 1.0f,
759 ((left) / (screenWidth / 2.0f)) - 1.0f,
760 ((screenHeight - top - height) / (screenHeight / 2.0f)) - 1.0f,
761 1.0f, 1.0f, 1.0f, 1.0f,
762 0.0f,
763 1.0f
764 );
765 guiRenderer->render();
766 guiRenderer->bindTexture(boundTexture);
767 x+= textStyle->width;
768 } else {
769 // do line break
770 if (k == lineConstraints[lineIdx].idx) {
771 y+= lineConstraints[lineIdx].height;
772 lineIdx++;
773 if (lineIdx == lineConstraints.size()) break;
774 //
775 x = 0;
777 // no op
778 } else
780 x = (maxLineWidth - lineConstraints[lineIdx].width) / 2;
781 } else
783 x = maxLineWidth - lineConstraints[lineIdx].width;
784 }
785 if (lineConstraints[lineIdx - 1].spaceWrap == true) {
786 skipSpaces = true;
787 }
788 //
789 {
790 float left = x + xIndentLeft;
791 float top = y + yIndentTop;
792 float width = lineConstraints[lineIdx].width;
793 float height = lineConstraints[lineIdx].height;
794 if (guiRenderer->isQuadVisible(
795 ((left) / (screenWidth / 2.0f)) - 1.0f,
796 ((screenHeight - top) / (screenHeight / 2.0f)) - 1.0f,
797 ((left + width) / (screenWidth / 2.0f)) - 1.0f,
798 ((screenHeight - top) / (screenHeight / 2.0f)) - 1.0f,
799 ((left + width) / (screenWidth / 2.0f)) - 1.0f,
800 ((screenHeight - top - height) / (screenHeight / 2.0f)) - 1.0f,
801 ((left) / (screenWidth / 2.0f)) - 1.0f, ((screenHeight - top - height) / (screenHeight / 2.0f)) - 1.0f) == false) {
802 //
803 if (visible == true) {
804 visible = false;
805 charEndIdx = lineCharIdxs[lineIdx == 0?0:lineConstraints[lineIdx - 1].idx];
806 break;
807 }
808 }
809 }
810 }
811 // skip spaces if requested
812 if (skipSpaces == true) {
813 if (line[k] == ' ') {
814 continue;
815 } else {
816 skipSpaces = false;
817 }
818 }
819 // otherwise draw
820 auto character = currentFont->getCharacter(line[k]);
821 if (character != nullptr) {
822 float left = x + xIndentLeft;
823 float top = y + yIndentTop + (lineConstraints[lineIdx].baseLine - currentFont->getBaseLine()) + (lineConstraints[lineIdx].height - lineConstraints[lineIdx].lineHeight);
824 if (boundTexture == -1) {
825 boundTexture = currentFont->getTextureId();
826 guiRenderer->bindTexture(boundTexture);
827 lastColor = currentColor;
828 } else
829 if (boundTexture != currentFont->getTextureId()) {
830 boundTexture = currentFont->getTextureId();
831 guiRenderer->render();
832 guiRenderer->bindTexture(boundTexture);
833 lastColor = currentColor;
834 } else
835 if (currentColor.equals(lastColor) == false) {
836 guiRenderer->render();
837 lastColor = currentColor;
838 }
839 currentFont->drawCharacter(guiRenderer, character, left, top, currentColor);
840 x+= character->getXAdvance();
841 }
842 }
843 }
844 //
845 if (lineConstraints[lineConstraints.size() - 1].idx == line.size()) {
846 y+= lineConstraints[lineConstraints.size() - 1].height;
847 }
848 }
849 }
850
851 //
852 guiRenderer->render();
853
854 //
855 line.clear();
856 lineCharIdxs.clear();
857 lineConstraints.clear();
858 }
859
860 //
861 // Console::println("y: " + to_string(y) + " / " + to_string(autoHeight));
862
863 //
864 guiRenderer->bindTexture(0);
865}
866
867void GUIStyledTextNode::unsetTextStyle(int startIdx, int endIdx) {
868 // TODO: a.drewke
869}
870
871void GUIStyledTextNode::setTextStyle(int startIdx, int endIdx, const GUIColor& color, const string& font, const string& url) {
872 Console::print("GUIStyledTextNode::setTextStyle(): " + to_string(startIdx) + " ... " + to_string(endIdx) + ": '");
873 for (auto i = startIdx; i <= endIdx; i++) Console::print(string() + text.charAt(i));
874 Console::print("'");
875 Console::print(", color = " + to_string(color.getRed()) + ", " + to_string(color.getGreen()) + ", " + to_string(color.getBlue()) + ", " + to_string(color.getAlpha()));
876 Console::print(", url = '" + url + "'");
877 Console::println();
878 unsetTextStyle(startIdx, endIdx);
879 // TODO: a.drewke
880 auto _font = font.empty() == true?nullptr:GUI::getFont(screenNode->getApplicationRootPathName(), font);;
881 if (_font != nullptr) _font->initialize();
882 styles.insert(
883 styles.end(),
884 {
885 .startIdx = startIdx,
886 .endIdx = endIdx,
887 .color = color,
888 .font = _font,
889 .url = url,
890 .image = nullptr,
891 .textureId = -1,
892 .width = -1,
893 .height = -1
894 }
895 );
896}
897
898void GUIStyledTextNode::setTextStyle(int startIdx, int endIdx, const string& font, const string& url) {
899 Console::print("GUIStyledTextNode::setTextStyle(): " + to_string(startIdx) + " ... " + to_string(endIdx) + ": '");
900 for (auto i = startIdx; i <= endIdx; i++) Console::print(string() + text.charAt(i));
901 Console::print("'");
902 Console::print(", url = '" + url + "'");
903 Console::println();
904 unsetTextStyle(startIdx, endIdx);
905 // TODO: a.drewke
906 auto _font = font.empty() == true?nullptr:GUI::getFont(screenNode->getApplicationRootPathName(), font);;
907 if (_font != nullptr) _font->initialize();
908 styles.insert(
909 styles.end(),
910 {
911 .startIdx = startIdx,
912 .endIdx = endIdx,
913 .color = color,
914 .font = _font,
915 .url = url,
916 .image = nullptr,
917 .textureId = -1,
918 .width = -1,
919 .height = -1
920 }
921 );
922}
923
924void GUIStyledTextNode::setImage(int idx, const string& image, const string& url, int width, int height) {
925 Console::println("GUIStyledTextNode::setImage(): " + to_string(idx) + ": " + image + ", url = '" + url + "', width = " + to_string(width) + ", height = " + to_string(height));
926 unsetTextStyle(idx,idx);
927 // TODO: a.drewke
928 auto _image = image.empty() == true?nullptr:GUI::getImage(screenNode->getApplicationRootPathName(), image);
929 styles.insert(
930 styles.end(),
931 {
932 .startIdx = idx,
933 .endIdx = idx,
934 .color = color,
935 .font = nullptr,
936 .url = url,
937 .image = _image,
938 .textureId = Engine::getInstance()->getTextureManager()->addTexture(_image, 0),
939 .width = width == -1?_image->getWidth():width,
940 .height = height == -1?_image->getHeight():height,
941 }
942 );
943}
944
946 for (auto& style: styles) {
947 if (style.font != nullptr) font->dispose();
948 if (style.image != nullptr) Engine::getInstance()->getTextureManager()->removeTexture(style.image->getId());
949 }
950 styles.clear();
951}
Engine main class.
Definition: Engine.h:122
bool equals(const Color4Base &c) const
Compares this color with given color.
Definition: Color4Base.h:245
Color 4 definition.
Definition: Color4.h:20
GUI module class.
Definition: GUI.h:66
static Texture * getImage(const string &applicationRootPath, const string &fileName)
Get image.
Definition: GUI.cpp:145
static GUIFont * getFont(const string &applicationRootPath, const string &fileName)
Get font.
Definition: GUI.cpp:108
GUI element node conditions.
static STATIC_DLL_IMPEXT GUINode_AlignmentHorizontal * RIGHT
static STATIC_DLL_IMPEXT GUINode_AlignmentHorizontal * CENTER
static STATIC_DLL_IMPEXT GUINode_AlignmentHorizontal * LEFT
static STATIC_DLL_IMPEXT GUINode_AlignmentVertical * BOTTOM
static STATIC_DLL_IMPEXT GUINode_AlignmentVertical * CENTER
static STATIC_DLL_IMPEXT GUINode_AlignmentVertical * TOP
static STATIC_DLL_IMPEXT GUINode_RequestedConstraints_RequestedConstraintsType * AUTO
GUI node base class.
Definition: GUINode.h:63
float computeParentChildrenRenderOffsetXTotal()
Definition: GUINode.cpp:951
GUINode_Border border
Definition: GUINode.h:159
virtual void render(GUIRenderer *guiRenderer)
Render.
Definition: GUINode.cpp:508
GUINode_ComputedConstraints computedConstraints
Definition: GUINode.h:151
GUINode_Padding padding
Definition: GUINode.h:158
GUIScreenNode * screenNode
Definition: GUINode.h:146
virtual int getAutoWidth()
Definition: GUINode.cpp:159
GUINode_RequestedConstraints requestedConstraints
Definition: GUINode.h:150
float computeParentChildrenRenderOffsetYTotal()
Definition: GUINode.cpp:962
bool shouldRender()
Determine if to render.
Definition: GUINode.h:295
virtual void dispose()
Dispose node.
Definition: GUINode.cpp:460
GUINode_Alignments alignments
Definition: GUINode.h:149
GUI parent node base class thats supporting child nodes.
Definition: GUIParentNode.h:43
GUI screen node that represents a screen that can be rendered via GUI system.
Definition: GUIScreenNode.h:57
void invalidateLayout(GUINode *node)
Mark a node to be invalidated regarding layout.
const string & getApplicationRootPathName()
void dispose() override
Dispose node.
void setTextStyle(int startIdx, int endIdx, const GUIColor &color, const string &font=string(), const string &url=string())
Set text style.
void unsetTextStyle(int startIdx, int endIdx)
Unset text style.
void setText(const MutableString &text)
Set text.
void determineNextLineConstraints(int &i, int charEndIdx, int textStyleIdx)
Determine next line constraints.
TextStyle * getTextStyle(const vector< int > &lineCharIdxs, int lineCharIdx, int &textStyleIdx)
Get text style for.
void setImage(int idx, const string &image, const string &url=string(), int width=-1, int height=-1)
Set image.
void computeContentAlignment() override
Do content alignment.
void render(GUIRenderer *guiRenderer) override
Render.
The definition of a single character as defined in the AngelCode file format.
Definition: GUICharacter.h:11
GUI Font A font implementation that will parse the output of the AngelCode font tool available at:
Definition: GUIFont.h:48
GUICharacter * getCharacter(uint32_t charId)
Get character defintion.
Definition: GUIFont.h:111
void drawCharacter(GUIRenderer *guiRenderer, GUICharacter *character, int x, int y, const GUIColor &color=GUIColor::GUICOLOR_WHITE)
Draw character.
Definition: GUIFont.cpp:228
void bindTexture(int32_t textureId)
Bind texture.
bool isQuadVisible(float x1, float y1, float x2, float y2, float x3, float y3, float x4, float y4)
Definition: GUIRenderer.h:391
void addQuad(float x1, float y1, float colorR1, float colorG1, float colorB1, float colorA1, float tu1, float tv1, float x2, float y2, float colorR2, float colorG2, float colorB2, float colorA2, float tu2, float tv2, float x3, float y3, float colorR3, float colorG3, float colorB3, float colorA3, float tu3, float tv3, float x4, float y4, float colorR4, float colorG4, float colorB4, float colorA4, float tu4, float tv4)
Add quad Note: quad vertices order 1 2 +-—+ | | | | +-—+ 4 3.
Standard math functions.
Definition: Math.h:21
Console class.
Definition: Console.h:26
Float class.
Definition: Float.h:23
Mutable string class.
Definition: MutableString.h:16
MutableString & append(char c)
Append character.
MutableString & reset()
Reset.
MutableString & remove(int32_t idx, int32_t count)
Remove characters at idx with given length.
char charAt(int32_t idx) const
Get char at index.
const string & getString() const
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19
GUINode_AlignmentVertical * vertical
GUINode_AlignmentHorizontal * horizontal
GUI node border entity.
GUI node padding entity.
GUINode_RequestedConstraints_RequestedConstraintsType * widthType
GUINode_RequestedConstraints_RequestedConstraintsType * heightType
GUI node scale 9 grid entity.