232 lines
		
	
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			232 lines
		
	
	
	
		
			5.4 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 "calls/calls_userpic.h"
 | 
						|
 | 
						|
#include "data/data_peer.h"
 | 
						|
#include "main/main_session.h"
 | 
						|
#include "data/data_changes.h"
 | 
						|
#include "data/data_peer.h"
 | 
						|
#include "data/data_session.h"
 | 
						|
#include "data/data_cloud_file.h"
 | 
						|
#include "data/data_photo_media.h"
 | 
						|
#include "data/data_file_origin.h"
 | 
						|
#include "ui/empty_userpic.h"
 | 
						|
#include "ui/painter.h"
 | 
						|
#include "apiwrap.h" // requestFullPeer.
 | 
						|
#include "styles/style_calls.h"
 | 
						|
 | 
						|
namespace Calls {
 | 
						|
namespace {
 | 
						|
 | 
						|
} // namespace
 | 
						|
 | 
						|
Userpic::Userpic(
 | 
						|
	not_null<QWidget*> parent,
 | 
						|
	not_null<PeerData*> peer,
 | 
						|
	rpl::producer<bool> muted)
 | 
						|
: _content(parent)
 | 
						|
, _peer(peer) {
 | 
						|
	setGeometry(0, 0, 0);
 | 
						|
	setup(std::move(muted));
 | 
						|
}
 | 
						|
 | 
						|
Userpic::~Userpic() = default;
 | 
						|
 | 
						|
void Userpic::setVisible(bool visible) {
 | 
						|
	_content.setVisible(visible);
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::setGeometry(int x, int y, int size) {
 | 
						|
	if (this->size() != size) {
 | 
						|
		_userPhoto = QPixmap();
 | 
						|
		_userPhotoFull = false;
 | 
						|
	}
 | 
						|
	_content.setGeometry(x, y, size, size);
 | 
						|
	_content.update();
 | 
						|
	if (_userPhoto.isNull()) {
 | 
						|
		refreshPhoto();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::setup(rpl::producer<bool> muted) {
 | 
						|
	_content.show();
 | 
						|
	_content.setAttribute(Qt::WA_TransparentForMouseEvents);
 | 
						|
 | 
						|
	_content.paintRequest(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		paint();
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	std::move(
 | 
						|
		muted
 | 
						|
	) | rpl::start_with_next([=](bool muted) {
 | 
						|
		setMuted(muted);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_peer->session().changes().peerFlagsValue(
 | 
						|
		_peer,
 | 
						|
		Data::PeerUpdate::Flag::Photo
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		processPhoto();
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_peer->session().downloaderTaskFinished(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		refreshPhoto();
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_mutedAnimation.stop();
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::setMuteLayout(QPoint position, int size, int stroke) {
 | 
						|
	_mutePosition = position;
 | 
						|
	_muteSize = size;
 | 
						|
	_muteStroke = stroke;
 | 
						|
	_content.update();
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::paint() {
 | 
						|
	auto p = QPainter(&_content);
 | 
						|
 | 
						|
	p.drawPixmap(0, 0, _userPhoto);
 | 
						|
	if (_muted && _muteSize > 0) {
 | 
						|
		auto hq = PainterHighQualityEnabler(p);
 | 
						|
		auto pen = st::callBgOpaque->p;
 | 
						|
		pen.setWidth(_muteStroke);
 | 
						|
		p.setPen(pen);
 | 
						|
		p.setBrush(st::callHangupBg);
 | 
						|
		const auto rect = QRect(
 | 
						|
			_mutePosition.x() - _muteSize / 2,
 | 
						|
			_mutePosition.y() - _muteSize / 2,
 | 
						|
			_muteSize,
 | 
						|
			_muteSize);
 | 
						|
		p.drawEllipse(rect);
 | 
						|
		st::callMutedPeerIcon.paintInCenter(p, rect);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::setMuted(bool muted) {
 | 
						|
	if (_muted == muted) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	_muted = muted;
 | 
						|
	_content.update();
 | 
						|
	//_mutedAnimation.start(
 | 
						|
	//	[=] { _content.update(); },
 | 
						|
	//	_muted ? 0. : 1.,
 | 
						|
	//	_muted ? 1. : 0.,
 | 
						|
	//	st::fadeWrapDuration);
 | 
						|
}
 | 
						|
 | 
						|
int Userpic::size() const {
 | 
						|
	return _content.width();
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::processPhoto() {
 | 
						|
	_userpic = _peer->createUserpicView();
 | 
						|
	_peer->loadUserpic();
 | 
						|
	const auto photo = _peer->userpicPhotoId()
 | 
						|
		? _peer->owner().photo(_peer->userpicPhotoId()).get()
 | 
						|
		: nullptr;
 | 
						|
	if (isGoodPhoto(photo)) {
 | 
						|
		_photo = photo->createMediaView();
 | 
						|
		_photo->wanted(Data::PhotoSize::Thumbnail, _peer->userpicPhotoOrigin());
 | 
						|
	} else {
 | 
						|
		_photo = nullptr;
 | 
						|
		if (_peer->userpicPhotoUnknown() || (photo && photo->isNull())) {
 | 
						|
			_peer->session().api().requestFullPeer(_peer);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	refreshPhoto();
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::refreshPhoto() {
 | 
						|
	if (!size()) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	const auto isNewBigPhoto = [&] {
 | 
						|
		return _photo
 | 
						|
			&& (_photo->image(Data::PhotoSize::Thumbnail) != nullptr)
 | 
						|
			&& (_photo->owner()->id != _userPhotoId || !_userPhotoFull);
 | 
						|
	}();
 | 
						|
	if (isNewBigPhoto) {
 | 
						|
		_userPhotoId = _photo->owner()->id;
 | 
						|
		_userPhotoFull = true;
 | 
						|
		createCache(_photo->image(Data::PhotoSize::Thumbnail));
 | 
						|
	} else if (_userPhoto.isNull()) {
 | 
						|
		if (const auto cloud = _peer->userpicCloudImage(_userpic)) {
 | 
						|
			auto image = Image(base::duplicate(*cloud));
 | 
						|
			createCache(&image);
 | 
						|
		} else {
 | 
						|
			createCache(nullptr);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void Userpic::createCache(Image *image) {
 | 
						|
	const auto size = this->size();
 | 
						|
	const auto real = size * cIntRetinaFactor();
 | 
						|
	//_useTransparency
 | 
						|
	//	? (Images::Option::RoundLarge
 | 
						|
	//		| Images::Option::RoundSkipBottomLeft
 | 
						|
	//		| Images::Option::RoundSkipBottomRight)
 | 
						|
	//	: Images::Option::None;
 | 
						|
	if (image) {
 | 
						|
		auto width = image->width();
 | 
						|
		auto height = image->height();
 | 
						|
		if (width > height) {
 | 
						|
			width = qMax((width * real) / height, 1);
 | 
						|
			height = real;
 | 
						|
		} else {
 | 
						|
			height = qMax((height * real) / width, 1);
 | 
						|
			width = real;
 | 
						|
		}
 | 
						|
		_userPhoto = image->pixNoCache(
 | 
						|
			{ width, height },
 | 
						|
			{
 | 
						|
				.options = Images::Option::RoundCircle,
 | 
						|
				.outer = { size, size },
 | 
						|
			});
 | 
						|
		_userPhoto.setDevicePixelRatio(cRetinaFactor());
 | 
						|
	} else {
 | 
						|
		auto filled = QImage(
 | 
						|
			QSize(real, real),
 | 
						|
			QImage::Format_ARGB32_Premultiplied);
 | 
						|
		filled.setDevicePixelRatio(cRetinaFactor());
 | 
						|
		filled.fill(Qt::transparent);
 | 
						|
		{
 | 
						|
			auto p = QPainter(&filled);
 | 
						|
			Ui::EmptyUserpic(
 | 
						|
				Ui::EmptyUserpic::UserpicColor(
 | 
						|
					Data::PeerColorIndex(_peer->id)),
 | 
						|
				_peer->name()
 | 
						|
			).paintCircle(p, 0, 0, size, size);
 | 
						|
		}
 | 
						|
		//_userPhoto = Images::PixmapFast(Images::Round(
 | 
						|
		//	std::move(filled),
 | 
						|
		//	ImageRoundRadius::Large,
 | 
						|
		//	RectPart::TopLeft | RectPart::TopRight));
 | 
						|
		_userPhoto = Images::PixmapFast(std::move(filled));
 | 
						|
	}
 | 
						|
 | 
						|
	_content.update();
 | 
						|
}
 | 
						|
 | 
						|
bool Userpic::isGoodPhoto(PhotoData *photo) const {
 | 
						|
	if (!photo || photo->isNull()) {
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	const auto badAspect = [](int a, int b) {
 | 
						|
		return a > 10 * b;
 | 
						|
	};
 | 
						|
	const auto width = photo->width();
 | 
						|
	const auto height = photo->height();
 | 
						|
	return !badAspect(width, height) && !badAspect(height, width);
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Calls
 |