mirror of
https://github.com/telegramdesktop/tdesktop
synced 2025-01-10 08:51:12 +00:00
Text selection display fixes.
- Use QPainter current pen for selected text in cases we didn't specify it explicitly in the TextPalette. - Better rounding of QRectF from QFixed values to QRect selected background filling areas. - Using QPainter::viewport() instead of random huge QRect() in clipping, because it fails to apply this huge QRect() clip region in Retina paint devices: 2x scale QMatrix::map() is not defined for QRegion-s having rects with sum of dimensions larger than 100000.
This commit is contained in:
parent
9757489645
commit
8d354382a4
@ -55,7 +55,7 @@ defaultTextPalette: TextPalette {
|
|||||||
linkFg: windowActiveTextFg;
|
linkFg: windowActiveTextFg;
|
||||||
monoFg: windowSubTextFg;
|
monoFg: windowSubTextFg;
|
||||||
selectBg: msgInBgSelected;
|
selectBg: msgInBgSelected;
|
||||||
selectFg: historyTextInFgSelected;
|
selectFg: transparent; // use painter current pen instead
|
||||||
selectLinkFg: historyLinkInFgSelected;
|
selectLinkFg: historyLinkInFgSelected;
|
||||||
selectMonoFg: msgInMonoFgSelected;
|
selectMonoFg: msgInMonoFgSelected;
|
||||||
selectOverlay: msgSelectOverlay;
|
selectOverlay: msgSelectOverlay;
|
||||||
|
@ -1184,7 +1184,7 @@ void executeParsedCommand(const QString &command) {
|
|||||||
|
|
||||||
void MainWidget::sendMessage(const MessageToSend &message) {
|
void MainWidget::sendMessage(const MessageToSend &message) {
|
||||||
auto history = message.history;
|
auto history = message.history;
|
||||||
const auto &textWithTags = message.textWithTags;
|
auto &textWithTags = message.textWithTags;
|
||||||
|
|
||||||
readServerHistory(history);
|
readServerHistory(history);
|
||||||
_history->fastShowAtEnd(history);
|
_history->fastShowAtEnd(history);
|
||||||
|
@ -503,6 +503,7 @@ public:
|
|||||||
}
|
}
|
||||||
parse(options);
|
parse(options);
|
||||||
}
|
}
|
||||||
|
|
||||||
TextParser(Text *t, const TextWithEntities &textWithEntities, const TextParseOptions &options) : _t(t),
|
TextParser(Text *t, const TextWithEntities &textWithEntities, const TextParseOptions &options) : _t(t),
|
||||||
src(textWithEntities.text),
|
src(textWithEntities.text),
|
||||||
rich(options.flags & TextParseRichText),
|
rich(options.flags & TextParseRichText),
|
||||||
@ -534,6 +535,7 @@ public:
|
|||||||
}
|
}
|
||||||
parse(options);
|
parse(options);
|
||||||
}
|
}
|
||||||
|
|
||||||
void parse(const TextParseOptions &options) {
|
void parse(const TextParseOptions &options) {
|
||||||
if (options.maxw > 0 && options.maxh > 0) {
|
if (options.maxw > 0 && options.maxh > 0) {
|
||||||
stopAfterWidth = ((options.maxh / _t->_st->font->height) + 1) * options.maxw;
|
stopAfterWidth = ((options.maxh / _t->_st->font->height) + 1) * options.maxw;
|
||||||
@ -640,11 +642,11 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
enum LinkDisplayStatus {
|
enum LinkDisplayStatus {
|
||||||
LinkDisplayedFull,
|
LinkDisplayedFull,
|
||||||
LinkDisplayedElided,
|
LinkDisplayedElided,
|
||||||
};
|
};
|
||||||
|
|
||||||
struct TextLinkData {
|
struct TextLinkData {
|
||||||
TextLinkData() = default;
|
TextLinkData() = default;
|
||||||
TextLinkData(EntityInTextType type, const QString &text, const QString &data, LinkDisplayStatus displayStatus)
|
TextLinkData(EntityInTextType type, const QString &text, const QString &data, LinkDisplayStatus displayStatus)
|
||||||
@ -790,83 +792,18 @@ namespace {
|
|||||||
++stop;
|
++stop;
|
||||||
start = stop;
|
start = stop;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
class TextPainter {
|
class TextPainter {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
static inline uint16 _blockEnd(const Text *t, const Text::TextBlocks::const_iterator &i, const Text::TextBlocks::const_iterator &e) {
|
|
||||||
return (i + 1 == e) ? t->_text.size() : (*(i + 1))->from();
|
|
||||||
}
|
|
||||||
static inline uint16 _blockLength(const Text *t, const Text::TextBlocks::const_iterator &i, const Text::TextBlocks::const_iterator &e) {
|
|
||||||
return _blockEnd(t, i, e) - (*i)->from();
|
|
||||||
}
|
|
||||||
|
|
||||||
TextPainter(Painter *p, const Text *t) : _p(p), _t(t) {
|
TextPainter(Painter *p, const Text *t) : _p(p), _t(t) {
|
||||||
}
|
}
|
||||||
|
|
||||||
~TextPainter() {
|
~TextPainter() {
|
||||||
restoreAfterElided();
|
restoreAfterElided();
|
||||||
}
|
if (_p) {
|
||||||
|
_p->setPen(_originalPen);
|
||||||
void initNextParagraph(Text::TextBlocks::const_iterator i) {
|
|
||||||
_parStartBlock = i;
|
|
||||||
Text::TextBlocks::const_iterator e = _t->_blocks.cend();
|
|
||||||
if (i == e) {
|
|
||||||
_parStart = _t->_text.size();
|
|
||||||
_parLength = 0;
|
|
||||||
} else {
|
|
||||||
_parStart = (*i)->from();
|
|
||||||
for (; i != e; ++i) {
|
|
||||||
if ((*i)->type() == TextBlockTNewline) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_parLength = ((i == e) ? _t->_text.size() : (*i)->from()) - _parStart;
|
|
||||||
}
|
|
||||||
_parAnalysis.resize(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
void initParagraphBidi() {
|
|
||||||
if (!_parLength || !_parAnalysis.isEmpty()) return;
|
|
||||||
|
|
||||||
Text::TextBlocks::const_iterator i = _parStartBlock, e = _t->_blocks.cend(), n = i + 1;
|
|
||||||
|
|
||||||
bool ignore = false;
|
|
||||||
bool rtl = (_parDirection == Qt::RightToLeft);
|
|
||||||
if (!ignore && !rtl) {
|
|
||||||
ignore = true;
|
|
||||||
const ushort *start = reinterpret_cast<const ushort*>(_str) + _parStart;
|
|
||||||
const ushort *curr = start;
|
|
||||||
const ushort *end = start + _parLength;
|
|
||||||
while (curr < end) {
|
|
||||||
while (n != e && (*n)->from() <= _parStart + (curr - start)) {
|
|
||||||
i = n;
|
|
||||||
++n;
|
|
||||||
}
|
|
||||||
if ((*i)->type() != TextBlockTEmoji && *curr >= 0x590) {
|
|
||||||
ignore = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
++curr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_parAnalysis.resize(_parLength);
|
|
||||||
QScriptAnalysis *analysis = _parAnalysis.data();
|
|
||||||
|
|
||||||
BidiControl control(rtl);
|
|
||||||
|
|
||||||
_parHasBidi = false;
|
|
||||||
if (ignore) {
|
|
||||||
memset(analysis, 0, _parLength * sizeof(QScriptAnalysis));
|
|
||||||
if (rtl) {
|
|
||||||
for (int i = 0; i < _parLength; ++i)
|
|
||||||
analysis[i].bidiLevel = 1;
|
|
||||||
_parHasBidi = true;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
_parHasBidi = eBidiItemize(analysis, control);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -876,7 +813,9 @@ public:
|
|||||||
_blocksSize = _t->_blocks.size();
|
_blocksSize = _t->_blocks.size();
|
||||||
if (_p) {
|
if (_p) {
|
||||||
_p->setFont(_t->_st->font);
|
_p->setFont(_t->_st->font);
|
||||||
|
_textPalette = &_p->textPalette();
|
||||||
_originalPen = _p->pen();
|
_originalPen = _p->pen();
|
||||||
|
_originalPenSelected = (_textPalette->selectFg->c.alphaF() == 0) ? _originalPen : _textPalette->selectFg->p;
|
||||||
}
|
}
|
||||||
|
|
||||||
_x = left;
|
_x = left;
|
||||||
@ -898,7 +837,7 @@ public:
|
|||||||
_str = _t->_text.unicode();
|
_str = _t->_text.unicode();
|
||||||
|
|
||||||
if (_p) {
|
if (_p) {
|
||||||
auto clip = _p->hasClipping() ? _p->clipBoundingRect() : QRectF();
|
auto clip = _p->hasClipping() ? _p->clipBoundingRect() : QRect();
|
||||||
if (clip.width() > 0 || clip.height() > 0) {
|
if (clip.width() > 0 || clip.height() > 0) {
|
||||||
if (_yFrom < clip.y()) _yFrom = clip.y();
|
if (_yFrom < clip.y()) _yFrom = clip.y();
|
||||||
if (_yTo < 0 || _yTo > clip.y() + clip.height()) _yTo = clip.y() + clip.height();
|
if (_yTo < 0 || _yTo > clip.y() + clip.height()) _yTo = clip.y() + clip.height();
|
||||||
@ -930,8 +869,10 @@ public:
|
|||||||
|
|
||||||
if (_btype == TextBlockTNewline) {
|
if (_btype == TextBlockTNewline) {
|
||||||
if (!_lineHeight) _lineHeight = blockHeight;
|
if (!_lineHeight) _lineHeight = blockHeight;
|
||||||
ushort nextStart = _blockEnd(_t, i, e);
|
ushort nextStart = _t->countBlockEnd(i, e);
|
||||||
if (!drawLine(nextStart, i + 1, e)) return;
|
if (!drawLine(nextStart, i + 1, e)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
_y += _lineHeight;
|
_y += _lineHeight;
|
||||||
_lineHeight = 0;
|
_lineHeight = 0;
|
||||||
@ -1015,7 +956,9 @@ public:
|
|||||||
_lineHeight = f_lineHeight;
|
_lineHeight = f_lineHeight;
|
||||||
j_width = (j->f_width() >= 0) ? j->f_width() : -j->f_width();
|
j_width = (j->f_width() >= 0) ? j->f_width() : -j->f_width();
|
||||||
}
|
}
|
||||||
if (!drawLine(elidedLine ? ((j + 1 == en) ? _blockEnd(_t, i, e) : (j + 1)->from()) : j->from(), i, e)) return;
|
if (!drawLine(elidedLine ? ((j + 1 == en) ? _t->countBlockEnd(i, e) : (j + 1)->from()) : j->from(), i, e)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
_y += _lineHeight;
|
_y += _lineHeight;
|
||||||
_lineHeight = qMax(0, blockHeight);
|
_lineHeight = qMax(0, blockHeight);
|
||||||
_lineStart = j->from();
|
_lineStart = j->from();
|
||||||
@ -1041,7 +984,9 @@ public:
|
|||||||
if (elidedLine) {
|
if (elidedLine) {
|
||||||
_lineHeight = elidedLineHeight;
|
_lineHeight = elidedLineHeight;
|
||||||
}
|
}
|
||||||
if (!drawLine(elidedLine ? _blockEnd(_t, i, e) : b->from(), i, e)) return;
|
if (!drawLine(elidedLine ? _t->countBlockEnd(i, e) : b->from(), i, e)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
_y += _lineHeight;
|
_y += _lineHeight;
|
||||||
_lineHeight = qMax(0, blockHeight);
|
_lineHeight = qMax(0, blockHeight);
|
||||||
_lineStart = b->from();
|
_lineStart = b->from();
|
||||||
@ -1116,6 +1061,68 @@ public:
|
|||||||
return _lookupResult;
|
return _lookupResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void initNextParagraph(Text::TextBlocks::const_iterator i) {
|
||||||
|
_parStartBlock = i;
|
||||||
|
Text::TextBlocks::const_iterator e = _t->_blocks.cend();
|
||||||
|
if (i == e) {
|
||||||
|
_parStart = _t->_text.size();
|
||||||
|
_parLength = 0;
|
||||||
|
} else {
|
||||||
|
_parStart = (*i)->from();
|
||||||
|
for (; i != e; ++i) {
|
||||||
|
if ((*i)->type() == TextBlockTNewline) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_parLength = ((i == e) ? _t->_text.size() : (*i)->from()) - _parStart;
|
||||||
|
}
|
||||||
|
_parAnalysis.resize(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void initParagraphBidi() {
|
||||||
|
if (!_parLength || !_parAnalysis.isEmpty()) return;
|
||||||
|
|
||||||
|
Text::TextBlocks::const_iterator i = _parStartBlock, e = _t->_blocks.cend(), n = i + 1;
|
||||||
|
|
||||||
|
bool ignore = false;
|
||||||
|
bool rtl = (_parDirection == Qt::RightToLeft);
|
||||||
|
if (!ignore && !rtl) {
|
||||||
|
ignore = true;
|
||||||
|
const ushort *start = reinterpret_cast<const ushort*>(_str) + _parStart;
|
||||||
|
const ushort *curr = start;
|
||||||
|
const ushort *end = start + _parLength;
|
||||||
|
while (curr < end) {
|
||||||
|
while (n != e && (*n)->from() <= _parStart + (curr - start)) {
|
||||||
|
i = n;
|
||||||
|
++n;
|
||||||
|
}
|
||||||
|
if ((*i)->type() != TextBlockTEmoji && *curr >= 0x590) {
|
||||||
|
ignore = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
++curr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_parAnalysis.resize(_parLength);
|
||||||
|
QScriptAnalysis *analysis = _parAnalysis.data();
|
||||||
|
|
||||||
|
BidiControl control(rtl);
|
||||||
|
|
||||||
|
_parHasBidi = false;
|
||||||
|
if (ignore) {
|
||||||
|
memset(analysis, 0, _parLength * sizeof(QScriptAnalysis));
|
||||||
|
if (rtl) {
|
||||||
|
for (int i = 0; i < _parLength; ++i)
|
||||||
|
analysis[i].bidiLevel = 1;
|
||||||
|
_parHasBidi = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
_parHasBidi = eBidiItemize(analysis, control);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool drawLine(uint16 _lineEnd, const Text::TextBlocks::const_iterator &_endBlockIter, const Text::TextBlocks::const_iterator &_end) {
|
bool drawLine(uint16 _lineEnd, const Text::TextBlocks::const_iterator &_endBlockIter, const Text::TextBlocks::const_iterator &_end) {
|
||||||
_yDelta = (_lineHeight - _fontHeight) / 2;
|
_yDelta = (_lineHeight - _fontHeight) / 2;
|
||||||
if (_yTo >= 0 && (_y + _yDelta >= _yTo || _y >= _yTo)) return false;
|
if (_yTo >= 0 && (_y + _yDelta >= _yTo || _y >= _yTo)) return false;
|
||||||
@ -1128,12 +1135,14 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint16 trimmedLineEnd = _lineEnd;
|
uint16 trimmedLineEnd = _lineEnd;
|
||||||
for (; trimmedLineEnd > _lineStart; --trimmedLineEnd) {
|
// Disable text trimming because it causes render bugs in case of
|
||||||
QChar ch = _t->_text.at(trimmedLineEnd - 1);
|
// fully selected lines with pending spaces.
|
||||||
if ((ch != QChar::Space || trimmedLineEnd == _lineStart + 1) && ch != QChar::LineFeed) {
|
//for (; trimmedLineEnd > _lineStart; --trimmedLineEnd) {
|
||||||
break;
|
// QChar ch = _t->_text.at(trimmedLineEnd - 1);
|
||||||
}
|
// if ((ch != QChar::Space || trimmedLineEnd == _lineStart + 1) && ch != QChar::LineFeed) {
|
||||||
}
|
// break;
|
||||||
|
// }
|
||||||
|
//}
|
||||||
|
|
||||||
ITextBlock *_endBlock = (_endBlockIter == _end) ? nullptr : (*_endBlockIter);
|
ITextBlock *_endBlock = (_endBlockIter == _end) ? nullptr : (*_endBlockIter);
|
||||||
bool elidedLine = _elideLast && (_y + _lineHeight >= _yToElide);
|
bool elidedLine = _elideLast && (_y + _lineHeight >= _yToElide);
|
||||||
@ -1155,7 +1164,7 @@ public:
|
|||||||
|
|
||||||
int32 delta = (currentBlock->from() < _lineStart ? qMin(_lineStart - currentBlock->from(), 2) : 0);
|
int32 delta = (currentBlock->from() < _lineStart ? qMin(_lineStart - currentBlock->from(), 2) : 0);
|
||||||
_localFrom = _lineStart - delta;
|
_localFrom = _lineStart - delta;
|
||||||
int32 lineEnd = (_endBlock && _endBlock->from() < trimmedLineEnd && !elidedLine) ? qMin(uint16(trimmedLineEnd + 2), _blockEnd(_t, _endBlockIter, _end)) : trimmedLineEnd;
|
int32 lineEnd = (_endBlock && _endBlock->from() < trimmedLineEnd && !elidedLine) ? qMin(uint16(trimmedLineEnd + 2), _t->countBlockEnd(_endBlockIter, _end)) : trimmedLineEnd;
|
||||||
|
|
||||||
QString lineText = _t->_text.mid(_localFrom, lineEnd - _localFrom);
|
QString lineText = _t->_text.mid(_localFrom, lineEnd - _localFrom);
|
||||||
int32 lineStart = delta, lineLength = trimmedLineEnd - _lineStart;
|
int32 lineStart = delta, lineLength = trimmedLineEnd - _lineStart;
|
||||||
@ -1214,12 +1223,12 @@ public:
|
|||||||
|
|
||||||
if ((selectFromStart && _parDirection == Qt::LeftToRight) || (selectTillEnd && _parDirection == Qt::RightToLeft)) {
|
if ((selectFromStart && _parDirection == Qt::LeftToRight) || (selectTillEnd && _parDirection == Qt::RightToLeft)) {
|
||||||
if (x > _x) {
|
if (x > _x) {
|
||||||
_p->fillRect(QRectF(_x.toReal(), _y + _yDelta, (x - _x).toReal(), _fontHeight), _p->textPalette().selectBg);
|
fillSelectRange(_x, x);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ((selectTillEnd && _parDirection == Qt::LeftToRight) || (selectFromStart && _parDirection == Qt::RightToLeft)) {
|
if ((selectTillEnd && _parDirection == Qt::LeftToRight) || (selectFromStart && _parDirection == Qt::RightToLeft)) {
|
||||||
if (x < _x + _wLeft) {
|
if (x < _x + _wLeft) {
|
||||||
_p->fillRect(QRectF((x + _w - _wLeft).toReal(), _y + _yDelta, (_x + _wLeft - x).toReal(), _fontHeight), _p->textPalette().selectBg);
|
fillSelectRange(x + _w - _wLeft, _x + _w);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1320,21 +1329,29 @@ public:
|
|||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
const QChar *chFrom = _str + currentBlock->from(), *chTo = chFrom + ((nextBlock ? nextBlock->from() : _t->_text.size()) - currentBlock->from());
|
|
||||||
if (chTo > chFrom && (chTo - 1)->unicode() == QChar::Space) {
|
// Emoji with spaces after symbol lookup
|
||||||
if (rtl) {
|
auto chFrom = _str + currentBlock->from();
|
||||||
if (_lookupX < x + (si.width - currentBlock->f_width())) {
|
auto chTo = chFrom + ((nextBlock ? nextBlock->from() : _t->_text.size()) - currentBlock->from());
|
||||||
_lookupResult.symbol = (chTo - 1 - _str); // up to ending space, included, rtl
|
auto spacesWidth = (si.width - currentBlock->f_width());
|
||||||
_lookupResult.afterSymbol = (_lookupX < x + (si.width - currentBlock->f_width()) / 2) ? true : false;
|
auto spacesCount = 0;
|
||||||
return false;
|
while (chTo > chFrom && (chTo - 1)->unicode() == QChar::Space) {
|
||||||
}
|
++spacesCount;
|
||||||
} else if (_lookupX >= x + currentBlock->f_width()) {
|
|
||||||
_lookupResult.symbol = (chTo - 1 - _str); // up to ending space, inclided, ltr
|
|
||||||
_lookupResult.afterSymbol = (_lookupX >= x + currentBlock->f_width() + (currentBlock->f_rpadding() / 2)) ? true : false;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
--chTo;
|
--chTo;
|
||||||
}
|
}
|
||||||
|
if (spacesCount > 0) { // Check if we're over a space.
|
||||||
|
if (rtl) {
|
||||||
|
if (_lookupX < x + spacesWidth) {
|
||||||
|
_lookupResult.symbol = (chTo - _str); // up to a space, included, rtl
|
||||||
|
_lookupResult.afterSymbol = (_lookupX < x + (spacesWidth / 2)) ? true : false;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else if (_lookupX >= x + si.width - spacesWidth) {
|
||||||
|
_lookupResult.symbol = (chTo - _str); // up to a space, inclided, ltr
|
||||||
|
_lookupResult.afterSymbol = (_lookupX >= x + si.width - spacesWidth + (spacesWidth / 2)) ? true : false;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
if (_lookupX < x + (rtl ? (si.width - currentBlock->f_width()) : 0) + (currentBlock->f_width() / 2)) {
|
if (_lookupX < x + (rtl ? (si.width - currentBlock->f_width()) : 0) + (currentBlock->f_width() / 2)) {
|
||||||
_lookupResult.symbol = ((rtl && chTo > chFrom) ? (chTo - 1) : chFrom) - _str;
|
_lookupResult.symbol = ((rtl && chTo > chFrom) ? (chTo - 1) : chFrom) - _str;
|
||||||
_lookupResult.afterSymbol = (rtl && chTo > chFrom) ? true : false;
|
_lookupResult.afterSymbol = (rtl && chTo > chFrom) ? true : false;
|
||||||
@ -1345,27 +1362,29 @@ public:
|
|||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
} else if (_p && _type == TextBlockTEmoji) {
|
} else if (_p && _type == TextBlockTEmoji) {
|
||||||
QFixed glyphX = x;
|
auto glyphX = x;
|
||||||
|
auto spacesWidth = (si.width - currentBlock->f_width());
|
||||||
if (rtl) {
|
if (rtl) {
|
||||||
glyphX += (si.width - currentBlock->f_width());
|
glyphX += spacesWidth;
|
||||||
}
|
}
|
||||||
if (_localFrom + si.position < _selection.to) {
|
if (_localFrom + si.position < _selection.to) {
|
||||||
const QChar *chFrom = _str + currentBlock->from(), *chTo = chFrom + ((nextBlock ? nextBlock->from() : _t->_text.size()) - currentBlock->from());
|
auto chFrom = _str + currentBlock->from();
|
||||||
|
auto chTo = chFrom + ((nextBlock ? nextBlock->from() : _t->_text.size()) - currentBlock->from());
|
||||||
if (_localFrom + si.position >= _selection.from) { // could be without space
|
if (_localFrom + si.position >= _selection.from) { // could be without space
|
||||||
if (chTo == chFrom || (chTo - 1)->unicode() != QChar::Space || _selection.to >= (chTo - _str)) {
|
if (chTo == chFrom || (chTo - 1)->unicode() != QChar::Space || _selection.to >= (chTo - _str)) {
|
||||||
_p->fillRect(QRectF(x.toReal(), _y + _yDelta, si.width.toReal(), _fontHeight), _p->textPalette().selectBg);
|
fillSelectRange(x, x + si.width);
|
||||||
} else { // or with space
|
} else { // or with space
|
||||||
_p->fillRect(QRectF(glyphX.toReal(), _y + _yDelta, currentBlock->f_width().toReal(), _fontHeight), _p->textPalette().selectBg);
|
fillSelectRange(glyphX, glyphX + currentBlock->f_width());
|
||||||
}
|
}
|
||||||
} else if (chTo > chFrom && (chTo - 1)->unicode() == QChar::Space && (chTo - 1 - _str) >= _selection.from) {
|
} else if (chTo > chFrom && (chTo - 1)->unicode() == QChar::Space && (chTo - 1 - _str) >= _selection.from) {
|
||||||
if (rtl) { // rtl space only
|
if (rtl) { // rtl space only
|
||||||
_p->fillRect(QRectF(x.toReal(), _y + _yDelta, (glyphX - x).toReal(), _fontHeight), _p->textPalette().selectBg);
|
fillSelectRange(x, glyphX);
|
||||||
} else { // ltr space only
|
} else { // ltr space only
|
||||||
_p->fillRect(QRectF((x + currentBlock->f_width()).toReal(), _y + _yDelta, (si.width - currentBlock->f_width()).toReal(), _fontHeight), _p->textPalette().selectBg);
|
fillSelectRange(x + currentBlock->f_width(), x + si.width);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
emojiDraw(*_p, static_cast<EmojiBlock*>(currentBlock)->emoji, (glyphX + int(st::emojiPadding)).toInt(), _y + _yDelta + emojiY);
|
emojiDraw(*_p, static_cast<EmojiBlock*>(currentBlock)->emoji, (glyphX + st::emojiPadding).toInt(), _y + _yDelta + emojiY);
|
||||||
// } else if (_p && currentBlock->type() == TextBlockSkip) { // debug
|
// } else if (_p && currentBlock->type() == TextBlockSkip) { // debug
|
||||||
// _p->fillRect(QRect(x.toInt(), _y, currentBlock->width(), static_cast<SkipBlock*>(currentBlock)->height()), QColor(0, 0, 0, 32));
|
// _p->fillRect(QRect(x.toInt(), _y, currentBlock->width(), static_cast<SkipBlock*>(currentBlock)->height()), QColor(0, 0, 0, 32));
|
||||||
}
|
}
|
||||||
@ -1492,8 +1511,8 @@ public:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (rtl) selX = x + itemWidth - (selX - x) - selWidth;
|
if (rtl) selX = x + itemWidth - (selX - x) - selWidth;
|
||||||
selectedRect = QRect(qRound(selX.toReal()), _y + _yDelta, qRound(selWidth.toReal()), _fontHeight);
|
selectedRect = QRect(selX.toInt(), _y + _yDelta, (selX + selWidth).toInt() - selX.toInt(), _fontHeight);
|
||||||
_p->fillRect(selectedRect, _p->textPalette().selectBg);
|
fillSelectRange(selX, selX + selWidth);
|
||||||
}
|
}
|
||||||
if (Q_UNLIKELY(hasSelected)) {
|
if (Q_UNLIKELY(hasSelected)) {
|
||||||
if (Q_UNLIKELY(hasNotSelected)) {
|
if (Q_UNLIKELY(hasNotSelected)) {
|
||||||
@ -1502,7 +1521,7 @@ public:
|
|||||||
_p->setClipRect(selectedRect, Qt::IntersectClip);
|
_p->setClipRect(selectedRect, Qt::IntersectClip);
|
||||||
_p->setPen(*_currentPenSelected);
|
_p->setPen(*_currentPenSelected);
|
||||||
_p->drawTextItem(QPointF(x.toReal(), textY), gf);
|
_p->drawTextItem(QPointF(x.toReal(), textY), gf);
|
||||||
_p->setClipRegion((clippingEnabled ? clippingRegion : QRegion(QRect(0, 0, QFIXED_MAX - 1, QFIXED_MAX - 1))) - selectedRect);
|
_p->setClipRegion((clippingEnabled ? clippingRegion : QRegion(_p->viewport())) - selectedRect);
|
||||||
_p->setPen(*_currentPen);
|
_p->setPen(*_currentPen);
|
||||||
_p->drawTextItem(QPointF(x.toReal(), textY), gf);
|
_p->drawTextItem(QPointF(x.toReal(), textY), gf);
|
||||||
if (clippingEnabled) {
|
if (clippingEnabled) {
|
||||||
@ -1524,6 +1543,11 @@ public:
|
|||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
void fillSelectRange(QFixed from, QFixed to) {
|
||||||
|
auto left = from.toInt();
|
||||||
|
auto width = to.toInt() - left;
|
||||||
|
_p->fillRect(left, _y + _yDelta, width, _fontHeight, _textPalette->selectBg);
|
||||||
|
}
|
||||||
|
|
||||||
void elideSaveBlock(int32 blockIndex, ITextBlock *&_endBlock, int32 elideStart, int32 elideWidth) {
|
void elideSaveBlock(int32 blockIndex, ITextBlock *&_endBlock, int32 elideStart, int32 elideWidth) {
|
||||||
if (_elideSavedBlock) {
|
if (_elideSavedBlock) {
|
||||||
@ -2291,30 +2315,33 @@ private:
|
|||||||
void applyBlockProperties(ITextBlock *block) {
|
void applyBlockProperties(ITextBlock *block) {
|
||||||
eSetFont(block);
|
eSetFont(block);
|
||||||
if (_p) {
|
if (_p) {
|
||||||
auto &palette = _p->textPalette();
|
|
||||||
if (block->lnkIndex()) {
|
if (block->lnkIndex()) {
|
||||||
_currentPen = &palette.linkFg->p;
|
_currentPen = &_textPalette->linkFg->p;
|
||||||
_currentPenSelected = &palette.selectLinkFg->p;
|
_currentPenSelected = &_textPalette->selectLinkFg->p;
|
||||||
} else if ((block->flags() & TextBlockFCode) || (block->flags() & TextBlockFPre)) {
|
} else if ((block->flags() & TextBlockFCode) || (block->flags() & TextBlockFPre)) {
|
||||||
_currentPen = &palette.monoFg->p;
|
_currentPen = &_textPalette->monoFg->p;
|
||||||
_currentPenSelected = &palette.selectMonoFg->p;
|
_currentPenSelected = &_textPalette->selectMonoFg->p;
|
||||||
} else {
|
} else {
|
||||||
_currentPen = &_originalPen;
|
_currentPen = &_originalPen;
|
||||||
_currentPenSelected = &palette.selectFg->p;
|
_currentPenSelected = &_originalPenSelected;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Painter *_p;
|
Painter *_p = nullptr;
|
||||||
const Text *_t;
|
const style::TextPalette *_textPalette = nullptr;
|
||||||
|
const Text *_t = nullptr;
|
||||||
bool _elideLast = false;
|
bool _elideLast = false;
|
||||||
bool _breakEverywhere = false;
|
bool _breakEverywhere = false;
|
||||||
int32 _elideRemoveFromEnd = 0;
|
int _elideRemoveFromEnd = 0;
|
||||||
style::align _align;
|
style::align _align = style::al_topleft;
|
||||||
QPen _originalPen;
|
QPen _originalPen;
|
||||||
|
QPen _originalPenSelected;
|
||||||
const QPen *_currentPen = nullptr;
|
const QPen *_currentPen = nullptr;
|
||||||
const QPen *_currentPenSelected = nullptr;
|
const QPen *_currentPenSelected = nullptr;
|
||||||
int32 _yFrom, _yTo, _yToElide;
|
int _yFrom = 0;
|
||||||
|
int _yTo = 0;
|
||||||
|
int _yToElide = 0;
|
||||||
TextSelection _selection = { 0, 0 };
|
TextSelection _selection = { 0, 0 };
|
||||||
bool _fullWidthSelection = true;
|
bool _fullWidthSelection = true;
|
||||||
const QChar *_str = nullptr;
|
const QChar *_str = nullptr;
|
||||||
@ -2322,23 +2349,25 @@ private:
|
|||||||
// current paragraph data
|
// current paragraph data
|
||||||
Text::TextBlocks::const_iterator _parStartBlock;
|
Text::TextBlocks::const_iterator _parStartBlock;
|
||||||
Qt::LayoutDirection _parDirection;
|
Qt::LayoutDirection _parDirection;
|
||||||
int32 _parStart, _parLength;
|
int _parStart = 0;
|
||||||
bool _parHasBidi;
|
int _parLength = 0;
|
||||||
|
bool _parHasBidi = false;
|
||||||
QVarLengthArray<QScriptAnalysis, 4096> _parAnalysis;
|
QVarLengthArray<QScriptAnalysis, 4096> _parAnalysis;
|
||||||
|
|
||||||
// current line data
|
// current line data
|
||||||
QTextEngine *_e;
|
QTextEngine *_e = nullptr;
|
||||||
style::font _f;
|
style::font _f;
|
||||||
QFixed _x, _w, _wLeft;
|
QFixed _x, _w, _wLeft;
|
||||||
int32 _y, _yDelta, _lineHeight, _fontHeight;
|
int32 _y, _yDelta, _lineHeight, _fontHeight;
|
||||||
|
|
||||||
// elided hack support
|
// elided hack support
|
||||||
int32 _blocksSize;
|
int _blocksSize = 0;
|
||||||
int32 _elideSavedIndex;
|
int _elideSavedIndex = 0;
|
||||||
ITextBlock *_elideSavedBlock = nullptr;
|
ITextBlock *_elideSavedBlock = nullptr;
|
||||||
|
|
||||||
int32 _lineStart, _localFrom;
|
int _lineStart = 0;
|
||||||
int32 _lineStartBlock;
|
int _localFrom = 0;
|
||||||
|
int _lineStartBlock = 0;
|
||||||
|
|
||||||
// link and symbol resolve
|
// link and symbol resolve
|
||||||
QFixed _lookupX = 0;
|
QFixed _lookupX = 0;
|
||||||
@ -2860,6 +2889,14 @@ bool Text::isEmpty() const {
|
|||||||
return _blocks.empty() || _blocks[0]->type() == TextBlockTSkip;
|
return _blocks.empty() || _blocks[0]->type() == TextBlockTSkip;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint16 Text::countBlockEnd(const TextBlocks::const_iterator &i, const TextBlocks::const_iterator &e) const {
|
||||||
|
return (i + 1 == e) ? _text.size() : (*(i + 1))->from();
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16 Text::countBlockLength(const Text::TextBlocks::const_iterator &i, const Text::TextBlocks::const_iterator &e) const {
|
||||||
|
return countBlockEnd(i, e) - (*i)->from();
|
||||||
|
}
|
||||||
|
|
||||||
template <typename AppendPartCallback, typename ClickHandlerStartCallback, typename ClickHandlerFinishCallback, typename FlagsChangeCallback>
|
template <typename AppendPartCallback, typename ClickHandlerStartCallback, typename ClickHandlerFinishCallback, typename FlagsChangeCallback>
|
||||||
void Text::enumerateText(TextSelection selection, AppendPartCallback appendPartCallback, ClickHandlerStartCallback clickHandlerStartCallback, ClickHandlerFinishCallback clickHandlerFinishCallback, FlagsChangeCallback flagsChangeCallback) const {
|
void Text::enumerateText(TextSelection selection, AppendPartCallback appendPartCallback, ClickHandlerStartCallback clickHandlerStartCallback, ClickHandlerFinishCallback clickHandlerFinishCallback, FlagsChangeCallback flagsChangeCallback) const {
|
||||||
if (isEmpty() || selection.empty()) {
|
if (isEmpty() || selection.empty()) {
|
||||||
@ -2907,7 +2944,7 @@ void Text::enumerateText(TextSelection selection, AppendPartCallback appendPartC
|
|||||||
|
|
||||||
if (!blockLnkIndex) {
|
if (!blockLnkIndex) {
|
||||||
auto rangeFrom = qMax(selection.from, blockFrom);
|
auto rangeFrom = qMax(selection.from, blockFrom);
|
||||||
auto rangeTo = qMin(selection.to, uint16(blockFrom + TextPainter::_blockLength(this, i, e)));
|
auto rangeTo = qMin(selection.to, uint16(blockFrom + countBlockLength(i, e)));
|
||||||
if (rangeTo > rangeFrom) {
|
if (rangeTo > rangeFrom) {
|
||||||
appendPartCallback(_text.midRef(rangeFrom, rangeTo - rangeFrom));
|
appendPartCallback(_text.midRef(rangeFrom, rangeTo - rangeFrom));
|
||||||
}
|
}
|
||||||
|
@ -83,7 +83,6 @@ typedef QMap<QChar, TextCustomTag> TextCustomTagsMap;
|
|||||||
class ITextBlock;
|
class ITextBlock;
|
||||||
class Text {
|
class Text {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
Text(int32 minResizeWidth = QFIXED_MAX);
|
Text(int32 minResizeWidth = QFIXED_MAX);
|
||||||
Text(const style::TextStyle &st, const QString &text, const TextParseOptions &options = _defaultOptions, int32 minResizeWidth = QFIXED_MAX, bool richText = false);
|
Text(const style::TextStyle &st, const QString &text, const TextParseOptions &options = _defaultOptions, int32 minResizeWidth = QFIXED_MAX, bool richText = false);
|
||||||
Text(const Text &other);
|
Text(const Text &other);
|
||||||
@ -135,8 +134,7 @@ public:
|
|||||||
};
|
};
|
||||||
Q_DECLARE_FLAGS(Flags, Flag);
|
Q_DECLARE_FLAGS(Flags, Flag);
|
||||||
|
|
||||||
StateRequest() {
|
StateRequest() = default;
|
||||||
}
|
|
||||||
|
|
||||||
style::align align = style::al_left;
|
style::align align = style::al_left;
|
||||||
Flags flags = Flag::LookupLink;
|
Flags flags = Flag::LookupLink;
|
||||||
@ -152,8 +150,7 @@ public:
|
|||||||
return getState(rtl() ? (outerw - x - width) : x, y, width, request);
|
return getState(rtl() ? (outerw - x - width) : x, y, width, request);
|
||||||
}
|
}
|
||||||
struct StateRequestElided : public StateRequest {
|
struct StateRequestElided : public StateRequest {
|
||||||
StateRequestElided() {
|
StateRequestElided() = default;
|
||||||
}
|
|
||||||
StateRequestElided(const StateRequest &other) : StateRequest(other) {
|
StateRequestElided(const StateRequest &other) : StateRequest(other) {
|
||||||
}
|
}
|
||||||
int lines = 1;
|
int lines = 1;
|
||||||
@ -205,6 +202,11 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
using TextBlocks = QVector<ITextBlock*>;
|
||||||
|
using TextLinks = QVector<ClickHandlerPtr>;
|
||||||
|
|
||||||
|
uint16 countBlockEnd(const TextBlocks::const_iterator &i, const TextBlocks::const_iterator &e) const;
|
||||||
|
uint16 countBlockLength(const Text::TextBlocks::const_iterator &i, const Text::TextBlocks::const_iterator &e) const;
|
||||||
|
|
||||||
// Template method for originalText(), originalTextWithEntities().
|
// Template method for originalText(), originalTextWithEntities().
|
||||||
template <typename AppendPartCallback, typename ClickHandlerStartCallback, typename ClickHandlerFinishCallback, typename FlagsChangeCallback>
|
template <typename AppendPartCallback, typename ClickHandlerStartCallback, typename ClickHandlerFinishCallback, typename FlagsChangeCallback>
|
||||||
@ -229,10 +231,7 @@ private:
|
|||||||
QString _text;
|
QString _text;
|
||||||
const style::TextStyle *_st = nullptr;
|
const style::TextStyle *_st = nullptr;
|
||||||
|
|
||||||
typedef QVector<ITextBlock*> TextBlocks;
|
|
||||||
TextBlocks _blocks;
|
TextBlocks _blocks;
|
||||||
|
|
||||||
typedef QVector<ClickHandlerPtr> TextLinks;
|
|
||||||
TextLinks _links;
|
TextLinks _links;
|
||||||
|
|
||||||
Qt::LayoutDirection _startDir = Qt::LayoutDirectionAuto;
|
Qt::LayoutDirection _startDir = Qt::LayoutDirectionAuto;
|
||||||
|
@ -350,6 +350,18 @@ TextBlock::TextBlock(const style::font &font, const QString &str, QFixed minResi
|
|||||||
EmojiBlock::EmojiBlock(const style::font &font, const QString &str, uint16 from, uint16 length, uchar flags, uint16 lnkIndex, EmojiPtr emoji) : ITextBlock(font, str, from, length, flags, lnkIndex), emoji(emoji) {
|
EmojiBlock::EmojiBlock(const style::font &font, const QString &str, uint16 from, uint16 length, uchar flags, uint16 lnkIndex, EmojiPtr emoji) : ITextBlock(font, str, from, length, flags, lnkIndex), emoji(emoji) {
|
||||||
_flags |= ((TextBlockTEmoji & 0x0F) << 8);
|
_flags |= ((TextBlockTEmoji & 0x0F) << 8);
|
||||||
_width = int(st::emojiSize + 2 * st::emojiPadding);
|
_width = int(st::emojiSize + 2 * st::emojiPadding);
|
||||||
|
|
||||||
|
auto padding = 0;
|
||||||
|
for (auto i = length; i != 0;) {
|
||||||
|
auto ch = str[_from + (--i)];
|
||||||
|
if (ch.unicode() == QChar::Space) {
|
||||||
|
padding += font->spacew;
|
||||||
|
} else {
|
||||||
|
_rpadding = padding;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_lpadding = padding;
|
||||||
}
|
}
|
||||||
|
|
||||||
SkipBlock::SkipBlock(const style::font &font, const QString &str, uint16 from, int32 w, int32 h, uint16 lnkIndex) : ITextBlock(font, str, from, 1, 0, lnkIndex), _height(h) {
|
SkipBlock::SkipBlock(const style::font &font, const QString &str, uint16 from, int32 w, int32 h, uint16 lnkIndex) : ITextBlock(font, str, from, 1, 0, lnkIndex), _height(h) {
|
||||||
|
@ -41,15 +41,7 @@ enum TextBlockFlags {
|
|||||||
|
|
||||||
class ITextBlock {
|
class ITextBlock {
|
||||||
public:
|
public:
|
||||||
ITextBlock(const style::font &font, const QString &str, uint16 from, uint16 length, uchar flags, uint16 lnkIndex) : _from(from), _flags((flags & 0xFF) | ((lnkIndex & 0xFFFF) << 12)), _lpadding(0) {
|
ITextBlock(const style::font &font, const QString &str, uint16 from, uint16 length, uchar flags, uint16 lnkIndex) : _from(from), _flags((flags & 0xFF) | ((lnkIndex & 0xFFFF) << 12)) {
|
||||||
if (length) {
|
|
||||||
if (str.at(_from + length - 1).unicode() == QChar::Space) {
|
|
||||||
_rpadding = font->spacew;
|
|
||||||
}
|
|
||||||
if (length > 1 && str.at(0).unicode() == QChar::Space) {
|
|
||||||
_lpadding = font->spacew;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16 from() const {
|
uint16 from() const {
|
||||||
@ -97,11 +89,13 @@ public:
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
uint16 _from;
|
uint16 _from = 0;
|
||||||
|
|
||||||
uint32 _flags; // 4 bits empty, 16 bits lnkIndex, 4 bits type, 8 bits flags
|
uint32 _flags = 0; // 4 bits empty, 16 bits lnkIndex, 4 bits type, 8 bits flags
|
||||||
|
|
||||||
QFixed _width, _lpadding, _rpadding;
|
QFixed _width = 0;
|
||||||
|
QFixed _lpadding = 0;
|
||||||
|
QFixed _rpadding = 0;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -163,13 +157,11 @@ private:
|
|||||||
|
|
||||||
class TextBlock : public ITextBlock {
|
class TextBlock : public ITextBlock {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
ITextBlock *clone() const {
|
ITextBlock *clone() const {
|
||||||
return new TextBlock(*this);
|
return new TextBlock(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
TextBlock(const style::font &font, const QString &str, QFixed minResizeWidth, uint16 from, uint16 length, uchar flags, uint16 lnkIndex);
|
TextBlock(const style::font &font, const QString &str, QFixed minResizeWidth, uint16 from, uint16 length, uchar flags, uint16 lnkIndex);
|
||||||
|
|
||||||
friend class ITextBlock;
|
friend class ITextBlock;
|
||||||
@ -185,17 +177,16 @@ private:
|
|||||||
|
|
||||||
friend class BlockParser;
|
friend class BlockParser;
|
||||||
friend class TextPainter;
|
friend class TextPainter;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class EmojiBlock : public ITextBlock {
|
class EmojiBlock : public ITextBlock {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
ITextBlock *clone() const {
|
ITextBlock *clone() const {
|
||||||
return new EmojiBlock(*this);
|
return new EmojiBlock(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
EmojiBlock(const style::font &font, const QString &str, uint16 from, uint16 length, uchar flags, uint16 lnkIndex, EmojiPtr emoji);
|
EmojiBlock(const style::font &font, const QString &str, uint16 from, uint16 length, uchar flags, uint16 lnkIndex, EmojiPtr emoji);
|
||||||
|
|
||||||
EmojiPtr emoji = nullptr;
|
EmojiPtr emoji = nullptr;
|
||||||
@ -204,11 +195,11 @@ private:
|
|||||||
friend class TextParser;
|
friend class TextParser;
|
||||||
|
|
||||||
friend class TextPainter;
|
friend class TextPainter;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class SkipBlock : public ITextBlock {
|
class SkipBlock : public ITextBlock {
|
||||||
public:
|
public:
|
||||||
|
|
||||||
int32 height() const {
|
int32 height() const {
|
||||||
return _height;
|
return _height;
|
||||||
}
|
}
|
||||||
@ -218,7 +209,6 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
SkipBlock(const style::font &font, const QString &str, uint16 from, int32 w, int32 h, uint16 lnkIndex);
|
SkipBlock(const style::font &font, const QString &str, uint16 from, int32 w, int32 h, uint16 lnkIndex);
|
||||||
|
|
||||||
int32 _height;
|
int32 _height;
|
||||||
@ -227,4 +217,5 @@ private:
|
|||||||
friend class TextParser;
|
friend class TextParser;
|
||||||
|
|
||||||
friend class TextPainter;
|
friend class TextPainter;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user