416 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			416 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
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_peer_values.h"
 | 
						|
#include "data/data_channel.h"
 | 
						|
#include "data/data_chat.h"
 | 
						|
#include "data/data_peer.h"
 | 
						|
#include "data/data_document.h"
 | 
						|
#include "data/data_document_media.h"
 | 
						|
#include "data/data_changes.h"
 | 
						|
#include "data/data_session.h"
 | 
						|
#include "data/data_forum_topic.h"
 | 
						|
#include "data/stickers/data_custom_emoji.h"
 | 
						|
#include "info/profile/info_profile_values.h"
 | 
						|
#include "info/profile/info_profile_badge.h"
 | 
						|
#include "info/profile/info_profile_emoji_status_panel.h"
 | 
						|
#include "info/info_controller.h"
 | 
						|
#include "history/view/media/history_view_sticker_player.h"
 | 
						|
#include "lang/lang_keys.h"
 | 
						|
#include "ui/widgets/labels.h"
 | 
						|
#include "ui/text/text_utilities.h"
 | 
						|
#include "ui/special_buttons.h"
 | 
						|
#include "base/unixtime.h"
 | 
						|
#include "window/window_session_controller.h"
 | 
						|
#include "main/main_session.h"
 | 
						|
#include "settings/settings_premium.h"
 | 
						|
#include "chat_helpers/stickers_lottie.h"
 | 
						|
#include "apiwrap.h"
 | 
						|
#include "api/api_peer_photo.h"
 | 
						|
#include "styles/style_boxes.h"
 | 
						|
#include "styles/style_info.h"
 | 
						|
#include "styles/style_dialogs.h"
 | 
						|
 | 
						|
namespace Info::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) {
 | 
						|
		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);
 | 
						|
	}
 | 
						|
	return isGroup
 | 
						|
		? tr::lng_group_status(tr::now)
 | 
						|
		: tr::lng_channel_status(tr::now);
 | 
						|
};
 | 
						|
 | 
						|
} // namespace
 | 
						|
 | 
						|
Cover::Cover(
 | 
						|
	QWidget *parent,
 | 
						|
	not_null<PeerData*> peer,
 | 
						|
	not_null<Window::SessionController*> controller)
 | 
						|
: Cover(parent, peer, controller, NameValue(peer)) {
 | 
						|
}
 | 
						|
 | 
						|
Cover::Cover(
 | 
						|
	QWidget *parent,
 | 
						|
	not_null<Data::ForumTopic*> topic,
 | 
						|
	not_null<Window::SessionController*> controller)
 | 
						|
: Cover(
 | 
						|
	parent,
 | 
						|
	topic->channel(),
 | 
						|
	topic,
 | 
						|
	controller,
 | 
						|
	TitleValue(topic)) {
 | 
						|
}
 | 
						|
 | 
						|
Cover::Cover(
 | 
						|
	QWidget *parent,
 | 
						|
	not_null<PeerData*> peer,
 | 
						|
	not_null<Window::SessionController*> controller,
 | 
						|
	rpl::producer<QString> title)
 | 
						|
: Cover(
 | 
						|
	parent,
 | 
						|
	peer,
 | 
						|
	nullptr,
 | 
						|
	controller,
 | 
						|
	std::move(title)) {
 | 
						|
}
 | 
						|
 | 
						|
[[nodiscard]] const style::InfoProfileCover &CoverStyle(
 | 
						|
		not_null<PeerData*> peer,
 | 
						|
		Data::ForumTopic *topic) {
 | 
						|
	return topic
 | 
						|
		? st::infoTopicCover
 | 
						|
		: peer->isMegagroup()
 | 
						|
		? st::infoProfileMegagroupCover
 | 
						|
		: st::infoProfileCover;
 | 
						|
}
 | 
						|
 | 
						|
Cover::Cover(
 | 
						|
	QWidget *parent,
 | 
						|
	not_null<PeerData*> peer,
 | 
						|
	Data::ForumTopic *topic,
 | 
						|
	not_null<Window::SessionController*> controller,
 | 
						|
	rpl::producer<QString> title)
 | 
						|
: FixedHeightWidget(parent, CoverStyle(peer, topic).height)
 | 
						|
, _st(CoverStyle(peer, topic))
 | 
						|
, _controller(controller)
 | 
						|
, _peer(peer)
 | 
						|
, _emojiStatusPanel(peer->isSelf()
 | 
						|
	? std::make_unique<EmojiStatusPanel>()
 | 
						|
	: nullptr)
 | 
						|
, _badge(
 | 
						|
	std::make_unique<Badge>(
 | 
						|
		this,
 | 
						|
		st::infoPeerBadge,
 | 
						|
		peer,
 | 
						|
		_emojiStatusPanel.get(),
 | 
						|
		[=] {
 | 
						|
			return controller->isGifPausedAtLeastFor(
 | 
						|
				Window::GifPauseReason::Layer);
 | 
						|
		}))
 | 
						|
, _userpic(topic
 | 
						|
	? nullptr
 | 
						|
	: object_ptr<Ui::UserpicButton>(
 | 
						|
		this,
 | 
						|
		controller,
 | 
						|
		_peer,
 | 
						|
		Ui::UserpicButton::Role::OpenPhoto,
 | 
						|
		_st.photo))
 | 
						|
, _iconView(topic ? object_ptr<Ui::RpWidget>(this) : nullptr)
 | 
						|
, _name(this, _st.name)
 | 
						|
, _status(this, _st.status)
 | 
						|
, _refreshStatusTimer([this] { refreshStatusText(); }) {
 | 
						|
	if (topic) {
 | 
						|
		setupIcon(topic);
 | 
						|
	}
 | 
						|
 | 
						|
	_peer->updateFull();
 | 
						|
 | 
						|
	_name->setSelectable(true);
 | 
						|
	_name->setContextCopyText(tr::lng_profile_copy_fullname(tr::now));
 | 
						|
 | 
						|
	if (!_peer->isMegagroup()) {
 | 
						|
		_status->setAttribute(Qt::WA_TransparentForMouseEvents);
 | 
						|
	}
 | 
						|
 | 
						|
	_badge->setPremiumClickCallback([=] {
 | 
						|
		if (const auto panel = _emojiStatusPanel.get()) {
 | 
						|
			panel->show(_controller, _badge->widget(), _badge->sizeTag());
 | 
						|
		} else {
 | 
						|
			::Settings::ShowEmojiStatusPremium(_controller, _peer);
 | 
						|
		}
 | 
						|
	});
 | 
						|
	_badge->updated() | rpl::start_with_next([=] {
 | 
						|
		refreshNameGeometry(width());
 | 
						|
	}, _name->lifetime());
 | 
						|
 | 
						|
	initViewers(std::move(title));
 | 
						|
	setupChildGeometry();
 | 
						|
 | 
						|
	if (_userpic) {
 | 
						|
		_userpic->uploadPhotoRequests(
 | 
						|
		) | rpl::start_with_next([=] {
 | 
						|
			_peer->session().api().peerPhoto().upload(
 | 
						|
				_peer,
 | 
						|
				_userpic->takeResultImage());
 | 
						|
		}, _userpic->lifetime());
 | 
						|
	} else {
 | 
						|
		// #TODO forum icon change on click if possible
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Cover::setupIconPlayer(not_null<Data::ForumTopic*> topic) {
 | 
						|
	IconIdValue(
 | 
						|
		topic
 | 
						|
	) | rpl::map([=](DocumentId id) {
 | 
						|
		return topic->owner().customEmojiManager().resolve(id);
 | 
						|
	}) | rpl::flatten_latest(
 | 
						|
	) | rpl::map([=](not_null<DocumentData*> document) {
 | 
						|
		const auto media = document->createMediaView();
 | 
						|
		media->checkStickerLarge();
 | 
						|
		media->goodThumbnailWanted();
 | 
						|
 | 
						|
		return rpl::single() | rpl::then(
 | 
						|
			document->owner().session().downloaderTaskFinished()
 | 
						|
		) | rpl::filter([=] {
 | 
						|
			return media->loaded();
 | 
						|
		}) | rpl::take(1) | rpl::map([=] {
 | 
						|
			auto result = std::shared_ptr<StickerPlayer>();
 | 
						|
			const auto sticker = document->sticker();
 | 
						|
			if (sticker->isLottie()) {
 | 
						|
				result = std::make_shared<HistoryView::LottiePlayer>(
 | 
						|
					ChatHelpers::LottiePlayerFromDocument(
 | 
						|
						media.get(),
 | 
						|
						ChatHelpers::StickerLottieSize::StickerSet,
 | 
						|
						_st.photo.size,
 | 
						|
						Lottie::Quality::High));
 | 
						|
			} else if (sticker->isWebm()) {
 | 
						|
				result = std::make_shared<HistoryView::WebmPlayer>(
 | 
						|
					media->owner()->location(),
 | 
						|
					media->bytes(),
 | 
						|
					_st.photo.size);
 | 
						|
			} else {
 | 
						|
				result = std::make_shared<HistoryView::StaticStickerPlayer>(
 | 
						|
					media->owner()->location(),
 | 
						|
					media->bytes(),
 | 
						|
					_st.photo.size);
 | 
						|
			}
 | 
						|
			result->setRepaintCallback([=] { _iconView->update(); });
 | 
						|
			return result;
 | 
						|
		});
 | 
						|
	}) | rpl::flatten_latest(
 | 
						|
	) | rpl::start_with_next([=](std::shared_ptr<StickerPlayer> player) {
 | 
						|
		_iconPlayer = std::move(player);
 | 
						|
	}, lifetime());
 | 
						|
}
 | 
						|
 | 
						|
void Cover::setupIconImage(not_null<Data::ForumTopic*> topic) {
 | 
						|
	rpl::combine(
 | 
						|
		TitleValue(topic),
 | 
						|
		ColorIdValue(topic)
 | 
						|
	) | rpl::map([=](const QString &title, int32 colorId) {
 | 
						|
		using namespace Data;
 | 
						|
		return ForumTopicIconFrame(colorId, title, st::infoForumTopicIcon);
 | 
						|
	}) | rpl::start_with_next([=](QImage &&image) {
 | 
						|
		_iconImage = std::move(image);
 | 
						|
		_iconView->update();
 | 
						|
	}, lifetime());
 | 
						|
}
 | 
						|
 | 
						|
void Cover::setupIcon(not_null<Data::ForumTopic*> topic) {
 | 
						|
	setupIconPlayer(topic);
 | 
						|
	setupIconImage(topic);
 | 
						|
 | 
						|
	_iconView->resize(_st.photo.size);
 | 
						|
	_iconView->paintRequest(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		auto p = QPainter(_iconView.data());
 | 
						|
		const auto paint = [&](const QImage &image) {
 | 
						|
			const auto size = image.size() / image.devicePixelRatio();
 | 
						|
			p.drawImage(
 | 
						|
				(_st.photo.size.width() - size.width()) / 2,
 | 
						|
				(_st.photo.size.height() - size.height()) / 2,
 | 
						|
				image);
 | 
						|
		};
 | 
						|
		if (_iconPlayer && _iconPlayer->ready()) {
 | 
						|
			paint(_iconPlayer->frame(
 | 
						|
				_st.photo.size,
 | 
						|
				QColor(0, 0, 0, 0),
 | 
						|
				false,
 | 
						|
				crl::now(),
 | 
						|
				_controller->isGifPausedAtLeastFor(
 | 
						|
					Window::GifPauseReason::Layer)).image);
 | 
						|
			_iconPlayer->markFrameShown();
 | 
						|
		} else if (!topic->iconId() && !_iconImage.isNull()) {
 | 
						|
			paint(_iconImage);
 | 
						|
		}
 | 
						|
	}, _iconView->lifetime());
 | 
						|
}
 | 
						|
 | 
						|
void Cover::setupChildGeometry() {
 | 
						|
	widthValue(
 | 
						|
	) | rpl::start_with_next([this](int newWidth) {
 | 
						|
		if (_userpic) {
 | 
						|
			_userpic->moveToLeft(_st.photoLeft, _st.photoTop, newWidth);
 | 
						|
		} else {
 | 
						|
			_iconView->moveToLeft(_st.photoLeft, _st.photoTop, 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;
 | 
						|
}
 | 
						|
 | 
						|
void Cover::initViewers(rpl::producer<QString> title) {
 | 
						|
	using Flag = Data::PeerUpdate::Flag;
 | 
						|
	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());
 | 
						|
	} else if (_peer->isSelf()) {
 | 
						|
		refreshUploadPhotoOverlay();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Cover::refreshUploadPhotoOverlay() {
 | 
						|
	if (!_userpic) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	_userpic->switchChangePhotoOverlay([&] {
 | 
						|
		if (const auto chat = _peer->asChat()) {
 | 
						|
			return chat->canEditInformation();
 | 
						|
		} else if (const auto channel = _peer->asChannel()) {
 | 
						|
			return channel->canEditInformation();
 | 
						|
		}
 | 
						|
		return _peer->isSelf();
 | 
						|
	}());
 | 
						|
}
 | 
						|
 | 
						|
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>([=] {
 | 
						|
			_showSection.fire(Section::Type::Members);
 | 
						|
		}));
 | 
						|
	}
 | 
						|
	refreshStatusGeometry(width());
 | 
						|
}
 | 
						|
 | 
						|
Cover::~Cover() {
 | 
						|
}
 | 
						|
 | 
						|
void Cover::refreshNameGeometry(int newWidth) {
 | 
						|
	auto nameWidth = newWidth - _st.nameLeft - _st.rightSkip;
 | 
						|
	if (const auto widget = _badge->widget()) {
 | 
						|
		nameWidth -= st::infoVerifiedCheckPosition.x() + widget->width();
 | 
						|
	}
 | 
						|
	_name->resizeToNaturalWidth(nameWidth);
 | 
						|
	_name->moveToLeft(_st.nameLeft, _st.nameTop, newWidth);
 | 
						|
	const auto badgeLeft = _st.nameLeft + _name->width();
 | 
						|
	const auto badgeTop = _st.nameTop;
 | 
						|
	const auto badgeBottom = _st.nameTop + _name->height();
 | 
						|
	_badge->move(badgeLeft, badgeTop, badgeBottom);
 | 
						|
}
 | 
						|
 | 
						|
void Cover::refreshStatusGeometry(int newWidth) {
 | 
						|
	auto statusWidth = newWidth - _st.statusLeft - _st.rightSkip;
 | 
						|
	_status->resizeToWidth(statusWidth);
 | 
						|
	_status->moveToLeft(_st.statusLeft, _st.statusTop, newWidth);
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Info::Profile
 |