ContextAlbum: Improve album cover fading

This commit is contained in:
Jonas Kvinge
2022-06-05 18:20:44 +02:00
parent 4f1f2b6fc6
commit 23f3d2095b
2 changed files with 170 additions and 90 deletions

View File

@@ -44,6 +44,7 @@
#include "contextalbum.h" #include "contextalbum.h"
const int ContextAlbum::kWidgetSpacing = 40; const int ContextAlbum::kWidgetSpacing = 40;
const int ContextAlbum::kFadeTimeLineMs = 1000;
ContextAlbum::ContextAlbum(QWidget *parent) ContextAlbum::ContextAlbum(QWidget *parent)
: QWidget(parent), : QWidget(parent),
@@ -51,10 +52,10 @@ ContextAlbum::ContextAlbum(QWidget *parent)
context_view_(nullptr), context_view_(nullptr),
album_cover_choice_controller_(nullptr), album_cover_choice_controller_(nullptr),
downloading_covers_(false), downloading_covers_(false),
timeline_fade_(new QTimeLine(1000, this)), timeline_fade_(new QTimeLine(kFadeTimeLineMs, this)),
image_strawberry_(":/pictures/strawberry.png"), image_strawberry_(":/pictures/strawberry.png"),
image_original_(image_strawberry_), image_original_(image_strawberry_),
pixmap_previous_opacity_(0), pixmap_current_opacity_(1.0),
prev_width_(width()) { prev_width_(width()) {
setObjectName("context-widget-album"); setObjectName("context-widget-album");
@@ -63,12 +64,14 @@ ContextAlbum::ContextAlbum(QWidget *parent)
cover_loader_options_.pad_output_image_ = true; cover_loader_options_.pad_output_image_ = true;
cover_loader_options_.scale_output_image_ = true; cover_loader_options_.scale_output_image_ = true;
QImage image = ImageUtils::ScaleAndPad(image_strawberry_, cover_loader_options_.scale_output_image_, cover_loader_options_.pad_output_image_, cover_loader_options_.desired_height_); QImage image = ImageUtils::ScaleAndPad(image_strawberry_, cover_loader_options_.scale_output_image_, cover_loader_options_.pad_output_image_, cover_loader_options_.desired_height_);
if (!image.isNull()) pixmap_current_ = QPixmap::fromImage(image); if (!image.isNull()) {
pixmap_current_ = QPixmap::fromImage(image);
setFixedHeight(pixmap_current_.height());
}
setFixedHeight(image.height()); timeline_fade_->setDirection(QTimeLine::Forward);
QObject::connect(timeline_fade_, &QTimeLine::valueChanged, this, &ContextAlbum::FadeCurrentCover);
QObject::connect(timeline_fade_, &QTimeLine::valueChanged, this, &ContextAlbum::FadePreviousTrack); QObject::connect(timeline_fade_, &QTimeLine::finished, this, &ContextAlbum::FadeCurrentCoverFinished);
timeline_fade_->setDirection(QTimeLine::Backward); // 1.0 -> 0.0
} }
@@ -93,15 +96,27 @@ QSize ContextAlbum::sizeHint() const {
} }
void ContextAlbum::contextMenuEvent(QContextMenuEvent *e) { void ContextAlbum::resizeEvent(QResizeEvent *e) {
if (menu_ && image_original_ != image_strawberry_) { if (width() != prev_width_) {
menu_->popup(mapToGlobal(e->pos())); ScaleCover();
} ScalePreviousCovers();
else { prev_width_ = width();
QWidget::contextMenuEvent(e);
} }
QWidget::resizeEvent(e);
}
void ContextAlbum::paintEvent(QPaintEvent*) {
QPainter p(this);
p.setRenderHint(QPainter::SmoothPixmapTransform);
DrawPreviousCovers(&p);
DrawImage(&p, pixmap_current_, pixmap_current_opacity_);
DrawSpinner(&p);
p.end();
} }
void ContextAlbum::mouseDoubleClickEvent(QMouseEvent *e) { void ContextAlbum::mouseDoubleClickEvent(QMouseEvent *e) {
@@ -113,96 +128,143 @@ void ContextAlbum::mouseDoubleClickEvent(QMouseEvent *e) {
} }
void ContextAlbum::paintEvent(QPaintEvent*) { void ContextAlbum::contextMenuEvent(QContextMenuEvent *e) {
QPainter p(this); if (menu_ && image_original_ != image_strawberry_) {
menu_->popup(mapToGlobal(e->pos()));
DrawImage(&p);
// Draw the previous track's image if we're fading
if (!pixmap_previous_.isNull()) {
p.setOpacity(pixmap_previous_opacity_);
p.drawPixmap(0, 0, pixmap_previous_);
} }
else {
} QWidget::contextMenuEvent(e);
void ContextAlbum::DrawImage(QPainter *p) {
p->setRenderHint(QPainter::SmoothPixmapTransform);
int current_width = width();
if (current_width != prev_width_) {
cover_loader_options_.desired_height_ = width();
QImage image = ImageUtils::ScaleAndPad(image_original_, cover_loader_options_.scale_output_image_, cover_loader_options_.pad_output_image_, cover_loader_options_.desired_height_);
if (image.isNull()) pixmap_current_ = QPixmap();
else pixmap_current_ = QPixmap::fromImage(image);
prev_width_ = width();
setFixedHeight(image.height());
} }
p->drawPixmap(0, 0, pixmap_current_.width(), pixmap_current_.height(), pixmap_current_);
if (downloading_covers_ && spinner_animation_) {
p->drawPixmap(50, 50, 16, 16, spinner_animation_->currentPixmap());
}
}
void ContextAlbum::FadePreviousTrack(const qreal value) {
pixmap_previous_opacity_ = value;
if (qFuzzyCompare(pixmap_previous_opacity_, qreal(0.0))) {
image_previous_ = QImage();
pixmap_previous_ = QPixmap();
}
update();
if (value == 0 && image_original_ == image_strawberry_) {
emit FadeStopFinished();
}
}
void ContextAlbum::ScaleCover() {
cover_loader_options_.desired_height_ = width();
QImage image = ImageUtils::ScaleAndPad(image_original_, cover_loader_options_.scale_output_image_, cover_loader_options_.pad_output_image_, cover_loader_options_.desired_height_);
if (image.isNull()) pixmap_current_ = QPixmap();
else pixmap_current_ = QPixmap::fromImage(image);
prev_width_ = width();
update();
} }
void ContextAlbum::SetImage(QImage image) { void ContextAlbum::SetImage(QImage image) {
if (image.isNull()) image = image_strawberry_; if (image.isNull()) {
image = image_strawberry_;
}
if (downloading_covers_) { if (downloading_covers_) {
downloading_covers_ = false; downloading_covers_ = false;
spinner_animation_.reset(); spinner_animation_.reset();
} }
// Cache the current pixmap so we can fade between them QImage image_previous = image_original_;
pixmap_previous_ = QPixmap(width(), width()); QPixmap pixmap_previous = pixmap_current_;
pixmap_previous_.fill(palette().window().color()); qreal opacity_previous = pixmap_current_opacity_;
pixmap_previous_opacity_ = 1.0;
QPainter p(&pixmap_previous_);
DrawImage(&p);
p.end();
image_previous_ = image_original_;
image_original_ = image; image_original_ = image;
pixmap_current_opacity_ = 0.0;
ScaleCover(); ScaleCover();
// Were we waiting for this cover to load before we started fading? if (!pixmap_previous.isNull()) {
if (!pixmap_previous_.isNull() && timeline_fade_) { std::shared_ptr<PreviousCover> previous_cover = std::make_shared<PreviousCover>();
previous_cover->image = image_previous;
previous_cover->pixmap = pixmap_previous;
previous_cover->opacity = opacity_previous;
previous_cover->timeline.reset(new QTimeLine(kFadeTimeLineMs), [](QTimeLine *timeline) { timeline->deleteLater(); });
previous_cover->timeline->setDirection(QTimeLine::Backward);
previous_cover->timeline->setCurrentTime(timeline_fade_->state() == QTimeLine::Running ? timeline_fade_->currentTime() : kFadeTimeLineMs);
QObject::connect(previous_cover->timeline.get(), &QTimeLine::valueChanged, this, [this, previous_cover]() { FadePreviousCover(previous_cover); });
QObject::connect(previous_cover->timeline.get(), &QTimeLine::finished, this, [this, previous_cover]() { FadePreviousCoverFinished(previous_cover); });
previous_covers_ << previous_cover;
previous_cover->timeline->start();
}
if (timeline_fade_->state() == QTimeLine::Running) {
timeline_fade_->stop(); timeline_fade_->stop();
timeline_fade_->setDirection(QTimeLine::Backward); // 1.0 -> 0.0 }
timeline_fade_->start(); timeline_fade_->start();
}
void ContextAlbum::DrawImage(QPainter *p, const QPixmap &pixmap, const qreal opacity) {
if (qFuzzyCompare(opacity, qreal(0.0))) return;
p->setOpacity(opacity);
p->drawPixmap(0, 0, pixmap.width(), pixmap.height(), pixmap);
}
void ContextAlbum::DrawSpinner(QPainter *p) {
if (downloading_covers_) {
p->drawPixmap(50, 50, 16, 16, spinner_animation_->currentPixmap());
}
}
void ContextAlbum::DrawPreviousCovers(QPainter *p) {
for (std::shared_ptr<PreviousCover> previous_cover : previous_covers_) {
DrawImage(p, previous_cover->pixmap, previous_cover->opacity);
}
}
void ContextAlbum::FadeCurrentCover(const qreal value) {
if (value <= pixmap_current_opacity_) return;
pixmap_current_opacity_ = value;
update();
}
void ContextAlbum::FadeCurrentCoverFinished() {
if (image_original_ == image_strawberry_) {
emit FadeStopFinished();
}
}
void ContextAlbum::FadePreviousCover(std::shared_ptr<PreviousCover> previous_cover) {
if (previous_cover->timeline->currentValue() >= previous_cover->opacity) return;
previous_cover->opacity = previous_cover->timeline->currentValue();
}
void ContextAlbum::FadePreviousCoverFinished(std::shared_ptr<PreviousCover> previous_cover) {
previous_covers_.removeAll(previous_cover);
}
void ContextAlbum::ScaleCover() {
cover_loader_options_.desired_height_ = width();
int height = 0;
QImage image = ImageUtils::ScaleAndPad(image_original_, cover_loader_options_.scale_output_image_, cover_loader_options_.pad_output_image_, cover_loader_options_.desired_height_);
if (image.isNull()) {
pixmap_current_ = QPixmap();
}
else {
pixmap_current_ = QPixmap::fromImage(image);
height = pixmap_current_.height();
}
setFixedHeight(height);
}
void ContextAlbum::ScalePreviousCovers() {
for (std::shared_ptr<PreviousCover> previous_cover : previous_covers_) {
if (previous_cover->pixmap.width() == width()) continue;
QImage image = ImageUtils::ScaleAndPad(previous_cover->image, cover_loader_options_.scale_output_image_, cover_loader_options_.pad_output_image_, cover_loader_options_.desired_height_);
if (image.isNull()) {
previous_cover->pixmap = QPixmap();
}
else {
previous_cover->pixmap = QPixmap::fromImage(image);
}
} }
} }

View File

@@ -27,6 +27,7 @@
#include <QtGlobal> #include <QtGlobal>
#include <QObject> #include <QObject>
#include <QWidget> #include <QWidget>
#include <QList>
#include <QString> #include <QString>
#include <QImage> #include <QImage>
#include <QPixmap> #include <QPixmap>
@@ -54,12 +55,27 @@ class ContextAlbum : public QWidget {
protected: protected:
QSize sizeHint() const override; QSize sizeHint() const override;
void paintEvent(QPaintEvent*) override; void paintEvent(QPaintEvent*) override;
void contextMenuEvent(QContextMenuEvent *e) override; void resizeEvent(QResizeEvent *e) override;
void mouseDoubleClickEvent(QMouseEvent *e) override; void mouseDoubleClickEvent(QMouseEvent *e) override;
void contextMenuEvent(QContextMenuEvent *e) override;
private: private:
void DrawImage(QPainter *p);
struct PreviousCover {
PreviousCover() : opacity(0.0) {}
QImage image;
QPixmap pixmap;
qreal opacity;
std::shared_ptr<QTimeLine> timeline;
};
QList<std::shared_ptr<PreviousCover>> previous_covers_;
void DrawImage(QPainter *p, const QPixmap &pixmap, const qreal opacity);
void DrawSpinner(QPainter *p);
void DrawPreviousCovers(QPainter *p);
void ScaleCover(); void ScaleCover();
void ScalePreviousCovers();
void GetCoverAutomatically(); void GetCoverAutomatically();
signals: signals:
@@ -68,13 +84,17 @@ class ContextAlbum : public QWidget {
private slots: private slots:
void Update() { update(); } void Update() { update(); }
void AutomaticCoverSearchDone(); void AutomaticCoverSearchDone();
void FadePreviousTrack(const qreal value); void FadeCurrentCover(const qreal value);
void FadeCurrentCoverFinished();
void FadePreviousCover(std::shared_ptr<PreviousCover> previouscover);
void FadePreviousCoverFinished(std::shared_ptr<PreviousCover> previouscover);
public slots: public slots:
void SearchCoverInProgress(); void SearchCoverInProgress();
private: private:
static const int kWidgetSpacing; static const int kWidgetSpacing;
static const int kFadeTimeLineMs;
private: private:
QMenu *menu_; QMenu *menu_;
@@ -85,10 +105,8 @@ class ContextAlbum : public QWidget {
QTimeLine *timeline_fade_; QTimeLine *timeline_fade_;
QImage image_strawberry_; QImage image_strawberry_;
QImage image_original_; QImage image_original_;
QImage image_previous_;
QPixmap pixmap_current_; QPixmap pixmap_current_;
QPixmap pixmap_previous_; qreal pixmap_current_opacity_;
qreal pixmap_previous_opacity_;
std::unique_ptr<QMovie> spinner_animation_; std::unique_ptr<QMovie> spinner_animation_;
int prev_width_; int prev_width_;
}; };