210 lines
5.1 KiB
C++
210 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 * style::DevicePixelRatio(),
|
|
0,
|
|
leftWidth * style::DevicePixelRatio(),
|
|
_topSkip * retina);
|
|
}
|
|
|
|
if (rightWidth > 0) {
|
|
q.setOpacity(rightAlpha);
|
|
q.drawPixmap(
|
|
rightCoord,
|
|
0,
|
|
_cacheOver,
|
|
0,
|
|
0,
|
|
rightWidth * style::DevicePixelRatio(),
|
|
_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
|