myEnsureResized -> Ui::SendPendingMoveResizeEvents. myGrab -> Ui::GrabWidget. myGrabImage -> Ui::GrabWidgetToImage.
		
			
				
	
	
		
			179 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			179 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
This file is part of Telegram Desktop,
 | 
						|
the official desktop version of Telegram messaging app, see https://telegram.org
 | 
						|
 | 
						|
Telegram Desktop is free software: you can redistribute it and/or modify
 | 
						|
it under the terms of the GNU General Public License as published by
 | 
						|
the Free Software Foundation, either version 3 of the License, or
 | 
						|
(at your option) any later version.
 | 
						|
 | 
						|
It is distributed in the hope that it will be useful,
 | 
						|
but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
						|
GNU General Public License for more details.
 | 
						|
 | 
						|
In addition, as a special exception, the copyright holders give permission
 | 
						|
to link the code of portions of this program with the OpenSSL library.
 | 
						|
 | 
						|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | 
						|
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
 | 
						|
*/
 | 
						|
#include "ui/effects/fade_animation.h"
 | 
						|
 | 
						|
namespace Ui {
 | 
						|
namespace {
 | 
						|
 | 
						|
constexpr int kWideScale = 5;
 | 
						|
 | 
						|
} // namespace
 | 
						|
 | 
						|
FadeAnimation::FadeAnimation(TWidget *widget, float64 scale)
 | 
						|
: _widget(widget)
 | 
						|
, _scale(scale) {
 | 
						|
}
 | 
						|
 | 
						|
bool FadeAnimation::paint(Painter &p) {
 | 
						|
	if (_cache.isNull()) return false;
 | 
						|
 | 
						|
	auto opacity = _animation.current(getms(), _visible ? 1. : 0.);
 | 
						|
	p.setOpacity(opacity);
 | 
						|
	if (_scale < 1.) {
 | 
						|
		PainterHighQualityEnabler hq(p);
 | 
						|
		auto targetRect = QRect(
 | 
						|
			(1 - kWideScale) / 2 * _size.width(),
 | 
						|
			(1 - kWideScale) / 2 * _size.height(),
 | 
						|
			kWideScale * _size.width(),
 | 
						|
			kWideScale * _size.height());
 | 
						|
		auto scale = opacity + (1. - opacity) * _scale;
 | 
						|
		auto shownWidth = anim::interpolate(
 | 
						|
			(1 - kWideScale) / 2 * _size.width(),
 | 
						|
			0,
 | 
						|
			scale);
 | 
						|
		auto shownHeight = anim::interpolate(
 | 
						|
			(1 - kWideScale) / 2 * _size.height(),
 | 
						|
			0,
 | 
						|
			scale);
 | 
						|
		auto margins = QMargins(
 | 
						|
			shownWidth,
 | 
						|
			shownHeight,
 | 
						|
			shownWidth,
 | 
						|
			shownHeight);
 | 
						|
		p.drawPixmap(targetRect.marginsAdded(margins), _cache);
 | 
						|
	} else {
 | 
						|
		p.drawPixmap(0, 0, _cache);
 | 
						|
	}
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::refreshCache() {
 | 
						|
	if (!_cache.isNull()) {
 | 
						|
		_cache = QPixmap();
 | 
						|
		_cache = grabContent();
 | 
						|
		Assert(!_cache.isNull());
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
QPixmap FadeAnimation::grabContent() {
 | 
						|
	SendPendingMoveResizeEvents(_widget);
 | 
						|
	_size = _widget->size();
 | 
						|
	if (_size.isEmpty()) {
 | 
						|
		auto image = QImage(
 | 
						|
			cIntRetinaFactor(),
 | 
						|
			cIntRetinaFactor(),
 | 
						|
			QImage::Format_ARGB32_Premultiplied);
 | 
						|
		image.fill(Qt::transparent);
 | 
						|
		return App::pixmapFromImageInPlace(std::move(image));
 | 
						|
	}
 | 
						|
	auto widgetContent = GrabWidget(_widget);
 | 
						|
	if (_scale < 1.) {
 | 
						|
		auto result = QImage(kWideScale * _size * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied);
 | 
						|
		result.setDevicePixelRatio(cRetinaFactor());
 | 
						|
		result.fill(Qt::transparent);
 | 
						|
		{
 | 
						|
			Painter p(&result);
 | 
						|
			p.drawPixmap((kWideScale - 1) / 2 * _size.width(), (kWideScale - 1) / 2 * _size.height(), widgetContent);
 | 
						|
		}
 | 
						|
		return App::pixmapFromImageInPlace(std::move(result));
 | 
						|
	}
 | 
						|
	return widgetContent;
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::setFinishedCallback(FinishedCallback &&callback) {
 | 
						|
	_finishedCallback = std::move(callback);
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::setUpdatedCallback(UpdatedCallback &&callback) {
 | 
						|
	_updatedCallback = std::move(callback);
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::show() {
 | 
						|
	if (!_visible) {
 | 
						|
		_visible = true;
 | 
						|
		_widget->showChildren();
 | 
						|
	}
 | 
						|
	stopAnimation();
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::hide() {
 | 
						|
	_visible = false;
 | 
						|
	stopAnimation();
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::stopAnimation() {
 | 
						|
	_animation.finish();
 | 
						|
	if (!_cache.isNull()) {
 | 
						|
		_cache = QPixmap();
 | 
						|
		if (_visible) {
 | 
						|
			_widget->showChildren();
 | 
						|
		}
 | 
						|
		if (_finishedCallback) {
 | 
						|
			_finishedCallback();
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if (_visible == _widget->isHidden()) {
 | 
						|
		_widget->setVisible(_visible);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::fadeIn(int duration) {
 | 
						|
	if (_visible) return;
 | 
						|
 | 
						|
	_visible = true;
 | 
						|
	startAnimation(duration);
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::fadeOut(int duration) {
 | 
						|
	if (!_visible) return;
 | 
						|
 | 
						|
	_visible = false;
 | 
						|
	startAnimation(duration);
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::startAnimation(int duration) {
 | 
						|
	if (_cache.isNull()) {
 | 
						|
		_widget->showChildren();
 | 
						|
		_cache = grabContent();
 | 
						|
		Assert(!_cache.isNull());
 | 
						|
		_widget->hideChildren();
 | 
						|
	}
 | 
						|
	auto from = _visible ? 0. : 1.;
 | 
						|
	auto to = _visible ? 1. : 0.;
 | 
						|
	_animation.start([this]() { updateCallback(); }, from, to, duration);
 | 
						|
	updateCallback();
 | 
						|
	if (_widget->isHidden()) {
 | 
						|
		_widget->show();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void FadeAnimation::updateCallback() {
 | 
						|
	if (_animation.animating()) {
 | 
						|
		_widget->update();
 | 
						|
		if (_updatedCallback) {
 | 
						|
			_updatedCallback(_animation.current(_visible ? 1. : 0.));
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		stopAnimation();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Ui
 |