mirror of
https://github.com/AyuGram/AyuGramDesktop.git
synced 2025-06-05 06:33:57 +02:00
Show nice collectible tooltip on wearing.
This commit is contained in:
parent
0523ae705a
commit
07fd9b3074
6 changed files with 357 additions and 12 deletions
|
@ -1170,3 +1170,9 @@ infoSharedMediaScroll: ScrollArea(defaultScrollArea) {
|
||||||
width: 5px;
|
width: 5px;
|
||||||
deltax: 2px;
|
deltax: 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
infoGiftTooltip: ImportantTooltip(defaultImportantTooltip) {
|
||||||
|
margin: margins(4px, 4px, 4px, 4px);
|
||||||
|
padding: margins(8px, 2px, 8px, 3px);
|
||||||
|
}
|
||||||
|
infoGiftTooltipFont: font(11px semibold);
|
||||||
|
|
|
@ -338,6 +338,10 @@ rpl::producer<Wrap> Controller::wrapValue() const {
|
||||||
return _widget->wrapValue();
|
return _widget->wrapValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
not_null<Ui::RpWidget*> Controller::wrapWidget() const {
|
||||||
|
return _widget;
|
||||||
|
}
|
||||||
|
|
||||||
bool Controller::validateMementoPeer(
|
bool Controller::validateMementoPeer(
|
||||||
not_null<ContentMemento*> memento) const {
|
not_null<ContentMemento*> memento) const {
|
||||||
return memento->peer() == peer()
|
return memento->peer() == peer()
|
||||||
|
|
|
@ -304,11 +304,12 @@ public:
|
||||||
return _section;
|
return _section;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool validateMementoPeer(
|
[[nodiscard]] bool validateMementoPeer(
|
||||||
not_null<ContentMemento*> memento) const;
|
not_null<ContentMemento*> memento) const;
|
||||||
|
|
||||||
Wrap wrap() const;
|
[[nodiscard]] Wrap wrap() const;
|
||||||
rpl::producer<Wrap> wrapValue() const;
|
[[nodiscard]] rpl::producer<Wrap> wrapValue() const;
|
||||||
|
[[nodiscard]] not_null<Ui::RpWidget*> wrapWidget() const;
|
||||||
void setSection(not_null<ContentMemento*> memento);
|
void setSection(not_null<ContentMemento*> memento);
|
||||||
|
|
||||||
Ui::SearchFieldController *searchFieldController() const {
|
Ui::SearchFieldController *searchFieldController() const {
|
||||||
|
|
|
@ -2753,7 +2753,8 @@ Cover *AddCover(
|
||||||
: container->add(object_ptr<Cover>(
|
: container->add(object_ptr<Cover>(
|
||||||
container,
|
container,
|
||||||
controller->parentController(),
|
controller->parentController(),
|
||||||
peer));
|
peer,
|
||||||
|
[=] { return controller->wrapWidget(); }));
|
||||||
result->showSection(
|
result->showSection(
|
||||||
) | rpl::start_with_next([=](Section section) {
|
) | rpl::start_with_next([=](Section section) {
|
||||||
controller->showSection(topic
|
controller->showSection(topic
|
||||||
|
|
|
@ -8,9 +8,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "info/profile/info_profile_cover.h"
|
#include "info/profile/info_profile_cover.h"
|
||||||
|
|
||||||
#include "api/api_user_privacy.h"
|
#include "api/api_user_privacy.h"
|
||||||
|
#include "base/timer_rpl.h"
|
||||||
#include "data/data_peer_values.h"
|
#include "data/data_peer_values.h"
|
||||||
#include "data/data_channel.h"
|
#include "data/data_channel.h"
|
||||||
#include "data/data_chat.h"
|
#include "data/data_chat.h"
|
||||||
|
#include "data/data_emoji_statuses.h"
|
||||||
#include "data/data_peer.h"
|
#include "data/data_peer.h"
|
||||||
#include "data/data_user.h"
|
#include "data/data_user.h"
|
||||||
#include "data/data_document.h"
|
#include "data/data_document.h"
|
||||||
|
@ -33,8 +35,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "ui/widgets/labels.h"
|
#include "ui/widgets/labels.h"
|
||||||
#include "ui/widgets/popup_menu.h"
|
#include "ui/widgets/popup_menu.h"
|
||||||
#include "ui/text/text_utilities.h"
|
#include "ui/text/text_utilities.h"
|
||||||
|
#include "ui/ui_utility.h"
|
||||||
|
#include "ui/painter.h"
|
||||||
#include "base/event_filter.h"
|
#include "base/event_filter.h"
|
||||||
#include "base/unixtime.h"
|
#include "base/unixtime.h"
|
||||||
|
#include "window/window_controller.h"
|
||||||
#include "window/window_session_controller.h"
|
#include "window/window_session_controller.h"
|
||||||
#include "main/main_session.h"
|
#include "main/main_session.h"
|
||||||
#include "settings/settings_premium.h"
|
#include "settings/settings_premium.h"
|
||||||
|
@ -49,6 +54,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
namespace Info::Profile {
|
namespace Info::Profile {
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
|
constexpr auto kWaitBeforeGiftBadge = crl::time(1000);
|
||||||
|
constexpr auto kGiftBadgeGlares = 3;
|
||||||
|
constexpr auto kGlareDurationStep = crl::time(320);
|
||||||
|
constexpr auto kGlareTimeout = crl::time(1000);
|
||||||
|
|
||||||
auto MembersStatusText(int count) {
|
auto MembersStatusText(int count) {
|
||||||
return tr::lng_chat_status_members(tr::now, lt_count_decimal, count);
|
return tr::lng_chat_status_members(tr::now, lt_count_decimal, count);
|
||||||
};
|
};
|
||||||
|
@ -106,6 +116,240 @@ auto ChatStatusText(int fullCount, int onlineCount, bool isGroup) {
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
|
class Cover::BadgeTooltip final : public Ui::RpWidget {
|
||||||
|
public:
|
||||||
|
BadgeTooltip(
|
||||||
|
not_null<QWidget*> parent,
|
||||||
|
std::shared_ptr<Data::EmojiStatusCollectible> collectible,
|
||||||
|
not_null<QWidget*> pointTo);
|
||||||
|
|
||||||
|
void fade(bool shown);
|
||||||
|
void finishAnimating();
|
||||||
|
|
||||||
|
[[nodiscard]] crl::time glarePeriod() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
|
||||||
|
void setupGeometry(not_null<QWidget*> pointTo);
|
||||||
|
void prepareImage();
|
||||||
|
void showGlare();
|
||||||
|
|
||||||
|
const style::ImportantTooltip &_st;
|
||||||
|
std::shared_ptr<Data::EmojiStatusCollectible> _collectible;
|
||||||
|
QString _text;
|
||||||
|
const style::font &_font;
|
||||||
|
QSize _inner;
|
||||||
|
QSize _outer;
|
||||||
|
int _stroke = 0;
|
||||||
|
int _skip = 0;
|
||||||
|
QSize _full;
|
||||||
|
int _glareSize = 0;
|
||||||
|
int _glareRange = 0;
|
||||||
|
crl::time _glareDuration = 0;
|
||||||
|
base::Timer _glareTimer;
|
||||||
|
|
||||||
|
Ui::Animations::Simple _showAnimation;
|
||||||
|
Ui::Animations::Simple _glareAnimation;
|
||||||
|
|
||||||
|
QImage _image;
|
||||||
|
int _glareRight = 0;
|
||||||
|
int _imageGlareRight = 0;
|
||||||
|
int _arrowMiddle = 0;
|
||||||
|
int _imageArrowMiddle = 0;
|
||||||
|
|
||||||
|
bool _shown = false;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
Cover::BadgeTooltip::BadgeTooltip(
|
||||||
|
not_null<QWidget*> parent,
|
||||||
|
std::shared_ptr<Data::EmojiStatusCollectible> collectible,
|
||||||
|
not_null<QWidget*> pointTo)
|
||||||
|
: Ui::RpWidget(parent)
|
||||||
|
, _st(st::infoGiftTooltip)
|
||||||
|
, _collectible(std::move(collectible))
|
||||||
|
, _text(_collectible->title)
|
||||||
|
, _font(st::infoGiftTooltipFont)
|
||||||
|
, _inner(_font->width(_text), _font->height)
|
||||||
|
, _outer(_inner.grownBy(_st.padding))
|
||||||
|
, _stroke(st::lineWidth)
|
||||||
|
, _skip(2 * _stroke)
|
||||||
|
, _full(_outer + QSize(2 * _skip, _st.arrow + 2 * _skip))
|
||||||
|
, _glareSize(_outer.height() * 3)
|
||||||
|
, _glareRange(_outer.width() + _glareSize)
|
||||||
|
, _glareDuration(_glareRange * kGlareDurationStep / _glareSize)
|
||||||
|
, _glareTimer([=] { showGlare(); }) {
|
||||||
|
resize(_full + QSize(0, _st.shift));
|
||||||
|
setupGeometry(pointTo);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cover::BadgeTooltip::fade(bool shown) {
|
||||||
|
if (_shown == shown) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
show();
|
||||||
|
_shown = shown;
|
||||||
|
_showAnimation.start([=] {
|
||||||
|
update();
|
||||||
|
if (!_showAnimation.animating()) {
|
||||||
|
if (!_shown) {
|
||||||
|
hide();
|
||||||
|
} else {
|
||||||
|
showGlare();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, _shown ? 0. : 1., _shown ? 1. : 0., _st.duration, anim::easeInCirc);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cover::BadgeTooltip::showGlare() {
|
||||||
|
_glareAnimation.start([=] {
|
||||||
|
update();
|
||||||
|
if (!_glareAnimation.animating()) {
|
||||||
|
_glareTimer.callOnce(kGlareTimeout);
|
||||||
|
}
|
||||||
|
}, 0., 1., _glareDuration);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cover::BadgeTooltip::finishAnimating() {
|
||||||
|
_showAnimation.stop();
|
||||||
|
if (!_shown) {
|
||||||
|
hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
crl::time Cover::BadgeTooltip::glarePeriod() const {
|
||||||
|
return _glareDuration + kGlareTimeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cover::BadgeTooltip::paintEvent(QPaintEvent *e) {
|
||||||
|
const auto glare = _glareAnimation.value(0.);
|
||||||
|
_glareRight = anim::interpolate(0, _glareRange, glare);
|
||||||
|
prepareImage();
|
||||||
|
|
||||||
|
auto p = QPainter(this);
|
||||||
|
const auto shown = _showAnimation.value(_shown ? 1. : 0.);
|
||||||
|
p.setOpacity(shown);
|
||||||
|
const auto imageHeight = _image.height() / _image.devicePixelRatio();
|
||||||
|
const auto top = anim::interpolate(0, height() - imageHeight, shown);
|
||||||
|
p.drawImage(0, top, _image);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cover::BadgeTooltip::setupGeometry(not_null<QWidget*> pointTo) {
|
||||||
|
auto widget = pointTo.get();
|
||||||
|
const auto parent = parentWidget();
|
||||||
|
|
||||||
|
const auto refresh = [=] {
|
||||||
|
const auto rect = Ui::MapFrom(parent, pointTo, pointTo->rect());
|
||||||
|
const auto point = QPoint(rect.center().x(), rect.y());
|
||||||
|
const auto left = point.x() - (width() / 2);
|
||||||
|
const auto skip = _st.padding.left();
|
||||||
|
setGeometry(
|
||||||
|
std::min(std::max(left, skip), parent->width() - width() - skip),
|
||||||
|
std::max(point.y() - height() - _st.margin.bottom(), skip),
|
||||||
|
width(),
|
||||||
|
height());
|
||||||
|
const auto arrowMiddle = point.x() - x();
|
||||||
|
if (_arrowMiddle != arrowMiddle) {
|
||||||
|
_arrowMiddle = arrowMiddle;
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
refresh();
|
||||||
|
while (widget && widget != parent) {
|
||||||
|
base::install_event_filter(this, widget, [=](not_null<QEvent*> e) {
|
||||||
|
if (e->type() == QEvent::Resize || e->type() == QEvent::Move || e->type() == QEvent::ZOrderChange) {
|
||||||
|
refresh();
|
||||||
|
raise();
|
||||||
|
}
|
||||||
|
return base::EventFilterResult::Continue;
|
||||||
|
});
|
||||||
|
widget = widget->parentWidget();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cover::BadgeTooltip::prepareImage() {
|
||||||
|
const auto ratio = style::DevicePixelRatio();
|
||||||
|
const auto arrow = _st.arrow;
|
||||||
|
const auto size = _full * ratio;
|
||||||
|
if (_image.size() != size) {
|
||||||
|
_image = QImage(size, QImage::Format_ARGB32_Premultiplied);
|
||||||
|
_image.setDevicePixelRatio(ratio);
|
||||||
|
} else if (_imageGlareRight == _glareRight
|
||||||
|
&& _imageArrowMiddle == _arrowMiddle) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_imageGlareRight = _glareRight;
|
||||||
|
_imageArrowMiddle = _arrowMiddle;
|
||||||
|
_image.fill(Qt::transparent);
|
||||||
|
|
||||||
|
const auto gfrom = _imageGlareRight - _glareSize;
|
||||||
|
const auto gtill = _imageGlareRight;
|
||||||
|
|
||||||
|
auto path = QPainterPath();
|
||||||
|
const auto width = _outer.width();
|
||||||
|
const auto height = _outer.height();
|
||||||
|
const auto radius = (height + 1) / 2;
|
||||||
|
const auto diameter = height;
|
||||||
|
path.moveTo(radius, 0);
|
||||||
|
path.lineTo(width - radius, 0);
|
||||||
|
path.arcTo(
|
||||||
|
QRect(QPoint(width - diameter, 0), QSize(diameter, diameter)),
|
||||||
|
90,
|
||||||
|
-180);
|
||||||
|
const auto xarrow = _arrowMiddle - _skip;
|
||||||
|
if (xarrow - arrow <= radius || xarrow + arrow >= width - radius) {
|
||||||
|
path.lineTo(radius, height);
|
||||||
|
} else {
|
||||||
|
path.lineTo(xarrow + arrow, height);
|
||||||
|
path.lineTo(xarrow, height + arrow);
|
||||||
|
path.lineTo(xarrow - arrow, height);
|
||||||
|
path.lineTo(radius, height);
|
||||||
|
}
|
||||||
|
path.arcTo(
|
||||||
|
QRect(QPoint(0, 0), QSize(diameter, diameter)),
|
||||||
|
-90,
|
||||||
|
-180);
|
||||||
|
path.closeSubpath();
|
||||||
|
|
||||||
|
auto p = QPainter(&_image);
|
||||||
|
auto hq = PainterHighQualityEnabler(p);
|
||||||
|
p.setPen(Qt::NoPen);
|
||||||
|
if (gtill > 0) {
|
||||||
|
auto gradient = QLinearGradient(gfrom, 0, gtill, 0);
|
||||||
|
gradient.setStops({
|
||||||
|
{ 0., _collectible->edgeColor },
|
||||||
|
{ 0.5, _collectible->centerColor },
|
||||||
|
{ 1., _collectible->edgeColor },
|
||||||
|
});
|
||||||
|
p.setBrush(gradient);
|
||||||
|
} else {
|
||||||
|
p.setBrush(_collectible->edgeColor);
|
||||||
|
}
|
||||||
|
p.translate(_skip, _skip);
|
||||||
|
p.drawPath(path);
|
||||||
|
p.setCompositionMode(QPainter::CompositionMode_Source);
|
||||||
|
p.setBrush(Qt::NoBrush);
|
||||||
|
auto copy = _collectible->textColor;
|
||||||
|
copy.setAlpha(0);
|
||||||
|
if (gtill > 0) {
|
||||||
|
auto gradient = QLinearGradient(gfrom, 0, gtill, 0);
|
||||||
|
gradient.setStops({
|
||||||
|
{ 0., copy },
|
||||||
|
{ 0.5, _collectible->textColor },
|
||||||
|
{ 1., copy },
|
||||||
|
});
|
||||||
|
p.setPen(QPen(gradient, _stroke));
|
||||||
|
} else {
|
||||||
|
p.setPen(QPen(copy, _stroke));
|
||||||
|
}
|
||||||
|
p.drawPath(path);
|
||||||
|
p.setCompositionMode(QPainter::CompositionMode_SourceOver);
|
||||||
|
p.setFont(_font);
|
||||||
|
p.setPen(QColor(255, 255, 255));
|
||||||
|
p.drawText(_st.padding.left(), _st.padding.top() + _font->ascent, _text);
|
||||||
|
}
|
||||||
|
|
||||||
TopicIconView::TopicIconView(
|
TopicIconView::TopicIconView(
|
||||||
not_null<Data::ForumTopic*> topic,
|
not_null<Data::ForumTopic*> topic,
|
||||||
Fn<bool()> paused,
|
Fn<bool()> paused,
|
||||||
|
@ -271,8 +515,16 @@ TopicIconButton::TopicIconButton(
|
||||||
Cover::Cover(
|
Cover::Cover(
|
||||||
QWidget *parent,
|
QWidget *parent,
|
||||||
not_null<Window::SessionController*> controller,
|
not_null<Window::SessionController*> controller,
|
||||||
not_null<PeerData*> peer)
|
not_null<PeerData*> peer,
|
||||||
: Cover(parent, controller, peer, Role::Info, NameValue(peer)) {
|
Fn<not_null<QWidget*>()> parentForTooltip)
|
||||||
|
: Cover(
|
||||||
|
parent,
|
||||||
|
controller,
|
||||||
|
peer,
|
||||||
|
nullptr,
|
||||||
|
Role::Info,
|
||||||
|
NameValue(peer),
|
||||||
|
parentForTooltip) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Cover::Cover(
|
Cover::Cover(
|
||||||
|
@ -285,7 +537,8 @@ Cover::Cover(
|
||||||
topic->channel(),
|
topic->channel(),
|
||||||
topic,
|
topic,
|
||||||
Role::Info,
|
Role::Info,
|
||||||
TitleValue(topic)) {
|
TitleValue(topic),
|
||||||
|
nullptr) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Cover::Cover(
|
Cover::Cover(
|
||||||
|
@ -300,7 +553,8 @@ Cover::Cover(
|
||||||
peer,
|
peer,
|
||||||
nullptr,
|
nullptr,
|
||||||
role,
|
role,
|
||||||
std::move(title)) {
|
std::move(title),
|
||||||
|
nullptr) {
|
||||||
}
|
}
|
||||||
|
|
||||||
[[nodiscard]] rpl::producer<Badge::Content> BotVerifyBadgeForPeer(
|
[[nodiscard]] rpl::producer<Badge::Content> BotVerifyBadgeForPeer(
|
||||||
|
@ -323,7 +577,8 @@ Cover::Cover(
|
||||||
not_null<PeerData*> peer,
|
not_null<PeerData*> peer,
|
||||||
Data::ForumTopic *topic,
|
Data::ForumTopic *topic,
|
||||||
Role role,
|
Role role,
|
||||||
rpl::producer<QString> title)
|
rpl::producer<QString> title,
|
||||||
|
Fn<not_null<QWidget*>()> parentForTooltip)
|
||||||
: FixedHeightWidget(parent, CoverStyle(peer, topic, role).height)
|
: FixedHeightWidget(parent, CoverStyle(peer, topic, role).height)
|
||||||
, _st(CoverStyle(peer, topic, role))
|
, _st(CoverStyle(peer, topic, role))
|
||||||
, _role(role)
|
, _role(role)
|
||||||
|
@ -343,12 +598,13 @@ Cover::Cover(
|
||||||
return controller->isGifPausedAtLeastFor(
|
return controller->isGifPausedAtLeastFor(
|
||||||
Window::GifPauseReason::Layer);
|
Window::GifPauseReason::Layer);
|
||||||
}))
|
}))
|
||||||
|
, _badgeContent(BadgeContentForPeer(peer))
|
||||||
, _badge(
|
, _badge(
|
||||||
std::make_unique<Badge>(
|
std::make_unique<Badge>(
|
||||||
this,
|
this,
|
||||||
st::infoPeerBadge,
|
st::infoPeerBadge,
|
||||||
&peer->session(),
|
&peer->session(),
|
||||||
BadgeContentForPeer(peer),
|
_badgeContent.value(),
|
||||||
_emojiStatusPanel.get(),
|
_emojiStatusPanel.get(),
|
||||||
[=] {
|
[=] {
|
||||||
return controller->isGifPausedAtLeastFor(
|
return controller->isGifPausedAtLeastFor(
|
||||||
|
@ -365,6 +621,8 @@ Cover::Cover(
|
||||||
return controller->isGifPausedAtLeastFor(
|
return controller->isGifPausedAtLeastFor(
|
||||||
Window::GifPauseReason::Layer);
|
Window::GifPauseReason::Layer);
|
||||||
}))
|
}))
|
||||||
|
, _parentForTooltip(std::move(parentForTooltip))
|
||||||
|
, _badgeTooltipHide([=] { hideBadgeTooltip(); })
|
||||||
, _userpic(topic
|
, _userpic(topic
|
||||||
? nullptr
|
? nullptr
|
||||||
: object_ptr<Ui::UserpicButton>(
|
: object_ptr<Ui::UserpicButton>(
|
||||||
|
@ -416,6 +674,7 @@ Cover::Cover(
|
||||||
|
|
||||||
initViewers(std::move(title));
|
initViewers(std::move(title));
|
||||||
setupChildGeometry();
|
setupChildGeometry();
|
||||||
|
setupUniqueBadgeTooltip();
|
||||||
|
|
||||||
if (_userpic) {
|
if (_userpic) {
|
||||||
} else if (topic->canEdit()) {
|
} else if (topic->canEdit()) {
|
||||||
|
@ -743,6 +1002,8 @@ void Cover::refreshStatusText() {
|
||||||
}
|
}
|
||||||
|
|
||||||
Cover::~Cover() {
|
Cover::~Cover() {
|
||||||
|
base::take(_badgeTooltip);
|
||||||
|
base::take(_badgeOldTooltips);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Cover::refreshNameGeometry(int newWidth) {
|
void Cover::refreshNameGeometry(int newWidth) {
|
||||||
|
@ -791,4 +1052,62 @@ void Cover::refreshStatusGeometry(int newWidth) {
|
||||||
newWidth);
|
newWidth);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Cover::hideBadgeTooltip() {
|
||||||
|
_badgeTooltipHide.cancel();
|
||||||
|
if (auto old = base::take(_badgeTooltip)) {
|
||||||
|
const auto raw = old.get();
|
||||||
|
_badgeOldTooltips.push_back(std::move(old));
|
||||||
|
|
||||||
|
raw->fade(false);
|
||||||
|
raw->shownValue(
|
||||||
|
) | rpl::filter(
|
||||||
|
!rpl::mappers::_1
|
||||||
|
) | rpl::start_with_next([=] {
|
||||||
|
const auto i = ranges::find(
|
||||||
|
_badgeOldTooltips,
|
||||||
|
raw,
|
||||||
|
&std::unique_ptr<BadgeTooltip>::get);
|
||||||
|
if (i != end(_badgeOldTooltips)) {
|
||||||
|
_badgeOldTooltips.erase(i);
|
||||||
|
}
|
||||||
|
}, raw->lifetime());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cover::setupUniqueBadgeTooltip() {
|
||||||
|
base::timer_once(kWaitBeforeGiftBadge) | rpl::then(
|
||||||
|
_badge->updated()
|
||||||
|
) | rpl::start_with_next([=] {
|
||||||
|
const auto widget = _badge->widget();
|
||||||
|
const auto &content = _badgeContent.current();
|
||||||
|
const auto &collectible = content.emojiStatusId.collectible;
|
||||||
|
const auto premium = (content.badge == BadgeType::Premium);
|
||||||
|
const auto id = (collectible && widget && premium)
|
||||||
|
? collectible->id
|
||||||
|
: uint64();
|
||||||
|
if (_badgeCollectibleId == id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
hideBadgeTooltip();
|
||||||
|
if (!collectible) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto parent = _parentForTooltip
|
||||||
|
? _parentForTooltip()
|
||||||
|
: _controller->window().widget()->bodyWidget();
|
||||||
|
_badgeTooltip = std::make_unique<BadgeTooltip>(
|
||||||
|
parent,
|
||||||
|
collectible,
|
||||||
|
widget);
|
||||||
|
const auto raw = _badgeTooltip.get();
|
||||||
|
raw->fade(true);
|
||||||
|
_badgeTooltipHide.callOnce(kGiftBadgeGlares * raw->glarePeriod()
|
||||||
|
- st::infoGiftTooltip.duration * 1.5);
|
||||||
|
}, lifetime());
|
||||||
|
|
||||||
|
if (const auto raw = _badgeTooltip.get()) {
|
||||||
|
raw->finishAnimating();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Info::Profile
|
} // namespace Info::Profile
|
||||||
|
|
|
@ -7,6 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include "info/profile/info_profile_badge.h"
|
||||||
#include "ui/wrap/padding_wrap.h"
|
#include "ui/wrap/padding_wrap.h"
|
||||||
#include "ui/abstract_button.h"
|
#include "ui/abstract_button.h"
|
||||||
#include "base/timer.h"
|
#include "base/timer.h"
|
||||||
|
@ -103,7 +104,8 @@ public:
|
||||||
Cover(
|
Cover(
|
||||||
QWidget *parent,
|
QWidget *parent,
|
||||||
not_null<Window::SessionController*> controller,
|
not_null<Window::SessionController*> controller,
|
||||||
not_null<PeerData*> peer);
|
not_null<PeerData*> peer,
|
||||||
|
Fn<not_null<QWidget*>()> parentForTooltip = nullptr);
|
||||||
Cover(
|
Cover(
|
||||||
QWidget *parent,
|
QWidget *parent,
|
||||||
not_null<Window::SessionController*> controller,
|
not_null<Window::SessionController*> controller,
|
||||||
|
@ -124,13 +126,16 @@ public:
|
||||||
[[nodiscard]] std::optional<QImage> updatedPersonalPhoto() const;
|
[[nodiscard]] std::optional<QImage> updatedPersonalPhoto() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
class BadgeTooltip;
|
||||||
|
|
||||||
Cover(
|
Cover(
|
||||||
QWidget *parent,
|
QWidget *parent,
|
||||||
not_null<Window::SessionController*> controller,
|
not_null<Window::SessionController*> controller,
|
||||||
not_null<PeerData*> peer,
|
not_null<PeerData*> peer,
|
||||||
Data::ForumTopic *topic,
|
Data::ForumTopic *topic,
|
||||||
Role role,
|
Role role,
|
||||||
rpl::producer<QString> title);
|
rpl::producer<QString> title,
|
||||||
|
Fn<not_null<QWidget*>()> parentForTooltip);
|
||||||
|
|
||||||
void setupShowLastSeen();
|
void setupShowLastSeen();
|
||||||
void setupChildGeometry();
|
void setupChildGeometry();
|
||||||
|
@ -139,7 +144,9 @@ private:
|
||||||
void refreshNameGeometry(int newWidth);
|
void refreshNameGeometry(int newWidth);
|
||||||
void refreshStatusGeometry(int newWidth);
|
void refreshStatusGeometry(int newWidth);
|
||||||
void refreshUploadPhotoOverlay();
|
void refreshUploadPhotoOverlay();
|
||||||
|
void setupUniqueBadgeTooltip();
|
||||||
void setupChangePersonal();
|
void setupChangePersonal();
|
||||||
|
void hideBadgeTooltip();
|
||||||
|
|
||||||
const style::InfoProfileCover &_st;
|
const style::InfoProfileCover &_st;
|
||||||
|
|
||||||
|
@ -148,10 +155,17 @@ private:
|
||||||
const not_null<PeerData*> _peer;
|
const not_null<PeerData*> _peer;
|
||||||
const std::unique_ptr<EmojiStatusPanel> _emojiStatusPanel;
|
const std::unique_ptr<EmojiStatusPanel> _emojiStatusPanel;
|
||||||
const std::unique_ptr<Badge> _botVerify;
|
const std::unique_ptr<Badge> _botVerify;
|
||||||
|
rpl::variable<Badge::Content> _badgeContent;
|
||||||
const std::unique_ptr<Badge> _badge;
|
const std::unique_ptr<Badge> _badge;
|
||||||
const std::unique_ptr<Badge> _verified;
|
const std::unique_ptr<Badge> _verified;
|
||||||
rpl::variable<int> _onlineCount;
|
rpl::variable<int> _onlineCount;
|
||||||
|
|
||||||
|
const Fn<not_null<QWidget*>()> _parentForTooltip;
|
||||||
|
std::unique_ptr<BadgeTooltip> _badgeTooltip;
|
||||||
|
std::vector<std::unique_ptr<BadgeTooltip>> _badgeOldTooltips;
|
||||||
|
base::Timer _badgeTooltipHide;
|
||||||
|
uint64 _badgeCollectibleId = 0;
|
||||||
|
|
||||||
const object_ptr<Ui::UserpicButton> _userpic;
|
const object_ptr<Ui::UserpicButton> _userpic;
|
||||||
Ui::UserpicButton *_changePersonal = nullptr;
|
Ui::UserpicButton *_changePersonal = nullptr;
|
||||||
std::optional<QImage> _personalChosen;
|
std::optional<QImage> _personalChosen;
|
||||||
|
|
Loading…
Add table
Reference in a new issue