/* This file is part of Telegram Desktop, the official desktop application for the Telegram messaging service. For license and copyright information please follow this link: https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #include "dialogs/dialogs_row.h" #include "ui/effects/ripple_animation.h" #include "ui/text/text_options.h" #include "ui/text/text_utilities.h" #include "ui/painter.h" #include "dialogs/dialogs_entry.h" #include "dialogs/ui/dialogs_video_userpic.h" #include "dialogs/ui/dialogs_layout.h" #include "data/data_folder.h" #include "data/data_forum.h" #include "data/data_session.h" #include "data/data_peer_values.h" #include "history/history.h" #include "history/history_item.h" #include "lang/lang_keys.h" #include "base/unixtime.h" #include "mainwidget.h" #include "styles/style_dialogs.h" namespace Dialogs { BasicRow::BasicRow() = default; BasicRow::~BasicRow() = default; void BasicRow::addRipple( QPoint origin, QSize size, Fn updateCallback) { if (!_ripple) { addRippleWithMask( origin, Ui::RippleAnimation::RectMask(size), std::move(updateCallback)); } else { _ripple->add(origin); } } void BasicRow::addRippleWithMask( QPoint origin, QImage mask, Fn updateCallback) { _ripple = std::make_unique( st::dialogsRipple, std::move(mask), std::move(updateCallback)); _ripple->add(origin); } void BasicRow::clearRipple() { _ripple = nullptr; } void BasicRow::stopLastRipple() { if (_ripple) { _ripple->lastStop(); } } void BasicRow::paintRipple( QPainter &p, int x, int y, int outerWidth, const QColor *colorOverride) const { if (_ripple) { _ripple->paint(p, x, y, outerWidth, colorOverride); if (_ripple->empty()) { _ripple.reset(); } } } void BasicRow::paintUserpic( Painter &p, not_null peer, Ui::VideoUserpic *videoUserpic, History *historyForCornerBadge, const Ui::PaintContext &context) const { PaintUserpic( p, peer, videoUserpic, _userpic, context.st->padding.left(), context.st->padding.top(), context.width, context.st->photoSize, context.paused); } Row::Row(Key key, int index, int top) : _id(key), _top(top), _index(index) { if (const auto history = key.history()) { updateCornerBadgeShown(history->peer); } } void Row::recountHeight(float64 narrowRatio) { if (const auto history = _id.history()) { _height = history->peer->isForum() ? anim::interpolate( st::forumDialogRow.height, st::defaultDialogRow.height, narrowRatio) : st::defaultDialogRow.height; } else if (_id.folder()) { _height = st::defaultDialogRow.height; } else { _height = st::forumTopicRow.height; } } uint64 Row::sortKey(FilterId filterId) const { return _id.entry()->sortKeyInChatList(filterId); } void Row::setCornerBadgeShown( bool shown, Fn updateCallback) const { const auto value = (shown ? 1 : 0); if (_cornerBadgeShown == value) { return; } const_cast(this)->_cornerBadgeShown = value; if (_cornerBadgeUserpic && _cornerBadgeUserpic->animation.animating()) { _cornerBadgeUserpic->animation.change( _cornerBadgeShown ? 1. : 0., st::dialogsOnlineBadgeDuration); } else if (updateCallback) { ensureCornerBadgeUserpic(); _cornerBadgeUserpic->animation.start( std::move(updateCallback), _cornerBadgeShown ? 0. : 1., _cornerBadgeShown ? 1. : 0., st::dialogsOnlineBadgeDuration); } if (!_cornerBadgeShown && _cornerBadgeUserpic && !_cornerBadgeUserpic->animation.animating()) { _cornerBadgeUserpic = nullptr; } } void Row::updateCornerBadgeShown( not_null peer, Fn updateCallback) const { const auto user = peer->asUser(); const auto now = user ? base::unixtime::now() : TimeId(); const auto shown = [&] { if (user) { return Data::IsUserOnline(user, now); } else if (const auto channel = peer->asChannel()) { return Data::ChannelHasActiveCall(channel); } return false; }(); setCornerBadgeShown(shown, std::move(updateCallback)); if (shown && user) { peer->owner().watchForOffline(user, now); } } void Row::ensureCornerBadgeUserpic() const { if (_cornerBadgeUserpic) { return; } _cornerBadgeUserpic = std::make_unique(); } void Row::PaintCornerBadgeFrame( not_null data, not_null peer, Ui::VideoUserpic *videoUserpic, std::shared_ptr &view, const Ui::PaintContext &context) { data->frame.fill(Qt::transparent); Painter q(&data->frame); PaintUserpic( q, peer, videoUserpic, view, 0, 0, data->frame.width() / data->frame.devicePixelRatio(), context.st->photoSize, context.paused); PainterHighQualityEnabler hq(q); q.setCompositionMode(QPainter::CompositionMode_Source); const auto size = peer->isUser() ? st::dialogsOnlineBadgeSize : st::dialogsCallBadgeSize; const auto stroke = st::dialogsOnlineBadgeStroke; const auto skip = peer->isUser() ? st::dialogsOnlineBadgeSkip : st::dialogsCallBadgeSkip; const auto shrink = (size / 2) * (1. - data->shown); auto pen = QPen(Qt::transparent); pen.setWidthF(stroke * data->shown); q.setPen(pen); q.setBrush(data->active ? st::dialogsOnlineBadgeFgActive : st::dialogsOnlineBadgeFg); q.drawEllipse(QRectF( context.st->photoSize - skip.x() - size, context.st->photoSize - skip.y() - size, size, size ).marginsRemoved({ shrink, shrink, shrink, shrink })); } void Row::paintUserpic( Painter &p, not_null peer, Ui::VideoUserpic *videoUserpic, History *historyForCornerBadge, const Ui::PaintContext &context) const { updateCornerBadgeShown(peer); const auto shown = _cornerBadgeUserpic ? _cornerBadgeUserpic->animation.value(_cornerBadgeShown ? 1. : 0.) : (_cornerBadgeShown ? 1. : 0.); if (!historyForCornerBadge || shown == 0.) { BasicRow::paintUserpic( p, peer, videoUserpic, historyForCornerBadge, context); if (!historyForCornerBadge || !_cornerBadgeShown) { _cornerBadgeUserpic = nullptr; } return; } ensureCornerBadgeUserpic(); const auto ratio = style::DevicePixelRatio(); const auto added = std::max({ -st::dialogsCallBadgeSkip.x(), -st::dialogsCallBadgeSkip.y(), 0 }); const auto frameSide = (context.st->photoSize + added) * style::DevicePixelRatio(); const auto frameSize = QSize(frameSide, frameSide); if (_cornerBadgeUserpic->frame.size() != frameSize) { _cornerBadgeUserpic->frame = QImage( frameSize, QImage::Format_ARGB32_Premultiplied); _cornerBadgeUserpic->frame.setDevicePixelRatio(ratio); } const auto key = peer->userpicUniqueKey(userpicView()); const auto frameIndex = videoUserpic ? videoUserpic->frameIndex() : -1; if (_cornerBadgeUserpic->shown != shown || _cornerBadgeUserpic->key != key || _cornerBadgeUserpic->active != context.active || _cornerBadgeUserpic->frameIndex != frameIndex || videoUserpic) { _cornerBadgeUserpic->shown = shown; _cornerBadgeUserpic->key = key; _cornerBadgeUserpic->active = context.active; _cornerBadgeUserpic->frameIndex = frameIndex; PaintCornerBadgeFrame( _cornerBadgeUserpic.get(), peer, videoUserpic, userpicView(), context); } p.drawImage( context.st->padding.left(), context.st->padding.top(), _cornerBadgeUserpic->frame); if (historyForCornerBadge->peer->isUser()) { return; } const auto actionPainter = historyForCornerBadge->sendActionPainter(); const auto bg = context.active ? st::dialogsBgActive : st::dialogsBg; const auto size = st::dialogsCallBadgeSize; const auto skip = st::dialogsCallBadgeSkip; p.setOpacity(shown); p.translate(context.st->padding.left(), context.st->padding.top()); actionPainter->paintSpeaking( p, context.st->photoSize - skip.x() - size, context.st->photoSize - skip.y() - size, context.width, bg, context.now); p.translate(-context.st->padding.left(), -context.st->padding.top()); p.setOpacity(1.); } bool Row::lookupIsInTopicJump(int x, int y) const { const auto history = this->history(); return history && history->lastItemDialogsView().isInTopicJump(x, y); } void Row::stopLastRipple() { BasicRow::stopLastRipple(); const auto history = this->history(); const auto view = history ? &history->lastItemDialogsView() : nullptr; if (view) { view->stopLastRipple(); } } void Row::addTopicJumpRipple( QPoint origin, not_null topicJumpCache, Fn updateCallback) { const auto history = this->history(); const auto view = history ? &history->lastItemDialogsView() : nullptr; if (view) { view->addTopicJumpRipple( origin, topicJumpCache, std::move(updateCallback)); _topicJumpRipple = 1; } } void Row::clearTopicJumpRipple() { if (_topicJumpRipple) { clearRipple(); _topicJumpRipple = 0; } } bool Row::topicJumpRipple() const { return _topicJumpRipple != 0; } FakeRow::FakeRow( Key searchInChat, not_null item, Fn repaint) : _searchInChat(searchInChat) , _item(item) , _repaint(std::move(repaint)) { invalidateTopic(); } void FakeRow::invalidateTopic() { _topic = _item->topic(); if (_topic) { return; } else if (const auto rootId = _item->topicRootId()) { if (const auto forum = _item->history()->asForum()) { if (!forum->topicDeleted(rootId)) { forum->requestTopic(rootId, crl::guard(this, [=] { _topic = _item->topic(); if (_topic) { _repaint(); } })); } } } } const Ui::Text::String &FakeRow::name() const { if (_name.isEmpty()) { const auto from = _searchInChat ? _item->displayFrom() : nullptr; const auto peer = from ? from : _item->history()->peer.get(); _name.setText( st::semiboldTextStyle, peer->name(), Ui::NameTextOptions()); } return _name; } } // namespace Dialogs