init
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
This commit is contained in:
785
Telegram/SourceFiles/dialogs/dialogs_row.cpp
Normal file
785
Telegram/SourceFiles/dialogs/dialogs_row.cpp
Normal file
@@ -0,0 +1,785 @@
|
||||
/*
|
||||
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/chat/chat_theme.h" // CountAverageColor.
|
||||
#include "ui/color_contrast.h"
|
||||
#include "ui/effects/credits_graphics.h"
|
||||
#include "ui/effects/outline_segments.h"
|
||||
#include "ui/effects/ripple_animation.h"
|
||||
#include "ui/effects/round_checkbox.h"
|
||||
#include "ui/image/image_prepare.h"
|
||||
#include "ui/text/custom_emoji_text_badge.h"
|
||||
#include "ui/text/format_values.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_stories.h"
|
||||
#include "data/data_peer_values.h"
|
||||
#include "data/data_user.h"
|
||||
#include "history/history.h"
|
||||
#include "history/history_item.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "base/unixtime.h"
|
||||
#include "styles/style_calls.h"
|
||||
#include "styles/style_dialogs.h"
|
||||
|
||||
namespace Dialogs {
|
||||
namespace {
|
||||
|
||||
constexpr auto kTopLayer = 2;
|
||||
constexpr auto kBottomLayer = 1;
|
||||
constexpr auto kNoneLayer = 0;
|
||||
constexpr auto kBlurRadius = 24;
|
||||
|
||||
[[nodiscard]] const QPainterPath &SubscriptionOutlinePath() {
|
||||
static auto path = QPainterPath();
|
||||
if (!path.isEmpty()) {
|
||||
return path;
|
||||
}
|
||||
const auto scaledMoveTo = [&](float64 x, float64 y) {
|
||||
path.moveTo(style::ConvertFloatScale(x), style::ConvertFloatScale(y));
|
||||
};
|
||||
const auto scaledLineTo = [&](float64 x, float64 y) {
|
||||
path.lineTo(style::ConvertFloatScale(x), style::ConvertFloatScale(y));
|
||||
};
|
||||
const auto scaledCubicTo = [&](
|
||||
float64 x1,
|
||||
float64 y1,
|
||||
float64 x2,
|
||||
float64 y2,
|
||||
float64 x3,
|
||||
float64 y3) {
|
||||
path.cubicTo(
|
||||
style::ConvertFloatScale(x1),
|
||||
style::ConvertFloatScale(y1),
|
||||
style::ConvertFloatScale(x2),
|
||||
style::ConvertFloatScale(y2),
|
||||
style::ConvertFloatScale(x3),
|
||||
style::ConvertFloatScale(y3));
|
||||
};
|
||||
const auto scaledTranslate = [&](float64 x, float64 y) {
|
||||
path.translate(
|
||||
style::ConvertFloatScale(x),
|
||||
style::ConvertFloatScale(y));
|
||||
};
|
||||
|
||||
scaledMoveTo(42.3009, 18.3345);
|
||||
scaledLineTo(44.3285, 14.1203);
|
||||
scaledCubicTo(44.6152, 13.6549, 45.7858, 13.3542, 46.1909, 13.5523);
|
||||
scaledCubicTo(46.3355, 13.6044, 47.0064, 13.7541, 47.3833, 14.5053);
|
||||
scaledLineTo(49.3924 * 1.0071, 18.4206 * 0.9905);
|
||||
// 49.5459 * 1.007, 18.7336 * 0.9897.
|
||||
scaledCubicTo(49.8927213, 18.5406439, 52.5473, 18.8491, 53.3141, 18.8789);
|
||||
scaledCubicTo(53.6484, 18.8441, 55.8914, 20.0065, 54.3752, 20.7818);
|
||||
scaledCubicTo(54.1725, 20.8744, 41.3467, 31.3217, 41.3467, 31.3217);
|
||||
scaledCubicTo(40.7918, 31.5944, 41.2661, 31.4116, 40.8968, 30.9483);
|
||||
scaledCubicTo(39.9809, 30.3111, 40.0577, 25.4542, 40.1925, 25.5408);
|
||||
scaledCubicTo(39.9835, 25.6454, 38.4545, 22.9776, 37.8121, 22.3477);
|
||||
scaledLineTo(37.3236, 21.4448);
|
||||
scaledCubicTo(37.0943, 20.8845, 37.2524, 20.4742, 37.4164, 19.7765);
|
||||
scaledCubicTo(37.4703, 19.4582, 38.1756, 19.0759, 38.4504, 19.0422);
|
||||
scaledLineTo(41.6566, 18.6449);
|
||||
scaledCubicTo(41.5344, 18.6041, 42.2622, 18.6087, 42.3009, 18.3345);
|
||||
scaledTranslate(-42.3009, -18.3345);
|
||||
scaledTranslate(1.2, 0.4);
|
||||
|
||||
return path;
|
||||
}
|
||||
|
||||
[[nodiscard]] const QImage &SubscriptionIcon() {
|
||||
static auto starImage = QImage();
|
||||
if (!starImage.isNull()) {
|
||||
return starImage;
|
||||
}
|
||||
starImage = Ui::GenerateStars(st::dialogsSubscriptionBadgeSize, 1);
|
||||
return starImage;
|
||||
}
|
||||
|
||||
[[nodiscard]] QImage CornerBadgeTTL(
|
||||
not_null<PeerData*> peer,
|
||||
Ui::PeerUserpicView &view,
|
||||
int photoSize) {
|
||||
const auto ttl = peer->messagesTTL();
|
||||
if (!ttl) {
|
||||
return QImage();
|
||||
}
|
||||
const auto ratio = style::DevicePixelRatio();
|
||||
const auto fullSize = photoSize;
|
||||
const auto partRect = CornerBadgeTTLRect(fullSize);
|
||||
const auto &partSize = partRect.width();
|
||||
const auto partSkip = fullSize - partSize;
|
||||
auto result = Images::Circle(BlurredDarkenedPart(
|
||||
PeerData::GenerateUserpicImage(peer, view, fullSize * ratio, 0),
|
||||
QRect(
|
||||
QPoint(partSkip, partSkip) * ratio,
|
||||
QSize(partSize, partSize) * ratio)));
|
||||
result.setDevicePixelRatio(ratio);
|
||||
|
||||
auto q = QPainter(&result);
|
||||
PainterHighQualityEnabler hq(q);
|
||||
|
||||
const auto innerRect = QRect(QPoint(), partRect.size())
|
||||
- st::dialogsTTLBadgeInnerMargins;
|
||||
const auto ttlText = Ui::FormatTTLTiny(ttl);
|
||||
|
||||
q.setFont(st::dialogsScamFont);
|
||||
q.setPen(st::premiumButtonFg);
|
||||
q.drawText(
|
||||
innerRect,
|
||||
(ttlText.size() > 2) ? ttlText.mid(0, 2) : ttlText,
|
||||
style::al_center);
|
||||
|
||||
constexpr auto kPenWidth = 1.5;
|
||||
|
||||
const auto penWidth = style::ConvertScaleExact(kPenWidth);
|
||||
auto pen = QPen(st::premiumButtonFg);
|
||||
pen.setJoinStyle(Qt::RoundJoin);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
pen.setWidthF(penWidth);
|
||||
|
||||
q.setPen(pen);
|
||||
q.setBrush(Qt::NoBrush);
|
||||
q.drawArc(innerRect, arc::kQuarterLength, arc::kHalfLength);
|
||||
|
||||
q.setClipRect(innerRect
|
||||
- QMargins(innerRect.width() / 2, -penWidth, -penWidth, -penWidth));
|
||||
pen.setStyle(Qt::DotLine);
|
||||
q.setPen(pen);
|
||||
q.drawEllipse(innerRect);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
QRect CornerBadgeTTLRect(int photoSize) {
|
||||
const auto &partSize = st::dialogsTTLBadgeSize;
|
||||
return QRect(
|
||||
photoSize - partSize + st::dialogsTTLBadgeSkip.x(),
|
||||
photoSize - partSize + st::dialogsTTLBadgeSkip.y(),
|
||||
partSize,
|
||||
partSize);
|
||||
}
|
||||
|
||||
QImage BlurredDarkenedPart(QImage image, QRect part) {
|
||||
auto blurred = Images::BlurLargeImage(
|
||||
std::move(image),
|
||||
kBlurRadius).copy(part);
|
||||
|
||||
constexpr auto kMinAcceptableContrast = 4.5;
|
||||
const auto averageColor = Ui::CountAverageColor(blurred);
|
||||
const auto contrast = Ui::CountContrast(
|
||||
averageColor,
|
||||
st::premiumButtonFg->c);
|
||||
if (contrast < kMinAcceptableContrast) {
|
||||
constexpr auto kDarkerBy = 0.2;
|
||||
auto painterPart = QPainter(&blurred);
|
||||
painterPart.setOpacity(kDarkerBy);
|
||||
painterPart.fillRect(QRect(QPoint(), part.size()), Qt::black);
|
||||
}
|
||||
|
||||
blurred.setDevicePixelRatio(image.devicePixelRatio());
|
||||
return blurred;
|
||||
}
|
||||
|
||||
Row::CornerLayersManager::CornerLayersManager() = default;
|
||||
|
||||
bool Row::CornerLayersManager::isSameLayer(Layer layer) const {
|
||||
return isFinished() && (_nextLayer == layer);
|
||||
}
|
||||
|
||||
void Row::CornerLayersManager::setLayer(
|
||||
Layer layer,
|
||||
Fn<void()> updateCallback) {
|
||||
if (_nextLayer == layer) {
|
||||
return;
|
||||
}
|
||||
_lastFrameShown = false;
|
||||
_prevLayer = _nextLayer;
|
||||
_nextLayer = layer;
|
||||
if (_animation.animating()) {
|
||||
_animation.change(
|
||||
1.,
|
||||
st::dialogsOnlineBadgeDuration * (1. - _animation.value(1.)));
|
||||
} else if (updateCallback) {
|
||||
_animation.start(
|
||||
std::move(updateCallback),
|
||||
0.,
|
||||
1.,
|
||||
st::dialogsOnlineBadgeDuration);
|
||||
}
|
||||
}
|
||||
|
||||
float64 Row::CornerLayersManager::progressForLayer(Layer layer) const {
|
||||
return (_nextLayer == layer)
|
||||
? progress()
|
||||
: (_prevLayer == layer)
|
||||
? (1. - progress())
|
||||
: 0.;
|
||||
}
|
||||
|
||||
float64 Row::CornerLayersManager::progress() const {
|
||||
return _animation.value(1.);
|
||||
}
|
||||
|
||||
bool Row::CornerLayersManager::isFinished() const {
|
||||
return (progress() == 1.) && _lastFrameShown;
|
||||
}
|
||||
|
||||
void Row::CornerLayersManager::markFrameShown() {
|
||||
if (progress() == 1.) {
|
||||
_lastFrameShown = true;
|
||||
}
|
||||
}
|
||||
|
||||
bool Row::CornerLayersManager::isDisplayedNone() const {
|
||||
return (progress() == 1.) && (_nextLayer == 0);
|
||||
}
|
||||
|
||||
BasicRow::BasicRow() = default;
|
||||
BasicRow::~BasicRow() = default;
|
||||
|
||||
void BasicRow::addRipple(
|
||||
QPoint origin,
|
||||
QSize size,
|
||||
Fn<void()> updateCallback) {
|
||||
if (!_ripple) {
|
||||
addRippleWithMask(
|
||||
origin,
|
||||
Ui::RippleAnimation::RectMask(size),
|
||||
std::move(updateCallback));
|
||||
} else {
|
||||
_ripple->add(origin);
|
||||
}
|
||||
}
|
||||
|
||||
void BasicRow::addRippleWithMask(
|
||||
QPoint origin,
|
||||
QImage mask,
|
||||
Fn<void()> updateCallback) {
|
||||
_ripple = std::make_unique<Ui::RippleAnimation>(
|
||||
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<Entry*> entry,
|
||||
PeerData *peer,
|
||||
Ui::VideoUserpic *videoUserpic,
|
||||
const Ui::PaintContext &context,
|
||||
bool hasUnreadBadgesAbove) const {
|
||||
PaintUserpic(p, entry, peer, videoUserpic, _userpic, context);
|
||||
}
|
||||
|
||||
Row::Row(Key key, int index, int top) : _id(key), _top(top), _index(index) {
|
||||
if (const auto history = key.history()) {
|
||||
updateCornerBadgeShown(history->peer);
|
||||
}
|
||||
}
|
||||
|
||||
Row::~Row() {
|
||||
clearTopicJumpRipple();
|
||||
}
|
||||
|
||||
const style::DialogRow &Row::ComputeSt(
|
||||
not_null<const Entry*> entry,
|
||||
FilterId filterId) {
|
||||
if (const auto history = entry->asHistory()) {
|
||||
const auto hasTags = entry->hasChatsFilterTags(filterId);
|
||||
const auto wideRow = history->isForum()
|
||||
|| history->amMonoforumAdmin();
|
||||
return wideRow
|
||||
? (hasTags ? st::taggedForumDialogRow : st::forumDialogRow)
|
||||
: hasTags
|
||||
? st::taggedDialogRow
|
||||
: st::defaultDialogRow;
|
||||
} else if (entry->asTopic()) {
|
||||
return st::forumTopicRow;
|
||||
}
|
||||
return st::defaultDialogRow;
|
||||
}
|
||||
|
||||
void Row::recountHeight(float64 narrowRatio, FilterId filterId) {
|
||||
const auto &st = ComputeSt(_id.entry(), filterId);
|
||||
_height = ((&st == &st::defaultDialogRow) || !_id.history())
|
||||
? st::defaultDialogRow.height
|
||||
: anim::interpolate(
|
||||
st.height,
|
||||
st::defaultDialogRow.height,
|
||||
narrowRatio);
|
||||
}
|
||||
|
||||
uint64 Row::sortKey(FilterId filterId) const {
|
||||
return _id.entry()->sortKeyInChatList(filterId);
|
||||
}
|
||||
|
||||
void Row::setCornerBadgeShown(
|
||||
CornerLayersManager::Layer nextLayer,
|
||||
Fn<void()> updateCallback) const {
|
||||
const auto cornerBadgeShown = (nextLayer ? 1 : 0);
|
||||
if (_cornerBadgeShown == cornerBadgeShown) {
|
||||
if (!cornerBadgeShown) {
|
||||
return;
|
||||
} else if (_cornerBadgeUserpic
|
||||
&& _cornerBadgeUserpic->layersManager.isSameLayer(nextLayer)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
const_cast<Row*>(this)->_cornerBadgeShown = cornerBadgeShown;
|
||||
ensureCornerBadgeUserpic();
|
||||
_cornerBadgeUserpic->layersManager.setLayer(
|
||||
nextLayer,
|
||||
std::move(updateCallback));
|
||||
if (!_cornerBadgeShown
|
||||
&& _cornerBadgeUserpic
|
||||
&& _cornerBadgeUserpic->layersManager.isDisplayedNone()) {
|
||||
_cornerBadgeUserpic = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
void Row::updateCornerBadgeShown(
|
||||
not_null<PeerData*> peer,
|
||||
Fn<void()> updateCallback,
|
||||
bool hasUnreadBadgesAbove) const {
|
||||
const auto user = peer->asUser();
|
||||
const auto now = user ? base::unixtime::now() : TimeId();
|
||||
const auto channel = user ? nullptr : peer->asChannel();
|
||||
const auto nextLayer = [&] {
|
||||
if (hasUnreadBadgesAbove) {
|
||||
return kNoneLayer;
|
||||
} else if (user && Data::IsUserOnline(user, now)) {
|
||||
return kTopLayer;
|
||||
} else if (channel
|
||||
&& (Data::ChannelHasActiveCall(channel)
|
||||
|| Data::ChannelHasSubscriptionUntilDate(channel))) {
|
||||
return kTopLayer;
|
||||
} else if (peer->messagesTTL()) {
|
||||
return kBottomLayer;
|
||||
}
|
||||
return kNoneLayer;
|
||||
}();
|
||||
setCornerBadgeShown(nextLayer, std::move(updateCallback));
|
||||
if ((nextLayer == kTopLayer) && user) {
|
||||
peer->owner().watchForOffline(user, now);
|
||||
}
|
||||
}
|
||||
|
||||
void Row::ensureCornerBadgeUserpic() const {
|
||||
if (_cornerBadgeUserpic) {
|
||||
return;
|
||||
}
|
||||
_cornerBadgeUserpic = std::make_unique<CornerBadgeUserpic>();
|
||||
}
|
||||
|
||||
void Row::PaintCornerBadgeFrame(
|
||||
not_null<CornerBadgeUserpic*> data,
|
||||
int framePadding,
|
||||
not_null<Entry*> entry,
|
||||
PeerData *peer,
|
||||
Ui::VideoUserpic *videoUserpic,
|
||||
Ui::PeerUserpicView &view,
|
||||
const Ui::PaintContext &context,
|
||||
bool subscribed) {
|
||||
data->frame.fill(Qt::transparent);
|
||||
|
||||
Painter q(&data->frame);
|
||||
q.translate(framePadding, framePadding);
|
||||
auto hq = std::optional<PainterHighQualityEnabler>();
|
||||
const auto photoSize = context.st->photoSize;
|
||||
const auto storiesCount = data->storiesCount;
|
||||
if (storiesCount) {
|
||||
hq.emplace(q);
|
||||
const auto line = st::dialogsStoriesFull.lineTwice / 2.;
|
||||
const auto skip = line * 3 / 2.;
|
||||
const auto scale = 1. - (2 * skip / photoSize);
|
||||
const auto center = photoSize / 2.;
|
||||
q.save();
|
||||
q.translate(center, center);
|
||||
q.scale(scale, scale);
|
||||
q.translate(-center, -center);
|
||||
}
|
||||
q.translate(-context.st->padding.left(), -context.st->padding.top());
|
||||
PaintUserpic(
|
||||
q,
|
||||
entry,
|
||||
peer,
|
||||
videoUserpic,
|
||||
view,
|
||||
context);
|
||||
q.translate(context.st->padding.left(), context.st->padding.top());
|
||||
if (storiesCount) {
|
||||
q.restore();
|
||||
|
||||
const auto outline = QRectF(0, 0, photoSize, photoSize);
|
||||
const auto storiesUnread = st::dialogsStoriesFull.lineTwice / 2.;
|
||||
const auto storiesLine = st::dialogsStoriesFull.lineReadTwice / 2.;
|
||||
auto segments = std::vector<Ui::OutlineSegment>();
|
||||
if (data->storiesHasVideoStream) {
|
||||
const auto storiesVideoStreamBrush = st::attentionButtonFg->b;
|
||||
segments.push_back({ storiesVideoStreamBrush, storiesUnread });
|
||||
} else {
|
||||
const auto storiesUnreadCount = data->storiesUnreadCount;
|
||||
const auto storiesUnreadBrush = [&] {
|
||||
if (context.active || !storiesUnreadCount) {
|
||||
return st::dialogsUnreadBgMutedActive->b;
|
||||
}
|
||||
auto gradient = Ui::UnreadStoryOutlineGradient(outline);
|
||||
return QBrush(gradient);
|
||||
}();
|
||||
const auto storiesBrush = context.active
|
||||
? st::dialogsUnreadBgMutedActive->b
|
||||
: st::dialogsUnreadBgMuted->b;
|
||||
segments.reserve(storiesCount);
|
||||
const auto storiesReadCount = storiesCount - storiesUnreadCount;
|
||||
for (auto i = 0; i != storiesReadCount; ++i) {
|
||||
segments.push_back({ storiesBrush, storiesLine });
|
||||
}
|
||||
for (auto i = 0; i != storiesUnreadCount; ++i) {
|
||||
segments.push_back({ storiesUnreadBrush, storiesUnread });
|
||||
}
|
||||
}
|
||||
if (peer && (peer->forum() || peer->monoforum())) {
|
||||
const auto radius = context.st->photoSize
|
||||
* Ui::ForumUserpicRadiusMultiplier();
|
||||
Ui::PaintOutlineSegments(q, outline, radius, segments);
|
||||
} else {
|
||||
Ui::PaintOutlineSegments(q, outline, segments);
|
||||
}
|
||||
|
||||
if (data->storiesHasVideoStream) {
|
||||
Ui::PaintLiveBadge(q, 0, 0, photoSize);
|
||||
}
|
||||
}
|
||||
|
||||
if (subscribed) {
|
||||
if (!hq) {
|
||||
hq.emplace(q);
|
||||
}
|
||||
// TODO: Unnecessarily repaints on activating peer.
|
||||
q.setCompositionMode(QPainter::CompositionMode_Source);
|
||||
const auto &s = st::dialogsSubscriptionBadgeSkip;
|
||||
auto path = SubscriptionOutlinePath();
|
||||
const auto x = photoSize - s.x() - st::dialogsSubscriptionBadgeSize;
|
||||
const auto y = photoSize - s.y() - st::dialogsSubscriptionBadgeSize;
|
||||
q.translate(x, y);
|
||||
q.fillPath(path, Qt::transparent);
|
||||
q.setCompositionMode(QPainter::CompositionMode_SourceOver);
|
||||
q.resetTransform();
|
||||
q.drawImage(x, y, SubscriptionIcon());
|
||||
return;
|
||||
}
|
||||
|
||||
const auto &manager = data->layersManager;
|
||||
if (const auto p = manager.progressForLayer(kBottomLayer); p > 0.) {
|
||||
const auto size = photoSize;
|
||||
if (data->cacheTTL.isNull() && peer && peer->messagesTTL()) {
|
||||
data->cacheTTL = CornerBadgeTTL(peer, view, size);
|
||||
}
|
||||
q.setOpacity(p);
|
||||
const auto point = CornerBadgeTTLRect(size).topLeft();
|
||||
q.drawImage(point, data->cacheTTL);
|
||||
q.setOpacity(1.);
|
||||
}
|
||||
const auto topLayerProgress = manager.progressForLayer(kTopLayer);
|
||||
if (!topLayerProgress) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!hq) {
|
||||
hq.emplace(q);
|
||||
}
|
||||
q.setCompositionMode(QPainter::CompositionMode_Source);
|
||||
|
||||
const auto online = peer && peer->isUser();
|
||||
const auto size = online
|
||||
? st::dialogsOnlineBadgeSize
|
||||
: st::dialogsCallBadgeSize;
|
||||
const auto stroke = st::dialogsOnlineBadgeStroke;
|
||||
const auto skip = online
|
||||
? st::dialogsOnlineBadgeSkip
|
||||
: st::dialogsCallBadgeSkip;
|
||||
const auto shrink = (size / 2) * (1. - topLayerProgress);
|
||||
|
||||
auto pen = QPen(Qt::transparent);
|
||||
pen.setWidthF(stroke * topLayerProgress);
|
||||
q.setPen(pen);
|
||||
q.setBrush(data->active
|
||||
? st::dialogsOnlineBadgeFgActive
|
||||
: st::dialogsOnlineBadgeFg);
|
||||
q.drawEllipse(QRectF(
|
||||
photoSize - skip.x() - size,
|
||||
photoSize - skip.y() - size,
|
||||
size,
|
||||
size
|
||||
).marginsRemoved({ shrink, shrink, shrink, shrink }));
|
||||
}
|
||||
|
||||
void Row::paintUserpic(
|
||||
Painter &p,
|
||||
not_null<Entry*> entry,
|
||||
PeerData *peer,
|
||||
Ui::VideoUserpic *videoUserpic,
|
||||
const Ui::PaintContext &context,
|
||||
bool hasUnreadBadgesAbove) const {
|
||||
if (peer) {
|
||||
updateCornerBadgeShown(peer, nullptr, hasUnreadBadgesAbove);
|
||||
}
|
||||
|
||||
const auto cornerBadgeShown = !_cornerBadgeUserpic
|
||||
? _cornerBadgeShown
|
||||
: !_cornerBadgeUserpic->layersManager.isDisplayedNone();
|
||||
const auto storiesPeer = peer
|
||||
? ((peer->isUser() || peer->isChannel()) ? peer : nullptr)
|
||||
: nullptr;
|
||||
const auto storiesFolder = peer ? nullptr : _id.folder();
|
||||
const auto storiesHas = storiesPeer
|
||||
? storiesPeer->hasActiveStories()
|
||||
: storiesFolder
|
||||
? (storiesFolder->storiesCount() > 0)
|
||||
: false;
|
||||
if (!cornerBadgeShown && !storiesHas) {
|
||||
BasicRow::paintUserpic(p, entry, peer, videoUserpic, context, false);
|
||||
if (!peer || !_cornerBadgeShown) {
|
||||
_cornerBadgeUserpic = nullptr;
|
||||
}
|
||||
return;
|
||||
}
|
||||
ensureCornerBadgeUserpic();
|
||||
const auto ratio = style::DevicePixelRatio();
|
||||
const auto framePadding = std::max({
|
||||
-st::dialogsCallBadgeSkip.x(),
|
||||
-st::dialogsCallBadgeSkip.y(),
|
||||
st::lineWidth * 2 });
|
||||
const auto frameSide = (2 * framePadding + context.st->photoSize)
|
||||
* ratio;
|
||||
const auto frameSize = QSize(frameSide, frameSide);
|
||||
const auto storiesSource = (storiesHas && storiesPeer)
|
||||
? storiesPeer->owner().stories().source(storiesPeer->id)
|
||||
: nullptr;
|
||||
const auto storiesCountReal = storiesSource
|
||||
? int(storiesSource->ids.size())
|
||||
: storiesFolder
|
||||
? storiesFolder->storiesCount()
|
||||
: storiesHas
|
||||
? 1
|
||||
: 0;
|
||||
const auto storiesUnreadCountReal = storiesSource
|
||||
? storiesSource->unreadCount()
|
||||
: storiesFolder
|
||||
? storiesFolder->storiesUnreadCount()
|
||||
: (storiesPeer && storiesPeer->hasUnreadStories())
|
||||
? 1
|
||||
: 0;
|
||||
const auto storiesHasVideoStream = storiesSource
|
||||
? storiesSource->hasVideoStream
|
||||
: (storiesPeer && storiesPeer->hasActiveVideoStream())
|
||||
? 1
|
||||
: 0;
|
||||
const auto limit = Ui::kOutlineSegmentsMax;
|
||||
const auto storiesCount = std::min(storiesCountReal, limit);
|
||||
const auto storiesUnreadCount = std::min(storiesUnreadCountReal, limit);
|
||||
if (_cornerBadgeUserpic->frame.size() != frameSize) {
|
||||
_cornerBadgeUserpic->frame = QImage(
|
||||
frameSize,
|
||||
QImage::Format_ARGB32_Premultiplied);
|
||||
_cornerBadgeUserpic->frame.setDevicePixelRatio(ratio);
|
||||
}
|
||||
auto key = peer ? peer->userpicUniqueKey(userpicView()) : InMemoryKey();
|
||||
key.first += peer ? peer->messagesTTL() : 0;
|
||||
const auto frameIndex = videoUserpic ? videoUserpic->frameIndex() : -1;
|
||||
const auto paletteVersionReal = style::PaletteVersion();
|
||||
const auto paletteVersion = (paletteVersionReal & ((1 << 17) - 1));
|
||||
const auto active = context.active ? 1 : 0;
|
||||
const auto keyChanged = (_cornerBadgeUserpic->key != key)
|
||||
|| (_cornerBadgeUserpic->paletteVersion != paletteVersion);
|
||||
if (keyChanged) {
|
||||
_cornerBadgeUserpic->cacheTTL = QImage();
|
||||
}
|
||||
const auto subscribed = Data::ChannelHasSubscriptionUntilDate(
|
||||
peer ? peer->asChannel() : nullptr);
|
||||
if (keyChanged
|
||||
|| !_cornerBadgeUserpic->layersManager.isFinished()
|
||||
|| _cornerBadgeUserpic->active != active
|
||||
|| _cornerBadgeUserpic->frameIndex != frameIndex
|
||||
|| _cornerBadgeUserpic->storiesCount != storiesCount
|
||||
|| _cornerBadgeUserpic->storiesUnreadCount != storiesUnreadCount
|
||||
|| _cornerBadgeUserpic->storiesHasVideoStream != storiesHasVideoStream
|
||||
|| videoUserpic) {
|
||||
_cornerBadgeUserpic->key = key;
|
||||
_cornerBadgeUserpic->paletteVersion = paletteVersion;
|
||||
_cornerBadgeUserpic->active = active;
|
||||
_cornerBadgeUserpic->storiesCount = storiesCount;
|
||||
_cornerBadgeUserpic->storiesUnreadCount = storiesUnreadCount;
|
||||
_cornerBadgeUserpic->storiesHasVideoStream = storiesHasVideoStream;
|
||||
_cornerBadgeUserpic->frameIndex = frameIndex;
|
||||
_cornerBadgeUserpic->layersManager.markFrameShown();
|
||||
PaintCornerBadgeFrame(
|
||||
_cornerBadgeUserpic.get(),
|
||||
framePadding,
|
||||
_id.entry(),
|
||||
peer,
|
||||
videoUserpic,
|
||||
userpicView(),
|
||||
context,
|
||||
subscribed);
|
||||
}
|
||||
p.drawImage(
|
||||
context.st->padding.left() - framePadding,
|
||||
context.st->padding.top() - framePadding,
|
||||
_cornerBadgeUserpic->frame);
|
||||
const auto history = _id.history();
|
||||
if (!history || history->peer->isUser() || subscribed) {
|
||||
return;
|
||||
}
|
||||
const auto actionPainter = history->sendActionPainter();
|
||||
const auto bg = context.active
|
||||
? st::dialogsBgActive
|
||||
: st::dialogsBg;
|
||||
const auto size = st::dialogsCallBadgeSize;
|
||||
const auto skip = st::dialogsCallBadgeSkip;
|
||||
p.setOpacity(
|
||||
_cornerBadgeUserpic->layersManager.progressForLayer(kTopLayer));
|
||||
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::clearRipple() {
|
||||
BasicRow::clearRipple();
|
||||
clearTopicJumpRipple();
|
||||
}
|
||||
|
||||
void Row::addTopicJumpRipple(
|
||||
QPoint origin,
|
||||
not_null<Ui::TopicJumpCache*> topicJumpCache,
|
||||
Fn<void()> 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) {
|
||||
return;
|
||||
}
|
||||
const auto history = this->history();
|
||||
const auto view = history ? &history->lastItemDialogsView() : nullptr;
|
||||
if (view) {
|
||||
view->clearRipple();
|
||||
}
|
||||
_topicJumpRipple = 0;
|
||||
}
|
||||
|
||||
bool Row::topicJumpRipple() const {
|
||||
return _topicJumpRipple != 0;
|
||||
}
|
||||
|
||||
FakeRow::FakeRow(
|
||||
Key searchInChat,
|
||||
not_null<HistoryItem*> item,
|
||||
Fn<void()> 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
|
||||
Reference in New Issue
Block a user