2022-05-19 00:30:46 +00:00
|
|
|
/*
|
|
|
|
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"
|
2022-06-12 05:23:23 +00:00
|
|
|
#include "styles/style_boxes.h"
|
2022-05-19 00:30:46 +00:00
|
|
|
|
|
|
|
namespace Ui {
|
|
|
|
|
|
|
|
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);
|
2022-06-12 05:23:23 +00:00
|
|
|
paintGlare(p);
|
|
|
|
|
2022-05-19 00:30:46 +00:00
|
|
|
const auto ripple = QColor(0, 0, 0, 36);
|
|
|
|
paintRipple(p, 0, 0, &ripple);
|
|
|
|
}
|
|
|
|
|
2022-06-12 05:23:23 +00:00
|
|
|
void GradientButton::paintGlare(QPainter &p) {
|
|
|
|
if (!_glare.glare.birthTime) {
|
|
|
|
return;
|
|
|
|
}
|
2022-10-11 13:13:20 +00:00
|
|
|
const auto progress = _glare.progress(crl::now());
|
2022-06-12 05:23:23 +00:00
|
|
|
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);
|
|
|
|
|
|
|
|
{
|
2022-09-16 20:23:27 +00:00
|
|
|
auto q = QPainter(&frame);
|
2022-06-12 05:23:23 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-19 00:30:46 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2022-06-14 09:42:43 +00:00
|
|
|
void GradientButton::setGlarePaused(bool paused) {
|
|
|
|
_glare.paused = paused;
|
|
|
|
}
|
|
|
|
|
2022-06-12 05:23:23 +00:00
|
|
|
void GradientButton::validateGlare() {
|
2022-10-11 13:13:20 +00:00
|
|
|
_glare.validate(
|
|
|
|
st::premiumButtonFg->c,
|
|
|
|
[=] { update(); },
|
|
|
|
st::gradientButtonGlareTimeout,
|
|
|
|
st::gradientButtonGlareDuration);
|
2022-06-12 05:23:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GradientButton::startGlareAnimation() {
|
|
|
|
validateGlare();
|
|
|
|
}
|
|
|
|
|
2022-05-19 00:30:46 +00:00
|
|
|
} // namespace Ui
|