tdesktop/Telegram/SourceFiles/ui/widgets/gradient_round_button.cpp

139 lines
3.5 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/widgets/gradient_round_button.h"
#include "ui/image/image_prepare.h"
#include "styles/style_boxes.h"
namespace Ui {
namespace {
constexpr auto kMaxGlareOpaque = 0.5;
} // namespace
GradientButton::GradientButton(QWidget *widget, QGradientStops stops)
: RippleButton(widget, st::defaultRippleAnimation)
, _stops(std::move(stops)) {
}
void GradientButton::paintEvent(QPaintEvent *e) {
QPainter p(this);
validateBg();
p.drawImage(0, 0, _bg);
paintGlare(p);
const auto ripple = QColor(0, 0, 0, 36);
paintRipple(p, 0, 0, &ripple);
}
void GradientButton::paintGlare(QPainter &p) {
if (!_glare.glare.birthTime) {
return;
}
const auto progress = (crl::now() - _glare.glare.birthTime)
/ float64(_glare.glare.deathTime - _glare.glare.birthTime);
const auto x = (-_glare.width) + (width() + _glare.width * 2) * progress;
const auto h = height();
const auto edgeWidth = _glare.width + st::roundRadiusLarge;
if (x > edgeWidth && x < (width() - edgeWidth)) {
p.drawTiledPixmap(x, 0, _glare.width, h, _glare.pixmap, 0, 0);
} else {
auto frame = QImage(
QSize(_glare.width, h) * style::DevicePixelRatio(),
QImage::Format_ARGB32_Premultiplied);
frame.setDevicePixelRatio(style::DevicePixelRatio());
frame.fill(Qt::transparent);
{
auto q = QPainter(&frame);
q.drawTiledPixmap(0, 0, _glare.width, h, _glare.pixmap, 0, 0);
q.setCompositionMode(QPainter::CompositionMode_DestinationIn);
q.drawImage(-x, 0, _bg, 0, 0);
}
p.drawImage(x, 0, frame);
}
}
void GradientButton::validateBg() {
const auto factor = devicePixelRatio();
if (!_bg.isNull()
&& (_bg.devicePixelRatio() == factor)
&& (_bg.size() == size() * factor)) {
return;
}
_bg = QImage(size() * factor, QImage::Format_ARGB32_Premultiplied);
_bg.setDevicePixelRatio(factor);
auto p = QPainter(&_bg);
auto gradient = QLinearGradient(QPointF(0, 0), QPointF(width(), 0));
gradient.setStops(_stops);
p.fillRect(rect(), gradient);
p.end();
_bg = Images::Round(std::move(_bg), ImageRoundRadius::Large);
}
void GradientButton::setGlarePaused(bool paused) {
_glare.paused = paused;
}
void GradientButton::validateGlare() {
if (anim::Disabled()) {
return;
}
_glare.width = st::gradientButtonGlareWidth;
_glare.animation.init([=](crl::time now) {
if (const auto diff = (now - _glare.glare.deathTime); diff > 0) {
if (diff > st::gradientButtonGlareTimeout && !_glare.paused) {
_glare.glare = Glare{
.birthTime = now,
.deathTime = now + st::gradientButtonGlareDuration,
};
update();
}
} else {
update();
}
});
_glare.animation.start();
{
auto pixmap = QPixmap(QSize(_glare.width, 1)
* style::DevicePixelRatio());
pixmap.setDevicePixelRatio(style::DevicePixelRatio());
pixmap.fill(Qt::transparent);
{
auto p = QPainter(&pixmap);
auto gradient = QLinearGradient(
QPointF(0, 0),
QPointF(_glare.width, 0));
auto color = st::premiumButtonFg->c;
color.setAlphaF(0);
const auto edge = color;
color.setAlphaF(kMaxGlareOpaque);
const auto middle = color;
gradient.setStops({
{ 0., edge },
{ .5, middle },
{ 1., edge },
});
p.fillRect(pixmap.rect(), QBrush(gradient));
}
_glare.pixmap = std::move(pixmap);
}
}
void GradientButton::startGlareAnimation() {
validateGlare();
}
} // namespace Ui