210 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			210 lines
		
	
	
	
		
			5.3 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 "ui/unread_badge.h"
 | 
						|
 | 
						|
#include "data/data_peer.h"
 | 
						|
#include "data/data_user.h"
 | 
						|
#include "data/data_session.h"
 | 
						|
#include "data/stickers/data_custom_emoji.h"
 | 
						|
#include "main/main_session.h"
 | 
						|
#include "lang/lang_keys.h"
 | 
						|
#include "ui/painter.h"
 | 
						|
#include "ui/power_saving.h"
 | 
						|
#include "ui/unread_badge_paint.h"
 | 
						|
#include "styles/style_dialogs.h"
 | 
						|
 | 
						|
namespace Ui {
 | 
						|
namespace {
 | 
						|
 | 
						|
constexpr auto kPlayStatusLimit = 2;
 | 
						|
 | 
						|
} // namespace
 | 
						|
 | 
						|
void UnreadBadge::setText(const QString &text, bool active) {
 | 
						|
	_text = text;
 | 
						|
	_active = active;
 | 
						|
	const auto st = Dialogs::Ui::UnreadBadgeStyle();
 | 
						|
	resize(
 | 
						|
		std::max(st.font->width(text) + 2 * st.padding, st.size),
 | 
						|
		st.size);
 | 
						|
	update();
 | 
						|
}
 | 
						|
 | 
						|
int UnreadBadge::textBaseline() const {
 | 
						|
	const auto st = Dialogs::Ui::UnreadBadgeStyle();
 | 
						|
	return ((st.size - st.font->height) / 2) + st.font->ascent;
 | 
						|
}
 | 
						|
 | 
						|
void UnreadBadge::paintEvent(QPaintEvent *e) {
 | 
						|
	if (_text.isEmpty()) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	auto p = QPainter(this);
 | 
						|
 | 
						|
	UnreadBadgeStyle unreadSt;
 | 
						|
	unreadSt.muted = !_active;
 | 
						|
	auto unreadRight = width();
 | 
						|
	auto unreadTop = 0;
 | 
						|
	PaintUnreadBadge(
 | 
						|
		p,
 | 
						|
		_text,
 | 
						|
		unreadRight,
 | 
						|
		unreadTop,
 | 
						|
		unreadSt);
 | 
						|
}
 | 
						|
 | 
						|
QSize ScamBadgeSize(bool fake) {
 | 
						|
	const auto phrase = fake
 | 
						|
		? tr::lng_fake_badge(tr::now)
 | 
						|
		: tr::lng_scam_badge(tr::now);
 | 
						|
	const auto phraseWidth = st::dialogsScamFont->width(phrase);
 | 
						|
	const auto width = st::dialogsScamPadding.left()
 | 
						|
		+ phraseWidth
 | 
						|
		+ st::dialogsScamPadding.right();
 | 
						|
	const auto height = st::dialogsScamPadding.top()
 | 
						|
		+ st::dialogsScamFont->height
 | 
						|
		+ st::dialogsScamPadding.bottom();
 | 
						|
	return { width, height };
 | 
						|
}
 | 
						|
 | 
						|
void DrawScamFakeBadge(
 | 
						|
		Painter &p,
 | 
						|
		QRect rect,
 | 
						|
		int outerWidth,
 | 
						|
		const style::color &color,
 | 
						|
		const QString &phrase,
 | 
						|
		int phraseWidth) {
 | 
						|
	PainterHighQualityEnabler hq(p);
 | 
						|
	auto pen = color->p;
 | 
						|
	pen.setWidth(st::lineWidth);
 | 
						|
	p.setPen(pen);
 | 
						|
	p.setBrush(Qt::NoBrush);
 | 
						|
	p.drawRoundedRect(rect, st::dialogsScamRadius, st::dialogsScamRadius);
 | 
						|
	p.setFont(st::dialogsScamFont);
 | 
						|
	p.drawTextLeft(
 | 
						|
		rect.x() + st::dialogsScamPadding.left(),
 | 
						|
		rect.y() + st::dialogsScamPadding.top(),
 | 
						|
		outerWidth,
 | 
						|
		phrase,
 | 
						|
		phraseWidth);
 | 
						|
}
 | 
						|
 | 
						|
void DrawScamBadge(
 | 
						|
		bool fake,
 | 
						|
		Painter &p,
 | 
						|
		QRect rect,
 | 
						|
		int outerWidth,
 | 
						|
		const style::color &color) {
 | 
						|
	const auto phrase = fake
 | 
						|
		? tr::lng_fake_badge(tr::now)
 | 
						|
		: tr::lng_scam_badge(tr::now);
 | 
						|
	DrawScamFakeBadge(
 | 
						|
		p,
 | 
						|
		rect,
 | 
						|
		outerWidth,
 | 
						|
		color,
 | 
						|
		phrase,
 | 
						|
		st::dialogsScamFont->width(phrase));
 | 
						|
}
 | 
						|
 | 
						|
PeerBadge::PeerBadge() = default;
 | 
						|
 | 
						|
PeerBadge::~PeerBadge() = default;
 | 
						|
 | 
						|
int PeerBadge::drawGetWidth(
 | 
						|
		Painter &p,
 | 
						|
		QRect rectForName,
 | 
						|
		int nameWidth,
 | 
						|
		int outerWidth,
 | 
						|
		const Descriptor &descriptor) {
 | 
						|
	Expects(descriptor.customEmojiRepaint != nullptr);
 | 
						|
 | 
						|
	const auto peer = descriptor.peer;
 | 
						|
	if ((peer->isScam() || peer->isFake()) && descriptor.scam) {
 | 
						|
		const auto phrase = peer->isScam()
 | 
						|
			? tr::lng_scam_badge(tr::now)
 | 
						|
			: tr::lng_fake_badge(tr::now);
 | 
						|
		const auto phraseWidth = st::dialogsScamFont->width(phrase);
 | 
						|
		const auto width = st::dialogsScamPadding.left()
 | 
						|
			+ phraseWidth
 | 
						|
			+ st::dialogsScamPadding.right();
 | 
						|
		const auto height = st::dialogsScamPadding.top()
 | 
						|
			+ st::dialogsScamFont->height
 | 
						|
			+ st::dialogsScamPadding.bottom();
 | 
						|
		const auto rect = QRect(
 | 
						|
			(rectForName.x()
 | 
						|
				+ qMin(
 | 
						|
					nameWidth + st::dialogsScamSkip,
 | 
						|
					rectForName.width() - width)),
 | 
						|
			rectForName.y() + (rectForName.height() - height) / 2,
 | 
						|
			width,
 | 
						|
			height);
 | 
						|
		DrawScamFakeBadge(
 | 
						|
			p,
 | 
						|
			rect,
 | 
						|
			outerWidth,
 | 
						|
			*descriptor.scam,
 | 
						|
			phrase,
 | 
						|
			phraseWidth);
 | 
						|
		return st::dialogsScamSkip + width;
 | 
						|
	} else if (peer->isVerified() && descriptor.verified) {
 | 
						|
		const auto iconw = descriptor.verified->width();
 | 
						|
		descriptor.verified->paint(
 | 
						|
			p,
 | 
						|
			rectForName.x() + qMin(nameWidth, rectForName.width() - iconw),
 | 
						|
			rectForName.y(),
 | 
						|
			outerWidth);
 | 
						|
		return iconw;
 | 
						|
	} else if (peer->isPremium()
 | 
						|
		&& descriptor.premium
 | 
						|
		&& peer->session().premiumBadgesShown()) {
 | 
						|
		const auto id = peer->isUser() ? peer->asUser()->emojiStatusId() : 0;
 | 
						|
		const auto iconw = descriptor.premium->width();
 | 
						|
		const auto iconx = rectForName.x()
 | 
						|
			+ qMin(nameWidth, rectForName.width() - iconw);
 | 
						|
		const auto icony = rectForName.y();
 | 
						|
		if (!id) {
 | 
						|
			_emojiStatus = nullptr;
 | 
						|
			descriptor.premium->paint(p, iconx, icony, outerWidth);
 | 
						|
			return iconw;
 | 
						|
		}
 | 
						|
		if (!_emojiStatus) {
 | 
						|
			_emojiStatus = std::make_unique<EmojiStatus>();
 | 
						|
			const auto size = st::emojiSize;
 | 
						|
			const auto emoji = Ui::Text::AdjustCustomEmojiSize(size);
 | 
						|
			_emojiStatus->skip = (size - emoji) / 2;
 | 
						|
		}
 | 
						|
		if (_emojiStatus->id != id) {
 | 
						|
			using namespace Ui::Text;
 | 
						|
			auto &manager = peer->session().data().customEmojiManager();
 | 
						|
			_emojiStatus->id = id;
 | 
						|
			_emojiStatus->emoji = std::make_unique<LimitedLoopsEmoji>(
 | 
						|
				manager.create(
 | 
						|
					id,
 | 
						|
					descriptor.customEmojiRepaint),
 | 
						|
				kPlayStatusLimit);
 | 
						|
		}
 | 
						|
		_emojiStatus->emoji->paint(p, {
 | 
						|
			.textColor = (*descriptor.premiumFg)->c,
 | 
						|
			.now = descriptor.now,
 | 
						|
			.position = QPoint(
 | 
						|
				iconx - 2 * _emojiStatus->skip,
 | 
						|
				icony + _emojiStatus->skip),
 | 
						|
			.paused = descriptor.paused || On(PowerSaving::kEmojiStatus),
 | 
						|
		});
 | 
						|
		return iconw - 4 * _emojiStatus->skip;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
void PeerBadge::unload() {
 | 
						|
	_emojiStatus = nullptr;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Ui
 |