lib_ui/ui/widgets/labels.cpp
2023-08-01 19:05:34 +02:00

1009 lines
27 KiB
C++

// This file is part of Desktop App Toolkit,
// a set of libraries for developing nice desktop applications.
//
// For license and copyright information please follow this link:
// https://github.com/desktop-app/legal/blob/master/LEGAL
//
#include "ui/widgets/labels.h"
#include "base/invoke_queued.h"
#include "ui/text/text_entity.h"
#include "ui/effects/animation_value.h"
#include "ui/widgets/popup_menu.h"
#include "ui/widgets/box_content_divider.h"
#include "ui/basic_click_handlers.h" // UrlClickHandler
#include "ui/inactive_press.h"
#include "ui/painter.h"
#include "base/qt/qt_common_adapters.h"
#include "styles/style_layers.h"
#include "styles/palette.h"
#include <QtWidgets/QApplication>
#include <QtGui/QClipboard>
#include <QtGui/QDrag>
#include <QtGui/QtEvents>
#include <QtCore/QMimeData>
namespace Ui {
namespace {
TextParseOptions _labelOptions = {
TextParseMultiline, // flags
0, // maxw
0, // maxh
Qt::LayoutDirectionAuto, // dir
};
TextParseOptions _labelMarkedOptions = {
TextParseMultiline | TextParseLinks | TextParseHashtags | TextParseMentions | TextParseBotCommands | TextParseMarkdown, // flags
0, // maxw
0, // maxh
Qt::LayoutDirectionAuto, // dir
};
} // namespace
CrossFadeAnimation::CrossFadeAnimation(
style::color bg,
Data &&was,
Data &&now)
: _bg(bg) {
const auto maxLines = qMax(was.lineWidths.size(), now.lineWidths.size());
auto fillDataTill = [&](Data &data) {
for (auto i = data.lineWidths.size(); i != maxLines; ++i) {
data.lineWidths.push_back(-1);
}
};
fillDataTill(was);
fillDataTill(now);
auto preparePart = [](const Data &data, int index, const Data &other) {
auto result = CrossFadeAnimation::Part();
auto lineWidth = data.lineWidths[index];
if (lineWidth < 0) {
lineWidth = other.lineWidths[index];
}
const auto pixelRatio = style::DevicePixelRatio();
auto fullWidth = data.full.width() / pixelRatio;
auto top = index * data.lineHeight + data.lineAddTop;
auto left = 0;
if (data.align & Qt::AlignHCenter) {
left += (fullWidth - lineWidth) / 2;
} else if (data.align & Qt::AlignRight) {
left += (fullWidth - lineWidth);
}
auto snapshotRect = data.full.rect().intersected(QRect(left * pixelRatio, top * pixelRatio, lineWidth * pixelRatio, data.font->height * pixelRatio));
if (!snapshotRect.isEmpty()) {
result.snapshot = PixmapFromImage(data.full.copy(snapshotRect));
result.snapshot.setDevicePixelRatio(pixelRatio);
}
result.position = data.position + QPoint(data.margin.left() + left, data.margin.top() + top);
return result;
};
for (int i = 0; i != maxLines; ++i) {
addLine(preparePart(was, i, now), preparePart(now, i, was));
}
}
void CrossFadeAnimation::addLine(Part was, Part now) {
_lines.push_back(Line(std::move(was), std::move(now)));
}
void CrossFadeAnimation::paintFrame(QPainter &p, float64 dt) {
auto progress = anim::linear(1., dt);
paintFrame(p, progress, 1. - progress, progress);
}
void CrossFadeAnimation::paintFrame(
QPainter &p,
float64 positionReady,
float64 alphaWas,
float64 alphaNow) {
if (_lines.isEmpty()) return;
for (const auto &line : std::as_const(_lines)) {
paintLine(p, line, positionReady, alphaWas, alphaNow);
}
}
void CrossFadeAnimation::paintLine(
QPainter &p,
const Line &line,
float64 positionReady,
float64 alphaWas,
float64 alphaNow) {
auto &snapshotWas = line.was.snapshot;
auto &snapshotNow = line.now.snapshot;
if (snapshotWas.isNull() && snapshotNow.isNull()) {
// This can happen if both labels have an empty line or if one
// label has an empty line where the second one already ended.
// In this case lineWidth is zero and snapshot is null.
return;
}
const auto pixelRatio = style::DevicePixelRatio();
auto positionWas = line.was.position;
auto positionNow = line.now.position;
auto left = anim::interpolate(positionWas.x(), positionNow.x(), positionReady);
auto topDelta = (snapshotNow.height() / pixelRatio) - (snapshotWas.height() / pixelRatio);
auto widthDelta = (snapshotNow.width() / pixelRatio) - (snapshotWas.width() / pixelRatio);
auto topWas = anim::interpolate(positionWas.y(), positionNow.y() + topDelta, positionReady);
auto topNow = topWas - topDelta;
p.setOpacity(alphaWas);
if (!snapshotWas.isNull()) {
p.drawPixmap(left, topWas, snapshotWas);
if (topDelta > 0) {
p.fillRect(left, topWas - topDelta, snapshotWas.width() / pixelRatio, topDelta, _bg);
}
if (widthDelta > 0) {
p.fillRect(left + (snapshotWas.width() / pixelRatio), topNow, widthDelta, snapshotNow.height() / pixelRatio, _bg);
}
}
p.setOpacity(alphaNow);
if (!snapshotNow.isNull()) {
p.drawPixmap(left, topNow, snapshotNow);
if (topDelta < 0) {
p.fillRect(left, topNow + topDelta, snapshotNow.width() / pixelRatio, -topDelta, _bg);
}
if (widthDelta < 0) {
p.fillRect(left + (snapshotNow.width() / pixelRatio), topWas, -widthDelta, snapshotWas.height() / pixelRatio, _bg);
}
}
}
LabelSimple::LabelSimple(
QWidget *parent,
const style::LabelSimple &st,
const QString &value)
: RpWidget(parent)
, _st(st) {
setText(value);
}
void LabelSimple::setText(const QString &value, bool *outTextChanged) {
if (_fullText == value) {
if (outTextChanged) *outTextChanged = false;
return;
}
_fullText = value;
_fullTextWidth = _st.font->width(_fullText);
if (!_st.maxWidth || _fullTextWidth <= _st.maxWidth) {
_text = _fullText;
_textWidth = _fullTextWidth;
} else {
auto newText = _st.font->elided(_fullText, _st.maxWidth);
if (newText == _text) {
if (outTextChanged) *outTextChanged = false;
return;
}
_text = newText;
_textWidth = _st.font->width(_text);
}
resize(_textWidth, _st.font->height);
update();
if (outTextChanged) *outTextChanged = true;
}
void LabelSimple::paintEvent(QPaintEvent *e) {
Painter p(this);
p.setFont(_st.font);
p.setPen(_st.textFg);
p.drawTextLeft(0, 0, width(), _text, _textWidth);
}
FlatLabel::FlatLabel(
QWidget *parent,
const style::FlatLabel &st,
const style::PopupMenu &stMenu)
: RpWidget(parent)
, _text(st.minWidth ? st.minWidth : QFIXED_MAX)
, _st(st)
, _stMenu(stMenu) {
init();
}
FlatLabel::FlatLabel(
QWidget *parent,
const QString &text,
const style::FlatLabel &st,
const style::PopupMenu &stMenu)
: RpWidget(parent)
, _text(st.minWidth ? st.minWidth : QFIXED_MAX)
, _st(st)
, _stMenu(stMenu) {
setText(text);
init();
}
FlatLabel::FlatLabel(
QWidget *parent,
rpl::producer<QString> &&text,
const style::FlatLabel &st,
const style::PopupMenu &stMenu)
: RpWidget(parent)
, _text(st.minWidth ? st.minWidth : QFIXED_MAX)
, _st(st)
, _stMenu(stMenu) {
textUpdated();
std::move(
text
) | rpl::start_with_next([this](const QString &value) {
setText(value);
}, lifetime());
init();
}
FlatLabel::FlatLabel(
QWidget *parent,
rpl::producer<TextWithEntities> &&text,
const style::FlatLabel &st,
const style::PopupMenu &stMenu)
: RpWidget(parent)
, _text(st.minWidth ? st.minWidth : QFIXED_MAX)
, _st(st)
, _stMenu(stMenu)
, _touchSelectTimer([=] { touchSelect(); }) {
textUpdated();
std::move(
text
) | rpl::start_with_next([this](const TextWithEntities &value) {
setMarkedText(value);
}, lifetime());
init();
}
void FlatLabel::init() {
_contextCopyText = Integration::Instance().phraseContextCopyText();
}
void FlatLabel::textUpdated() {
refreshSize();
setMouseTracking(_selectable || _text.hasLinks());
if (_text.hasSpoilers()) {
_text.setSpoilerLinkFilter([weak = Ui::MakeWeak(this)](
const ClickContext &context) {
return (context.button == Qt::LeftButton) && weak;
});
}
update();
}
void FlatLabel::setText(const QString &text) {
_text.setText(_st.style, text, _labelOptions);
textUpdated();
}
void FlatLabel::setMarkedText(
const TextWithEntities &textWithEntities,
const std::any &context) {
_text.setMarkedText(
_st.style,
textWithEntities,
_labelMarkedOptions,
context);
textUpdated();
}
void FlatLabel::setSelectable(bool selectable) {
if (_selectable != selectable) {
_selection = { 0, 0 };
_savedSelection = { 0, 0 };
_selectable = selectable;
setMouseTracking(_selectable || _text.hasLinks());
}
}
void FlatLabel::setDoubleClickSelectsParagraph(bool doubleClickSelectsParagraph) {
_doubleClickSelectsParagraph = doubleClickSelectsParagraph;
}
void FlatLabel::setContextCopyText(const QString &copyText) {
_contextCopyText = copyText;
}
void FlatLabel::setBreakEverywhere(bool breakEverywhere) {
_breakEverywhere = breakEverywhere;
}
void FlatLabel::setTryMakeSimilarLines(bool tryMakeSimilarLines) {
_tryMakeSimilarLines = tryMakeSimilarLines;
}
int FlatLabel::resizeGetHeight(int newWidth) {
_allowedWidth = newWidth;
_textWidth = countTextWidth();
return countTextHeight(_textWidth);
}
int FlatLabel::textMaxWidth() const {
return _text.maxWidth();
}
int FlatLabel::naturalWidth() const {
return (_st.align == style::al_top) ? -1 : textMaxWidth();
}
QMargins FlatLabel::getMargins() const {
return _st.margin;
}
int FlatLabel::countTextWidth() const {
const auto available = _allowedWidth
? _allowedWidth
: (_st.minWidth ? _st.minWidth : _text.maxWidth());
if (_allowedWidth > 0
&& _allowedWidth < _text.maxWidth()
&& _tryMakeSimilarLines) {
auto large = _allowedWidth;
auto small = _allowedWidth / 2;
const auto largeHeight = _text.countHeight(large, _breakEverywhere);
while (large - small > 1) {
const auto middle = (large + small) / 2;
if (largeHeight == _text.countHeight(middle, _breakEverywhere)) {
large = middle;
} else {
small = middle;
}
}
return large;
}
return available;
}
int FlatLabel::countTextHeight(int textWidth) {
_fullTextHeight = _text.countHeight(textWidth, _breakEverywhere);
return _st.maxHeight
? qMin(_fullTextHeight, _st.maxHeight)
: _fullTextHeight;
}
void FlatLabel::refreshSize() {
int textWidth = countTextWidth();
int textHeight = countTextHeight(textWidth);
int fullWidth = _st.margin.left() + textWidth + _st.margin.right();
int fullHeight = _st.margin.top() + textHeight + _st.margin.bottom();
resize(fullWidth, fullHeight);
}
void FlatLabel::setLink(uint16 lnkIndex, const ClickHandlerPtr &lnk) {
_text.setLink(lnkIndex, lnk);
}
void FlatLabel::setLinksTrusted() {
static const auto TrustedLinksFilter = [](
const ClickHandlerPtr &link,
Qt::MouseButton button) {
if (const auto url = dynamic_cast<UrlClickHandler*>(link.get())) {
url->UrlClickHandler::onClick({ button });
return false;
}
return true;
};
setClickHandlerFilter(TrustedLinksFilter);
}
void FlatLabel::setClickHandlerFilter(ClickHandlerFilter &&filter) {
_clickHandlerFilter = std::move(filter);
}
void FlatLabel::overrideLinkClickHandler(Fn<void()> handler) {
setClickHandlerFilter([=](
const ClickHandlerPtr &link,
Qt::MouseButton button) {
if (button != Qt::LeftButton) {
return true;
}
handler();
return false;
});
}
void FlatLabel::overrideLinkClickHandler(Fn<void(QString url)> handler) {
setClickHandlerFilter([=](
const ClickHandlerPtr &link,
Qt::MouseButton button) {
if (button != Qt::LeftButton) {
return true;
}
handler(link->url());
return false;
});
}
void FlatLabel::setContextMenuHook(Fn<void(ContextMenuRequest)> hook) {
_contextMenuHook = std::move(hook);
}
void FlatLabel::mouseMoveEvent(QMouseEvent *e) {
_lastMousePos = e->globalPos();
dragActionUpdate();
}
void FlatLabel::mousePressEvent(QMouseEvent *e) {
if (_contextMenu) {
e->accept();
return; // ignore mouse press, that was hiding context menu
}
dragActionStart(e->globalPos(), e->button());
}
Text::StateResult FlatLabel::dragActionStart(const QPoint &p, Qt::MouseButton button) {
_lastMousePos = p;
auto state = dragActionUpdate();
if (button != Qt::LeftButton) return state;
ClickHandler::pressed();
_dragAction = NoDrag;
_dragWasInactive = WasInactivePress(window());
if (_dragWasInactive) {
MarkInactivePress(window(), false);
}
if (ClickHandler::getPressed()) {
_dragStartPosition = mapFromGlobal(_lastMousePos);
_dragAction = PrepareDrag;
}
if (!_selectable || _dragAction != NoDrag) {
return state;
}
if (_trippleClickTimer.isActive() && (_lastMousePos - _trippleClickPoint).manhattanLength() < QApplication::startDragDistance()) {
if (state.uponSymbol) {
_selection = { state.symbol, state.symbol };
_savedSelection = { 0, 0 };
_dragSymbol = state.symbol;
_dragAction = Selecting;
_selectionType = TextSelectType::Paragraphs;
updateHover(state);
_trippleClickTimer.callOnce(QApplication::doubleClickInterval());
update();
}
}
if (_selectionType != TextSelectType::Paragraphs) {
_dragSymbol = state.symbol;
bool uponSelected = state.uponSymbol;
if (uponSelected) {
if (_dragSymbol < _selection.from || _dragSymbol >= _selection.to) {
uponSelected = false;
}
}
if (uponSelected) {
_dragStartPosition = mapFromGlobal(_lastMousePos);
_dragAction = PrepareDrag; // start text drag
} else if (!_dragWasInactive) {
if (state.afterSymbol) ++_dragSymbol;
_selection = { _dragSymbol, _dragSymbol };
_savedSelection = { 0, 0 };
_dragAction = Selecting;
update();
}
}
return state;
}
Text::StateResult FlatLabel::dragActionFinish(const QPoint &p, Qt::MouseButton button) {
_lastMousePos = p;
auto state = dragActionUpdate();
auto activated = ClickHandler::unpressed();
if (_dragAction == Dragging) {
activated = nullptr;
} else if (_dragAction == PrepareDrag) {
_selection = { 0, 0 };
_savedSelection = { 0, 0 };
update();
}
_dragAction = NoDrag;
_selectionType = TextSelectType::Letters;
if (activated) {
// _clickHandlerFilter may delete `this`. In that case we don't want
// to try to show a context menu or smth like that.
crl::on_main(this, [=] {
const auto guard = window();
if (!_clickHandlerFilter
|| _clickHandlerFilter(activated, button)) {
ActivateClickHandler(guard, activated, button);
}
});
}
if (QGuiApplication::clipboard()->supportsSelection()
&& !_selection.empty()) {
TextUtilities::SetClipboardText(
_text.toTextForMimeData(_selection),
QClipboard::Selection);
}
return state;
}
void FlatLabel::mouseReleaseEvent(QMouseEvent *e) {
dragActionFinish(e->globalPos(), e->button());
if (!rect().contains(e->pos())) {
leaveEvent(e);
}
}
void FlatLabel::mouseDoubleClickEvent(QMouseEvent *e) {
auto state = dragActionStart(e->globalPos(), e->button());
if (((_dragAction == Selecting) || (_dragAction == NoDrag)) && _selectionType == TextSelectType::Letters) {
if (state.uponSymbol) {
_dragSymbol = state.symbol;
_selectionType = _doubleClickSelectsParagraph ? TextSelectType::Paragraphs : TextSelectType::Words;
if (_dragAction == NoDrag) {
_dragAction = Selecting;
_selection = { state.symbol, state.symbol };
_savedSelection = { 0, 0 };
}
mouseMoveEvent(e);
_trippleClickPoint = e->globalPos();
_trippleClickTimer.callOnce(QApplication::doubleClickInterval());
}
}
}
void FlatLabel::enterEventHook(QEnterEvent *e) {
_lastMousePos = QCursor::pos();
dragActionUpdate();
}
void FlatLabel::leaveEventHook(QEvent *e) {
ClickHandler::clearActive(this);
}
void FlatLabel::focusOutEvent(QFocusEvent *e) {
if (!_selection.empty()) {
if (_contextMenu) {
_savedSelection = _selection;
}
_selection = { 0, 0 };
update();
}
}
void FlatLabel::focusInEvent(QFocusEvent *e) {
if (!_savedSelection.empty()) {
_selection = _savedSelection;
_savedSelection = { 0, 0 };
update();
}
}
void FlatLabel::keyPressEvent(QKeyEvent *e) {
e->ignore();
if (e->key() == Qt::Key_Copy || (e->key() == Qt::Key_C && e->modifiers().testFlag(Qt::ControlModifier))) {
if (!_selection.empty()) {
copySelectedText();
e->accept();
}
#ifdef Q_OS_MAC
} else if (e->key() == Qt::Key_E && e->modifiers().testFlag(Qt::ControlModifier)) {
auto selection = _selection.empty() ? (_contextMenu ? _savedSelection : _selection) : _selection;
if (!selection.empty()) {
TextUtilities::SetClipboardText(_text.toTextForMimeData(selection), QClipboard::FindBuffer);
}
#endif // Q_OS_MAC
}
}
void FlatLabel::contextMenuEvent(QContextMenuEvent *e) {
if (!_contextMenuHook && !_selectable && !_text.hasLinks()) {
return;
}
showContextMenu(e, ContextMenuReason::FromEvent);
}
bool FlatLabel::eventHook(QEvent *e) {
if (e->type() == QEvent::TouchBegin || e->type() == QEvent::TouchUpdate || e->type() == QEvent::TouchEnd || e->type() == QEvent::TouchCancel) {
QTouchEvent *ev = static_cast<QTouchEvent*>(e);
if (ev->device()->type() == base::TouchDevice::TouchScreen) {
touchEvent(ev);
return true;
}
}
return RpWidget::eventHook(e);
}
void FlatLabel::touchEvent(QTouchEvent *e) {
if (e->type() == QEvent::TouchCancel) { // cancel
if (!_touchInProgress) return;
_touchInProgress = false;
_touchSelectTimer.cancel();
_touchSelect = false;
_dragAction = NoDrag;
return;
}
if (!e->touchPoints().isEmpty()) {
_touchPrevPos = _touchPos;
_touchPos = e->touchPoints().cbegin()->screenPos().toPoint();
}
switch (e->type()) {
case QEvent::TouchBegin: {
if (_contextMenu) {
e->accept();
return; // ignore mouse press, that was hiding context menu
}
if (_touchInProgress) return;
if (e->touchPoints().isEmpty()) return;
_touchInProgress = true;
_touchSelectTimer.callOnce(QApplication::startDragTime());
_touchSelect = false;
_touchStart = _touchPrevPos = _touchPos;
} break;
case QEvent::TouchUpdate: {
if (!_touchInProgress) return;
if (_touchSelect) {
_lastMousePos = _touchPos;
dragActionUpdate();
}
} break;
case QEvent::TouchEnd: {
if (!_touchInProgress) return;
_touchInProgress = false;
auto weak = MakeWeak(this);
if (_touchSelect) {
dragActionFinish(_touchPos, Qt::RightButton);
QContextMenuEvent contextMenu(QContextMenuEvent::Mouse, mapFromGlobal(_touchPos), _touchPos);
showContextMenu(&contextMenu, ContextMenuReason::FromTouch);
} else { // one short tap -- like mouse click
dragActionStart(_touchPos, Qt::LeftButton);
dragActionFinish(_touchPos, Qt::LeftButton);
}
if (weak) {
_touchSelectTimer.cancel();
_touchSelect = false;
}
} break;
}
}
void FlatLabel::showContextMenu(QContextMenuEvent *e, ContextMenuReason reason) {
if (e->reason() == QContextMenuEvent::Mouse) {
_lastMousePos = e->globalPos();
} else {
_lastMousePos = QCursor::pos();
}
const auto state = dragActionUpdate();
const auto hasSelection = _selectable && !_selection.empty();
const auto uponSelection = _selectable
&& ((reason == ContextMenuReason::FromTouch && hasSelection)
|| (state.uponSymbol
&& (state.symbol >= _selection.from)
&& (state.symbol < _selection.to)));
_contextMenu = base::make_unique_q<PopupMenu>(this, _stMenu);
const auto request = ContextMenuRequest{
.menu = _contextMenu.get(),
.link = ClickHandler::getActive(),
.hasSelection = hasSelection,
.uponSelection = uponSelection,
.fullSelection = _selectable && _text.isFullSelection(_selection),
};
if (_contextMenuHook) {
_contextMenuHook(request);
} else {
fillContextMenu(request);
}
if (_contextMenu->empty()) {
_contextMenu = nullptr;
} else {
_contextMenu->popup(e->globalPos());
e->accept();
}
}
void FlatLabel::fillContextMenu(ContextMenuRequest request) {
if (request.fullSelection && !_contextCopyText.isEmpty()) {
request.menu->addAction(
_contextCopyText,
[=] { copyContextText(); });
} else if (request.uponSelection && !request.fullSelection) {
request.menu->addAction(
Integration::Instance().phraseContextCopySelected(),
[=] { copySelectedText(); });
} else if (_selectable
&& !request.hasSelection
&& !_contextCopyText.isEmpty()) {
request.menu->addAction(
_contextCopyText,
[=] { copyContextText(); });
}
if (request.link) {
const auto label = request.link->copyToClipboardContextItemText();
if (!label.isEmpty()) {
request.menu->addAction(
label,
[text = request.link->copyToClipboardText()] {
QGuiApplication::clipboard()->setText(text);
});
}
}
}
void FlatLabel::copySelectedText() {
const auto selection = _selection.empty() ? (_contextMenu ? _savedSelection : _selection) : _selection;
if (!selection.empty()) {
TextUtilities::SetClipboardText(_text.toTextForMimeData(selection));
}
}
void FlatLabel::copyContextText() {
TextUtilities::SetClipboardText(_text.toTextForMimeData());
}
void FlatLabel::touchSelect() {
_touchSelect = true;
dragActionStart(_touchPos, Qt::LeftButton);
}
void FlatLabel::executeDrag() {
if (_dragAction != Dragging) return;
auto state = getTextState(_dragStartPosition);
bool uponSelected = state.uponSymbol && _selection.from <= state.symbol;
if (uponSelected) {
if (_dragSymbol < _selection.from || _dragSymbol >= _selection.to) {
uponSelected = false;
}
}
const auto pressedHandler = ClickHandler::getPressed();
const auto selectedText = [&] {
if (uponSelected) {
return _text.toTextForMimeData(_selection);
} else if (pressedHandler) {
return TextForMimeData::Simple(pressedHandler->dragText());
}
return TextForMimeData();
}();
if (auto mimeData = TextUtilities::MimeDataFromText(selectedText)) {
auto drag = new QDrag(window());
drag->setMimeData(mimeData.release());
drag->exec(Qt::CopyAction);
// We don't receive mouseReleaseEvent when drag is finished.
ClickHandler::unpressed();
}
}
void FlatLabel::clickHandlerActiveChanged(const ClickHandlerPtr &action, bool active) {
update();
}
void FlatLabel::clickHandlerPressedChanged(const ClickHandlerPtr &action, bool active) {
update();
}
CrossFadeAnimation::Data FlatLabel::crossFadeData(
style::color bg,
QPoint basePosition) {
auto result = CrossFadeAnimation::Data();
result.full = GrabWidgetToImage(this, QRect(), bg->c);
const auto textWidth = width() - _st.margin.left() - _st.margin.right();
_text.countLineWidths(textWidth, &result.lineWidths, _breakEverywhere);
result.lineHeight = _st.style.font->height;
const auto addedHeight = (_st.style.lineHeight - result.lineHeight);
if (addedHeight > 0) {
result.lineAddTop = addedHeight / 2;
result.lineHeight += addedHeight;
}
result.position = basePosition + pos();
result.align = _st.align;
result.font = _st.style.font;
result.margin = _st.margin;
return result;
}
std::unique_ptr<CrossFadeAnimation> FlatLabel::CrossFade(
not_null<FlatLabel*> from,
not_null<FlatLabel*> to,
style::color bg,
QPoint fromPosition,
QPoint toPosition) {
return std::make_unique<CrossFadeAnimation>(
bg,
from->crossFadeData(bg, fromPosition),
to->crossFadeData(bg, toPosition));
}
Text::StateResult FlatLabel::dragActionUpdate() {
auto m = mapFromGlobal(_lastMousePos);
auto state = getTextState(m);
updateHover(state);
if (_dragAction == PrepareDrag && (m - _dragStartPosition).manhattanLength() >= QApplication::startDragDistance()) {
_dragAction = Dragging;
InvokeQueued(this, [=] { executeDrag(); });
}
return state;
}
void FlatLabel::updateHover(const Text::StateResult &state) {
bool lnkChanged = ClickHandler::setActive(state.link, this);
if (!_selectable) {
refreshCursor(state.uponSymbol);
return;
}
Qt::CursorShape cur = style::cur_default;
if (_dragAction == NoDrag) {
if (state.link) {
cur = style::cur_pointer;
} else if (state.uponSymbol) {
cur = style::cur_text;
}
} else {
if (_dragAction == Selecting) {
uint16 second = state.symbol;
if (state.afterSymbol && _selectionType == TextSelectType::Letters) {
++second;
}
auto selection = _text.adjustSelection({ qMin(second, _dragSymbol), qMax(second, _dragSymbol) }, _selectionType);
if (_selection != selection) {
_selection = selection;
_savedSelection = { 0, 0 };
setFocus();
update();
}
} else if (_dragAction == Dragging) {
}
if (ClickHandler::getPressed()) {
cur = style::cur_pointer;
} else if (_dragAction == Selecting) {
cur = style::cur_text;
}
}
if (_dragAction == Selecting) {
// checkSelectingScroll();
} else {
// noSelectingScroll();
}
if (_dragAction == NoDrag && (lnkChanged || cur != _cursor)) {
setCursor(_cursor = cur);
}
}
void FlatLabel::refreshCursor(bool uponSymbol) {
if (_dragAction != NoDrag) {
return;
}
bool needTextCursor = _selectable && uponSymbol;
style::cursor newCursor = needTextCursor ? style::cur_text : style::cur_default;
if (ClickHandler::getActive()) {
newCursor = style::cur_pointer;
}
if (newCursor != _cursor) {
_cursor = newCursor;
setCursor(_cursor);
}
}
Text::StateResult FlatLabel::getTextState(const QPoint &m) const {
Text::StateRequestElided request;
request.align = _st.align;
if (_selectable) {
request.flags |= Text::StateRequest::Flag::LookupSymbol;
}
int textWidth = width() - _st.margin.left() - _st.margin.right();
Text::StateResult state;
bool heightExceeded = _st.maxHeight && (_st.maxHeight < _fullTextHeight || textWidth < _text.maxWidth());
bool renderElided = _breakEverywhere || heightExceeded;
if (renderElided) {
auto lineHeight = qMax(_st.style.lineHeight, _st.style.font->height);
auto lines = _st.maxHeight ? qMax(_st.maxHeight / lineHeight, 1) : ((height() / lineHeight) + 2);
request.lines = lines;
if (_breakEverywhere) {
request.flags |= Text::StateRequest::Flag::BreakEverywhere;
}
state = _text.getStateElided(m - QPoint(_st.margin.left(), _st.margin.top()), textWidth, request);
} else {
state = _text.getState(m - QPoint(_st.margin.left(), _st.margin.top()), textWidth, request);
}
return state;
}
void FlatLabel::setOpacity(float64 o) {
_opacity = o;
update();
}
void FlatLabel::setTextColorOverride(std::optional<QColor> color) {
_textColorOverride = color;
update();
}
void FlatLabel::paintEvent(QPaintEvent *e) {
Painter p(this);
p.setOpacity(_opacity);
if (_textColorOverride) {
p.setPen(*_textColorOverride);
} else {
p.setPen(_st.textFg);
}
p.setTextPalette(_st.palette);
const auto textWidth = _textWidth
? _textWidth
: (width() - _st.margin.left() - _st.margin.right());
const auto textLeft = _textWidth
? ((_st.align & Qt::AlignLeft)
? _st.margin.left()
: (_st.align & Qt::AlignHCenter)
? ((width() - _textWidth) / 2)
: (width() - _st.margin.right() - _textWidth))
: _st.margin.left();
const auto selection = !_selection.empty()
? _selection
: _contextMenu
? _savedSelection
: _selection;
const auto heightExceeded = _st.maxHeight
&& (_st.maxHeight < _fullTextHeight || textWidth < _text.maxWidth());
const auto renderElided = _breakEverywhere || heightExceeded;
const auto lineHeight = qMax(_st.style.lineHeight, _st.style.font->height);
const auto lines = !renderElided
? 0
: _st.maxHeight
? qMax(_st.maxHeight / lineHeight, 1)
: ((height() / lineHeight) + 2);
const auto paused = _animationsPausedCallback
? _animationsPausedCallback()
: WhichAnimationsPaused::None;
_text.draw(p, {
.position = { textLeft, _st.margin.top() },
.availableWidth = textWidth,
.align = _st.align,
.clip = e->rect(),
.palette = &_st.palette,
.spoiler = Text::DefaultSpoilerCache(),
.now = crl::now(),
.pausedEmoji = (paused == WhichAnimationsPaused::CustomEmoji
|| paused == WhichAnimationsPaused::All),
.pausedSpoiler = (paused == WhichAnimationsPaused::Spoiler
|| paused == WhichAnimationsPaused::All),
.selection = selection,
.elisionLines = lines,
.elisionBreakEverywhere = renderElided && _breakEverywhere,
});
}
DividerLabel::DividerLabel(
QWidget *parent,
object_ptr<RpWidget> &&child,
const style::margins &padding,
RectParts parts)
: PaddingWrap(parent, std::move(child), padding)
, _background(this, st::boxDividerHeight, st::boxDividerBg, parts) {
}
int DividerLabel::naturalWidth() const {
return -1;
}
void DividerLabel::resizeEvent(QResizeEvent *e) {
_background->lower();
_background->setGeometry(rect());
return PaddingWrap::resizeEvent(e);
}
} // namespace Ui