208 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			208 lines
		
	
	
	
		
			5.1 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 "window/window_slide_animation.h"
 | 
						|
 | 
						|
#include "styles/style_window.h"
 | 
						|
#include "styles/style_boxes.h"
 | 
						|
 | 
						|
namespace Window {
 | 
						|
 | 
						|
void SlideAnimation::paintContents(QPainter &p) const {
 | 
						|
	const auto retina = style::DevicePixelRatio();
 | 
						|
 | 
						|
	const auto slideLeft = (_direction == SlideDirection::FromLeft);
 | 
						|
	const auto progress = _animation.value(slideLeft ? 0. : 1.);
 | 
						|
	if (_withFade) {
 | 
						|
		const auto dt = slideLeft
 | 
						|
			? (1. - progress)
 | 
						|
			: progress;
 | 
						|
		const auto easeOut = anim::easeOutCirc(1., dt);
 | 
						|
		const auto easeIn = anim::easeInCirc(1., dt);
 | 
						|
		const auto arrivingAlpha = easeIn;
 | 
						|
		const auto departingAlpha = 1. - easeOut;
 | 
						|
		const auto leftWidthFull = _cacheUnder.width() / retina;
 | 
						|
		const auto rightWidthFull = _cacheOver.width() / retina;
 | 
						|
		const auto leftCoord = slideLeft
 | 
						|
			? anim::interpolate(-leftWidthFull, 0, easeOut)
 | 
						|
			: anim::interpolate(0, -leftWidthFull, easeIn);
 | 
						|
		const auto leftAlpha = (slideLeft ? arrivingAlpha : departingAlpha);
 | 
						|
		const auto rightCoord = slideLeft
 | 
						|
			? anim::interpolate(0, rightWidthFull, easeIn)
 | 
						|
			: anim::interpolate(rightWidthFull, 0, easeOut);
 | 
						|
		const auto rightAlpha = (slideLeft ? departingAlpha : arrivingAlpha);
 | 
						|
 | 
						|
		const auto leftWidth = (leftWidthFull + leftCoord);
 | 
						|
		const auto rightWidth = rightWidthFull - rightCoord;
 | 
						|
 | 
						|
		if (!_mask.isNull()) {
 | 
						|
			auto frame = QImage(
 | 
						|
				_mask.size(),
 | 
						|
				QImage::Format_ARGB32_Premultiplied);
 | 
						|
			frame.setDevicePixelRatio(_mask.devicePixelRatio());
 | 
						|
			frame.fill(Qt::transparent);
 | 
						|
			QPainter q(&frame);
 | 
						|
 | 
						|
			if (leftWidth > 0) {
 | 
						|
				q.setOpacity(leftAlpha);
 | 
						|
				q.drawPixmap(
 | 
						|
					0,
 | 
						|
					0,
 | 
						|
					_cacheUnder,
 | 
						|
					(_cacheUnder.width() - leftWidth * cIntRetinaFactor()),
 | 
						|
					0,
 | 
						|
					leftWidth * cIntRetinaFactor(),
 | 
						|
					_topSkip * retina);
 | 
						|
			}
 | 
						|
 | 
						|
			if (rightWidth > 0) {
 | 
						|
				q.setOpacity(rightAlpha);
 | 
						|
				q.drawPixmap(
 | 
						|
					rightCoord,
 | 
						|
					0,
 | 
						|
					_cacheOver,
 | 
						|
					0,
 | 
						|
					0,
 | 
						|
					rightWidth * cIntRetinaFactor(),
 | 
						|
					_topSkip * retina);
 | 
						|
			}
 | 
						|
 | 
						|
			q.setOpacity(1.);
 | 
						|
			q.setCompositionMode(QPainter::CompositionMode_DestinationIn);
 | 
						|
			q.drawPixmap(0, 0, _mask);
 | 
						|
 | 
						|
			p.drawImage(0, 0, frame);
 | 
						|
		}
 | 
						|
 | 
						|
		if (leftWidth > 0) {
 | 
						|
			p.setOpacity(leftAlpha);
 | 
						|
			p.drawPixmap(
 | 
						|
				0,
 | 
						|
				_topSkip,
 | 
						|
				_cacheUnder,
 | 
						|
				(_cacheUnder.width() - leftWidth * retina),
 | 
						|
				_topSkip * retina,
 | 
						|
				leftWidth * retina,
 | 
						|
				_cacheUnder.height() - _topSkip * retina);
 | 
						|
		}
 | 
						|
		if (rightWidth > 0) {
 | 
						|
			p.setOpacity(rightAlpha);
 | 
						|
			p.drawPixmap(
 | 
						|
				rightCoord,
 | 
						|
				_topSkip,
 | 
						|
				_cacheOver,
 | 
						|
				0,
 | 
						|
				_topSkip * retina,
 | 
						|
				rightWidth * retina,
 | 
						|
				_cacheOver.height() - _topSkip * retina);
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		const auto coordUnder = anim::interpolate(
 | 
						|
			0,
 | 
						|
			-st::slideShift,
 | 
						|
			progress);
 | 
						|
		const auto coordOver = anim::interpolate(
 | 
						|
			_cacheOver.width() / retina,
 | 
						|
			0,
 | 
						|
			progress);
 | 
						|
		if (coordOver) {
 | 
						|
			p.drawPixmap(
 | 
						|
				QRect(0, 0, coordOver, _cacheUnder.height() / retina),
 | 
						|
				_cacheUnder,
 | 
						|
				QRect(
 | 
						|
					-coordUnder * retina,
 | 
						|
					0,
 | 
						|
					coordOver * retina,
 | 
						|
					_cacheUnder.height()));
 | 
						|
			p.setOpacity(progress);
 | 
						|
			p.fillRect(
 | 
						|
				0,
 | 
						|
				0,
 | 
						|
				coordOver, _cacheUnder.height() / retina,
 | 
						|
				st::slideFadeOutBg);
 | 
						|
			p.setOpacity(1);
 | 
						|
		}
 | 
						|
		p.drawPixmap(
 | 
						|
			QRect(QPoint(coordOver, 0), _cacheOver.size() / retina),
 | 
						|
			_cacheOver,
 | 
						|
			QRect(QPoint(), _cacheOver.size()));
 | 
						|
		p.setOpacity(progress);
 | 
						|
		st::slideShadow.fill(
 | 
						|
			p,
 | 
						|
			QRect(
 | 
						|
				coordOver - st::slideShadow.width(),
 | 
						|
				0,
 | 
						|
				st::slideShadow.width(),
 | 
						|
				_cacheOver.height() / retina));
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
float64 SlideAnimation::progress() const {
 | 
						|
	const auto slideLeft = (_direction == SlideDirection::FromLeft);
 | 
						|
	const auto progress = _animation.value(slideLeft ? 0. : 1.);
 | 
						|
	return slideLeft ? (1. - progress) : progress;
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setDirection(SlideDirection direction) {
 | 
						|
	_direction = direction;
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setPixmaps(
 | 
						|
		const QPixmap &oldContentCache,
 | 
						|
		const QPixmap &newContentCache) {
 | 
						|
	_cacheUnder = oldContentCache;
 | 
						|
	_cacheOver = newContentCache;
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setTopBarShadow(bool enabled) {
 | 
						|
	_topBarShadowEnabled = enabled;
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setTopSkip(int skip) {
 | 
						|
	_topSkip = skip;
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setWithFade(bool withFade) {
 | 
						|
	_withFade = withFade;
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setRepaintCallback(RepaintCallback &&callback) {
 | 
						|
	_repaintCallback = std::move(callback);
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setFinishedCallback(FinishedCallback &&callback) {
 | 
						|
	_finishedCallback = std::move(callback);
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::setTopBarMask(const QPixmap &mask) {
 | 
						|
	_mask = mask;
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::start() {
 | 
						|
	const auto fromLeft = (_direction == SlideDirection::FromLeft);
 | 
						|
	if (fromLeft) {
 | 
						|
		std::swap(_cacheUnder, _cacheOver);
 | 
						|
	}
 | 
						|
	_animation.start(
 | 
						|
		[this] { animationCallback(); },
 | 
						|
		fromLeft ? 1. : 0.,
 | 
						|
		fromLeft ? 0. : 1.,
 | 
						|
		st::slideDuration,
 | 
						|
		transition());
 | 
						|
	_repaintCallback();
 | 
						|
}
 | 
						|
 | 
						|
void SlideAnimation::animationCallback() {
 | 
						|
	_repaintCallback();
 | 
						|
	if (!_animation.animating()) {
 | 
						|
		if (const auto onstack = _finishedCallback) {
 | 
						|
			onstack();
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Window
 |