2016-03-04 23:03:47 +00:00
|
|
|
#include "specter/MultiLineTextView.hpp"
|
|
|
|
#include "specter/ViewResources.hpp"
|
2015-11-29 02:55:30 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
namespace specter {
|
2016-03-04 23:03:47 +00:00
|
|
|
static logvisor::Module Log("specter::MultiLineTextView");
|
2015-11-29 02:55:30 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
std::string MultiLineTextView::LineWrap(std::string_view str, int wrap) {
|
|
|
|
size_t rem = str.size();
|
|
|
|
const utf8proc_uint8_t* it = reinterpret_cast<const utf8proc_uint8_t*>(str.data());
|
2019-06-12 02:05:04 +00:00
|
|
|
uint32_t lCh = UINT32_MAX;
|
2018-12-08 05:24:02 +00:00
|
|
|
int adv = 0;
|
|
|
|
|
|
|
|
std::string ret;
|
|
|
|
ret.reserve(str.size());
|
|
|
|
size_t lastSpaceRem;
|
|
|
|
const utf8proc_uint8_t* lastSpaceIt = nullptr;
|
|
|
|
size_t rollbackPos;
|
|
|
|
while (rem) {
|
|
|
|
utf8proc_int32_t ch;
|
|
|
|
utf8proc_ssize_t sz = utf8proc_iterate(it, -1, &ch);
|
|
|
|
if (sz < 0)
|
2020-04-11 22:50:38 +00:00
|
|
|
Log.report(logvisor::Fatal, FMT_STRING("invalid UTF-8 char"));
|
2018-12-08 05:24:02 +00:00
|
|
|
if (ch == '\n') {
|
|
|
|
ret += '\n';
|
|
|
|
lCh = -1;
|
|
|
|
rem -= sz;
|
|
|
|
it += sz;
|
|
|
|
lastSpaceIt = nullptr;
|
|
|
|
adv = 0;
|
|
|
|
continue;
|
2015-12-07 00:52:07 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
const FontAtlas::Glyph* glyph = m_fontAtlas.lookupGlyph(ch);
|
|
|
|
if (!glyph) {
|
|
|
|
rem -= sz;
|
|
|
|
it += sz;
|
|
|
|
continue;
|
2015-12-07 00:52:07 +00:00
|
|
|
}
|
|
|
|
|
2019-06-12 02:05:04 +00:00
|
|
|
if (lCh != UINT32_MAX)
|
2018-12-08 05:24:02 +00:00
|
|
|
adv += TextView::DoKern(m_fontAtlas.lookupKern(lCh, glyph->m_glyphIdx), m_fontAtlas);
|
|
|
|
adv += glyph->m_advance;
|
|
|
|
|
|
|
|
if (adv > wrap && lastSpaceIt) {
|
|
|
|
ret.assign(ret.cbegin(), ret.cbegin() + rollbackPos);
|
|
|
|
ret += '\n';
|
2019-06-12 02:05:04 +00:00
|
|
|
lCh = UINT32_MAX;
|
2018-12-08 05:24:02 +00:00
|
|
|
rem = lastSpaceRem;
|
|
|
|
it = lastSpaceIt;
|
|
|
|
lastSpaceIt = nullptr;
|
|
|
|
adv = 0;
|
|
|
|
continue;
|
|
|
|
}
|
2015-12-07 00:52:07 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
if (sz == 1 && (it[0] == ' ' || it[0] == '-' || it[0] == '/' || it[0] == '\\')) {
|
|
|
|
lastSpaceIt = it + 1;
|
|
|
|
lastSpaceRem = rem - 1;
|
|
|
|
rollbackPos = ret.size() + 1;
|
|
|
|
}
|
|
|
|
for (utf8proc_ssize_t i = 0; i < sz; ++i)
|
|
|
|
ret += it[i];
|
|
|
|
lCh = glyph->m_glyphIdx;
|
|
|
|
rem -= sz;
|
|
|
|
it += sz;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2015-11-30 00:21:42 +00:00
|
|
|
}
|
2015-11-29 02:55:30 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
std::wstring MultiLineTextView::LineWrap(std::wstring_view str, int wrap) {
|
2019-06-12 02:05:04 +00:00
|
|
|
uint32_t lCh = UINT32_MAX;
|
2018-12-08 05:24:02 +00:00
|
|
|
int adv = 0;
|
|
|
|
|
|
|
|
std::wstring ret;
|
|
|
|
ret.reserve(str.size());
|
|
|
|
std::wstring_view::const_iterator lastSpaceIt = str.cend();
|
|
|
|
size_t rollbackPos;
|
|
|
|
for (std::wstring_view::const_iterator it = str.cbegin(); it != str.cend(); ++it) {
|
|
|
|
wchar_t ch = *it;
|
|
|
|
if (ch == L'\n') {
|
|
|
|
ret += L'\n';
|
|
|
|
lCh = -1;
|
|
|
|
lastSpaceIt = str.cend();
|
|
|
|
adv = 0;
|
|
|
|
continue;
|
2015-12-07 00:52:07 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
const FontAtlas::Glyph* glyph = m_fontAtlas.lookupGlyph(ch);
|
|
|
|
if (!glyph)
|
|
|
|
continue;
|
|
|
|
|
2019-06-12 02:05:04 +00:00
|
|
|
if (lCh != UINT32_MAX)
|
2018-12-08 05:24:02 +00:00
|
|
|
adv += TextView::DoKern(m_fontAtlas.lookupKern(lCh, glyph->m_glyphIdx), m_fontAtlas);
|
|
|
|
adv += glyph->m_advance;
|
|
|
|
|
|
|
|
if (adv > wrap && lastSpaceIt != str.cend()) {
|
|
|
|
ret.assign(ret.cbegin(), ret.cbegin() + rollbackPos);
|
|
|
|
ret += L'\n';
|
2019-06-12 02:05:04 +00:00
|
|
|
lCh = UINT32_MAX;
|
2018-12-08 05:24:02 +00:00
|
|
|
it = lastSpaceIt;
|
|
|
|
lastSpaceIt = str.cend();
|
|
|
|
adv = 0;
|
|
|
|
continue;
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
if (ch == L' ' || ch == L'-') {
|
|
|
|
lastSpaceIt = it + 1;
|
|
|
|
rollbackPos = ret.size() + 1;
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
2018-12-08 05:24:02 +00:00
|
|
|
ret += ch;
|
|
|
|
lCh = glyph->m_glyphIdx;
|
|
|
|
}
|
2015-11-29 02:55:30 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
return ret;
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
MultiLineTextView::MultiLineTextView(ViewResources& res, View& parentView, const FontAtlas& font,
|
|
|
|
TextView::Alignment align, size_t lineCapacity, float lineHeight)
|
|
|
|
: View(res, parentView)
|
|
|
|
, m_viewSystem(res)
|
|
|
|
, m_fontAtlas(font)
|
|
|
|
, m_align(align)
|
|
|
|
, m_lineCapacity(lineCapacity)
|
|
|
|
, m_lineHeight(lineHeight) {
|
|
|
|
commitResources(res, [&](boo::IGraphicsDataFactory::Context& ctx) -> bool {
|
|
|
|
buildResources(ctx, res);
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
}
|
2015-12-07 00:52:07 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
MultiLineTextView::MultiLineTextView(ViewResources& res, View& parentView, FontTag font, TextView::Alignment align,
|
|
|
|
size_t lineCapacity, float lineHeight)
|
|
|
|
: MultiLineTextView(res, parentView, res.m_textRes.m_fcache->lookupAtlas(font), align, lineCapacity, lineHeight) {}
|
|
|
|
|
|
|
|
void MultiLineTextView::typesetGlyphs(std::string_view str, const zeus::CColor& defaultColor, unsigned wrap) {
|
|
|
|
if (wrap) {
|
|
|
|
typesetGlyphs(LineWrap(str, wrap), defaultColor);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_width = 0;
|
|
|
|
size_t rem = str.size() + 1;
|
|
|
|
const utf8proc_uint8_t* it = reinterpret_cast<const utf8proc_uint8_t*>(str.data());
|
|
|
|
|
|
|
|
size_t lineCount = 0;
|
|
|
|
while (rem) {
|
|
|
|
utf8proc_int32_t ch;
|
|
|
|
utf8proc_ssize_t sz = utf8proc_iterate(it, -1, &ch);
|
|
|
|
if (sz < 0)
|
2020-04-11 22:50:38 +00:00
|
|
|
Log.report(logvisor::Fatal, FMT_STRING("invalid UTF-8 char"));
|
2018-12-08 05:24:02 +00:00
|
|
|
if (ch == '\n' || ch == '\0')
|
|
|
|
++lineCount;
|
|
|
|
rem -= sz;
|
|
|
|
it += sz;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_lines.reserve(lineCount);
|
|
|
|
rem = str.size() + 1;
|
|
|
|
it = reinterpret_cast<const utf8proc_uint8_t*>(str.data());
|
|
|
|
const utf8proc_uint8_t* beginIt = it;
|
|
|
|
size_t lineIt = 0;
|
|
|
|
|
|
|
|
while (rem) {
|
|
|
|
utf8proc_int32_t ch;
|
|
|
|
utf8proc_ssize_t sz = utf8proc_iterate(it, -1, &ch);
|
|
|
|
if (ch == '\n' || ch == '\0') {
|
2019-09-05 22:16:21 +00:00
|
|
|
TextView& tv = (lineIt < m_lines.size()) ? *m_lines[lineIt]
|
|
|
|
: *m_lines.emplace_back(std::make_unique<TextView>(
|
|
|
|
m_viewSystem, *this, m_fontAtlas, m_align, m_lineCapacity));
|
2018-12-08 05:24:02 +00:00
|
|
|
tv.typesetGlyphs(std::string((char*)beginIt, it - beginIt), defaultColor);
|
|
|
|
m_width = std::max(m_width, tv.nominalWidth());
|
|
|
|
beginIt = it + 1;
|
|
|
|
++lineIt;
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
2018-12-08 05:24:02 +00:00
|
|
|
rem -= sz;
|
|
|
|
it += sz;
|
|
|
|
}
|
|
|
|
|
|
|
|
updateSize();
|
|
|
|
}
|
2015-11-29 02:55:30 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
void MultiLineTextView::typesetGlyphs(std::wstring_view str, const zeus::CColor& defaultColor, unsigned wrap) {
|
|
|
|
if (wrap) {
|
|
|
|
typesetGlyphs(LineWrap(str, wrap), defaultColor);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_width = 0;
|
|
|
|
size_t rem = str.size() + 1;
|
|
|
|
auto it = str.cbegin();
|
|
|
|
|
|
|
|
size_t lineCount = 0;
|
|
|
|
while (rem) {
|
|
|
|
if (*it == L'\n' || *it == L'\0')
|
|
|
|
++lineCount;
|
|
|
|
--rem;
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_lines.reserve(lineCount);
|
|
|
|
rem = str.size() + 1;
|
|
|
|
it = str.cbegin();
|
|
|
|
auto beginIt = it;
|
|
|
|
size_t lineIt = 0;
|
|
|
|
|
|
|
|
while (rem) {
|
|
|
|
if (*it == L'\n' || *it == L'\0') {
|
2019-09-05 22:16:21 +00:00
|
|
|
TextView& tv = (lineIt < m_lines.size()) ? *m_lines[lineIt]
|
|
|
|
: *m_lines.emplace_back(std::make_unique<TextView>(
|
|
|
|
m_viewSystem, *this, m_fontAtlas, m_align, m_lineCapacity));
|
2018-12-08 05:24:02 +00:00
|
|
|
tv.typesetGlyphs(std::wstring(beginIt, it), defaultColor);
|
|
|
|
m_width = std::max(m_width, tv.nominalWidth());
|
|
|
|
beginIt = it + 1;
|
|
|
|
++lineIt;
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
2018-12-08 05:24:02 +00:00
|
|
|
--rem;
|
|
|
|
++it;
|
|
|
|
}
|
2015-11-29 02:55:30 +00:00
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
updateSize();
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
void MultiLineTextView::colorGlyphs(const zeus::CColor& newColor) {
|
|
|
|
for (std::unique_ptr<TextView>& tv : m_lines)
|
|
|
|
tv->colorGlyphs(newColor);
|
2015-12-13 21:00:30 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
void MultiLineTextView::resized(const boo::SWindowRect& root, const boo::SWindowRect& sub) {
|
|
|
|
View::resized(root, sub);
|
|
|
|
unsigned lHeight = unsigned(m_lineHeight * m_fontAtlas.FT_LineHeight()) >> 6;
|
|
|
|
unsigned decumHeight = lHeight * m_lines.size();
|
|
|
|
boo::SWindowRect tsub = sub;
|
|
|
|
tsub.location[1] += decumHeight;
|
|
|
|
tsub.size[1] = 10;
|
|
|
|
for (std::unique_ptr<TextView>& tv : m_lines) {
|
|
|
|
tsub.location[1] -= lHeight;
|
|
|
|
tv->resized(root, tsub);
|
|
|
|
}
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
void MultiLineTextView::draw(boo::IGraphicsCommandQueue* gfxQ) {
|
|
|
|
View::draw(gfxQ);
|
|
|
|
for (std::unique_ptr<TextView>& tv : m_lines)
|
|
|
|
tv->draw(gfxQ);
|
2015-11-29 02:55:30 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:24:02 +00:00
|
|
|
} // namespace specter
|