2
0
mirror of https://github.com/telegramdesktop/tdesktop synced 2025-08-23 10:47:11 +00:00
tdesktop/Telegram/SourceFiles/info/profile/info_profile_cover.cpp

355 lines
9.3 KiB
C++
Raw Normal View History

/*
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 "info/profile/info_profile_cover.h"
#include "data/data_photo.h"
#include "data/data_peer_values.h"
#include "data/data_channel.h"
#include "data/data_chat.h"
#include "data/data_changes.h"
#include "editor/photo_editor_layer_widget.h"
#include "info/profile/info_profile_values.h"
#include "info/info_controller.h"
#include "info/info_memento.h"
#include "lang/lang_keys.h"
#include "ui/widgets/labels.h"
#include "ui/widgets/buttons.h"
#include "ui/effects/ripple_animation.h"
#include "ui/text/text_utilities.h"
#include "ui/special_buttons.h"
2019-06-23 14:18:33 +02:00
#include "ui/unread_badge.h"
#include "base/unixtime.h"
#include "window/window_session_controller.h"
#include "core/application.h"
2019-07-24 13:45:24 +02:00
#include "main/main_session.h"
#include "settings/settings_premium.h"
#include "apiwrap.h"
#include "api/api_peer_photo.h"
2019-09-18 11:05:37 +03:00
#include "styles/style_boxes.h"
#include "styles/style_info.h"
namespace Info {
namespace Profile {
namespace {
auto MembersStatusText(int count) {
return tr::lng_chat_status_members(tr::now, lt_count_decimal, count);
};
auto OnlineStatusText(int count) {
return tr::lng_chat_status_online(tr::now, lt_count_decimal, count);
};
auto ChatStatusText(int fullCount, int onlineCount, bool isGroup) {
if (onlineCount > 1 && onlineCount <= fullCount) {
2019-06-19 17:09:03 +02:00
return tr::lng_chat_status_members_online(
tr::now,
lt_members_count,
MembersStatusText(fullCount),
lt_online_count,
OnlineStatusText(onlineCount));
} else if (fullCount > 0) {
return isGroup
? tr::lng_chat_status_members(
tr::now,
lt_count_decimal,
fullCount)
: tr::lng_chat_status_subscribers(
tr::now,
lt_count_decimal,
fullCount);
}
2019-06-19 17:09:03 +02:00
return isGroup
? tr::lng_group_status(tr::now)
: tr::lng_channel_status(tr::now);
};
} // namespace
Cover::Cover(
QWidget *parent,
2018-09-05 22:39:35 +03:00
not_null<PeerData*> peer,
not_null<Window::SessionController*> controller)
2019-06-10 17:47:22 +02:00
: Cover(parent, peer, controller, NameValue(
peer
) | rpl::map([=](const TextWithEntities &name) {
return name.text;
})) {
}
Cover::Cover(
QWidget *parent,
not_null<PeerData*> peer,
not_null<Window::SessionController*> controller,
rpl::producer<QString> title)
: FixedHeightWidget(
parent,
st::infoProfilePhotoTop
+ st::infoProfilePhoto.size.height()
+ st::infoProfilePhotoBottom)
, _controller(controller)
2018-09-05 22:39:35 +03:00
, _peer(peer)
, _userpic(
this,
2018-09-05 22:39:35 +03:00
controller,
_peer,
Ui::UserpicButton::Role::OpenPhoto,
st::infoProfilePhoto)
, _name(this, st::infoProfileNameLabel)
, _status(
this,
_peer->isMegagroup()
? st::infoProfileMegagroupStatusLabel
: st::infoProfileStatusLabel)
, _refreshStatusTimer([this] { refreshStatusText(); }) {
_peer->updateFull();
_name->setSelectable(true);
2019-06-19 17:09:03 +02:00
_name->setContextCopyText(tr::lng_profile_copy_fullname(tr::now));
if (!_peer->isMegagroup()) {
_status->setAttribute(Qt::WA_TransparentForMouseEvents);
}
2019-06-10 17:47:22 +02:00
initViewers(std::move(title));
setupChildGeometry();
_userpic->uploadPhotoRequests(
) | rpl::start_with_next([=] {
_peer->session().api().peerPhoto().upload(
_peer,
_userpic->takeResultImage());
}, _userpic->lifetime());
}
void Cover::setupChildGeometry() {
widthValue(
) | rpl::start_with_next([this](int newWidth) {
_userpic->moveToLeft(
st::infoProfilePhotoLeft,
st::infoProfilePhotoTop,
newWidth);
refreshNameGeometry(newWidth);
refreshStatusGeometry(newWidth);
}, lifetime());
}
Cover *Cover::setOnlineCount(rpl::producer<int> &&count) {
std::move(
count
) | rpl::start_with_next([this](int count) {
_onlineCount = count;
refreshStatusText();
}, lifetime());
return this;
}
2019-06-10 17:47:22 +02:00
void Cover::initViewers(rpl::producer<QString> title) {
using Flag = Data::PeerUpdate::Flag;
2019-06-10 17:47:22 +02:00
std::move(
title
) | rpl::start_with_next([=](const QString &title) {
_name->setText(title);
refreshNameGeometry(width());
}, lifetime());
_peer->session().changes().peerFlagsValue(
_peer,
Flag::OnlineStatus | Flag::Members
) | rpl::start_with_next(
[=] { refreshStatusText(); },
lifetime());
if (!_peer->isUser()) {
_peer->session().changes().peerFlagsValue(
_peer,
Flag::Rights
) | rpl::start_with_next(
[=] { refreshUploadPhotoOverlay(); },
lifetime());
2018-09-13 23:09:26 +03:00
} else if (_peer->isSelf()) {
refreshUploadPhotoOverlay();
}
2021-01-21 16:39:40 +04:00
BadgeValue(
_peer
2021-01-21 16:39:40 +04:00
) | rpl::start_with_next([=](Badge badge) {
if (badge == Badge::Premium
&& !_peer->session().premiumBadgesShown()) {
badge = Badge::None;
}
2021-01-21 16:39:40 +04:00
setBadge(badge);
2019-06-23 14:18:33 +02:00
}, lifetime());
}
void Cover::refreshUploadPhotoOverlay() {
_userpic->switchChangePhotoOverlay([&] {
2018-09-13 23:09:26 +03:00
if (const auto chat = _peer->asChat()) {
2019-01-03 18:39:19 +04:00
return chat->canEditInformation();
2018-09-13 23:09:26 +03:00
} else if (const auto channel = _peer->asChannel()) {
return channel->canEditInformation();
}
2018-09-13 23:09:26 +03:00
return _peer->isSelf();
}());
}
2021-01-21 16:39:40 +04:00
void Cover::setBadge(Badge badge) {
if (_badge == badge) {
2017-11-06 18:13:56 +04:00
return;
}
2021-01-21 16:39:40 +04:00
_badge = badge;
_verifiedCheck.destroy();
_scamFakeBadge.destroy();
switch (_badge) {
case Badge::Verified:
2022-05-20 19:28:50 +04:00
case Badge::Premium: {
const auto icon = (_badge == Badge::Verified)
? &st::infoVerifiedCheck
: &st::infoPremiumStar;
2017-11-06 18:13:56 +04:00
_verifiedCheck.create(this);
_verifiedCheck->show();
2022-05-20 19:28:50 +04:00
_verifiedCheck->resize(icon->size());
_verifiedCheck->paintRequest(
2022-05-20 19:28:50 +04:00
) | rpl::start_with_next([icon, check = _verifiedCheck.data()] {
Painter p(check);
2022-05-20 19:28:50 +04:00
icon->paint(p, 0, 0, check->width());
}, _verifiedCheck->lifetime());
if (_badge == Badge::Premium) {
const auto userId = peerToUser(_peer->id).bare;
_verifiedCheck->setClickedCallback([=] {
::Settings::ShowPremium(
_controller,
u"profile__%1"_q.arg(userId));
});
} else {
_verifiedCheck->setAttribute(Qt::WA_TransparentForMouseEvents);
}
2022-05-20 19:28:50 +04:00
} break;
2021-01-21 16:39:40 +04:00
case Badge::Scam:
case Badge::Fake: {
const auto fake = (_badge == Badge::Fake);
const auto size = Ui::ScamBadgeSize(fake);
2019-06-23 14:18:33 +02:00
const auto skip = st::infoVerifiedCheckPosition.x();
2021-01-21 16:39:40 +04:00
_scamFakeBadge.create(this);
_scamFakeBadge->show();
_scamFakeBadge->resize(
2019-06-23 14:18:33 +02:00
size.width() + 2 * skip,
size.height() + 2 * skip);
2021-01-21 16:39:40 +04:00
_scamFakeBadge->paintRequest(
) | rpl::start_with_next([=, badge = _scamFakeBadge.data()]{
2019-06-23 14:18:33 +02:00
Painter p(badge);
Ui::DrawScamBadge(
2021-01-21 16:39:40 +04:00
fake,
2019-06-23 14:18:33 +02:00
p,
badge->rect().marginsRemoved({ skip, skip, skip, skip }),
badge->width(),
st::attentionButtonFg);
2021-01-21 16:39:40 +04:00
}, _scamFakeBadge->lifetime());
} break;
2019-06-23 14:18:33 +02:00
}
refreshNameGeometry(width());
}
void Cover::refreshStatusText() {
auto hasMembersLink = [&] {
if (auto megagroup = _peer->asMegagroup()) {
return megagroup->canViewMembers();
}
return false;
}();
auto statusText = [&]() -> TextWithEntities {
using namespace Ui::Text;
auto currentTime = base::unixtime::now();
if (auto user = _peer->asUser()) {
const auto result = Data::OnlineTextFull(user, currentTime);
const auto showOnline = Data::OnlineTextActive(user, currentTime);
const auto updateIn = Data::OnlineChangeTimeout(user, currentTime);
if (showOnline) {
_refreshStatusTimer.callOnce(updateIn);
}
return showOnline
? PlainLink(result)
: TextWithEntities{ .text = result };
} else if (auto chat = _peer->asChat()) {
if (!chat->amIn()) {
return tr::lng_chat_status_unaccessible({}, WithEntities);
}
auto fullCount = std::max(
chat->count,
int(chat->participants.size()));
return { .text = ChatStatusText(fullCount, _onlineCount, true) };
} else if (auto channel = _peer->asChannel()) {
auto fullCount = qMax(channel->membersCount(), 1);
auto result = ChatStatusText(
fullCount,
_onlineCount,
channel->isMegagroup());
return hasMembersLink
? PlainLink(result)
: TextWithEntities{ .text = result };
}
return tr::lng_chat_status_unaccessible(tr::now, WithEntities);
}();
_status->setMarkedText(statusText);
if (hasMembersLink) {
_status->setLink(1, std::make_shared<LambdaClickHandler>([=] {
2018-09-05 22:39:35 +03:00
_showSection.fire(Section::Type::Members);
}));
}
refreshStatusGeometry(width());
}
2017-10-25 19:32:22 +03:00
Cover::~Cover() {
}
void Cover::refreshNameGeometry(int newWidth) {
2017-11-06 18:13:56 +04:00
auto nameLeft = st::infoProfileNameLeft;
auto nameTop = st::infoProfileNameTop;
auto nameWidth = newWidth
2017-11-06 18:13:56 +04:00
- nameLeft
- st::infoProfileNameRight;
2017-11-06 18:13:56 +04:00
if (_verifiedCheck) {
nameWidth -= st::infoVerifiedCheckPosition.x()
2019-06-23 14:18:33 +02:00
+ _verifiedCheck->width();
2021-01-21 16:39:40 +04:00
} else if (_scamFakeBadge) {
2019-06-23 14:18:33 +02:00
nameWidth -= st::infoVerifiedCheckPosition.x()
2021-01-21 16:39:40 +04:00
+ _scamFakeBadge->width();
2017-11-06 18:13:56 +04:00
}
_name->resizeToNaturalWidth(nameWidth);
_name->moveToLeft(nameLeft, nameTop, newWidth);
if (_verifiedCheck) {
2019-06-23 14:18:33 +02:00
const auto checkLeft = nameLeft
2017-11-06 18:13:56 +04:00
+ _name->width()
+ st::infoVerifiedCheckPosition.x();
2019-06-23 14:18:33 +02:00
const auto checkTop = nameTop
2017-11-06 18:13:56 +04:00
+ st::infoVerifiedCheckPosition.y();
_verifiedCheck->moveToLeft(checkLeft, checkTop, newWidth);
2021-01-21 16:39:40 +04:00
} else if (_scamFakeBadge) {
2019-06-23 14:18:33 +02:00
const auto skip = st::infoVerifiedCheckPosition.x();
const auto badgeLeft = nameLeft
+ _name->width()
+ st::infoVerifiedCheckPosition.x()
- skip;
const auto badgeTop = nameTop
2021-01-21 16:39:40 +04:00
+ (_name->height() - _scamFakeBadge->height()) / 2;
_scamFakeBadge->moveToLeft(badgeLeft, badgeTop, newWidth);
2017-11-06 18:13:56 +04:00
}
}
void Cover::refreshStatusGeometry(int newWidth) {
auto statusWidth = newWidth
- st::infoProfileStatusLeft
- st::infoProfileStatusRight;
_status->resizeToWidth(statusWidth);
_status->moveToLeft(
st::infoProfileStatusLeft,
st::infoProfileStatusTop,
newWidth);
}
} // namespace Profile
} // namespace Info