Clementine-audio-player-Mac.../src/widgets/tracksliderpopup.cpp

177 lines
5.8 KiB
C++

/* This file is part of Clementine.
Copyright 2011, David Sansome <me@davidsansome.com>
Clementine is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Clementine is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Clementine. If not, see <http://www.gnu.org/licenses/>.
*/
#include "tracksliderpopup.h"
#include <QBitmap>
#include <QCoreApplication>
#include <QMouseEvent>
#include <QPainter>
#include <QTimer>
#include <QWheelEvent>
#include <QtDebug>
#include "ui/qt_blurimage.h"
const int TrackSliderPopup::kTextMargin = 4;
const int TrackSliderPopup::kPointLength = 16;
const int TrackSliderPopup::kPointWidth = 4;
const int TrackSliderPopup::kBorderRadius = 4;
const qreal TrackSliderPopup::kBlurRadius = 20.0;
TrackSliderPopup::TrackSliderPopup(QWidget* parent)
: QWidget(parent),
font_metrics_(fontMetrics()),
small_font_metrics_(fontMetrics()) {
setAttribute(Qt::WA_TransparentForMouseEvents);
setMouseTracking(true);
font_.setPointSizeF(7.5);
font_.setBold(true);
small_font_.setPointSizeF(7.5);
font_metrics_ = QFontMetrics(font_);
small_font_metrics_ = QFontMetrics(small_font_);
}
void TrackSliderPopup::SetText(const QString& text) {
text_ = text;
UpdatePixmap();
}
void TrackSliderPopup::SetSmallText(const QString& text) {
small_text_ = text;
UpdatePixmap();
}
void TrackSliderPopup::SetPopupPosition(const QPoint& pos) {
pos_ = pos;
UpdatePosition();
}
void TrackSliderPopup::paintEvent(QPaintEvent*) {
QPainter p(this);
p.drawPixmap(0, 0, pixmap_);
}
void TrackSliderPopup::UpdatePixmap() {
const int text_width =
qMax(font_metrics_.width(text_), small_font_metrics_.width(small_text_));
const QRect text_rect1(kBlurRadius + kTextMargin, kBlurRadius + kTextMargin,
text_width + 2, font_metrics_.height());
const QRect text_rect2(kBlurRadius + kTextMargin, text_rect1.bottom(),
text_width, small_font_metrics_.height());
const int bubble_bottom = text_rect2.bottom() + kTextMargin;
const QRect total_rect(0, 0, text_rect1.right() + kBlurRadius + kTextMargin,
kBlurRadius + bubble_bottom + kPointLength);
const QRect bubble_rect(kBlurRadius, kBlurRadius,
total_rect.width() - kBlurRadius * 2,
bubble_bottom - kBlurRadius);
if (background_cache_.size() != total_rect.size()) {
const QColor highlight(
palette().color(QPalette::Active, QPalette::Highlight));
const QColor bg_color_1(highlight.lighter(110));
const QColor bg_color_2(highlight.darker(120));
QPolygon pointy;
pointy << QPoint(total_rect.width() / 2 - kPointWidth, bubble_bottom)
<< QPoint(total_rect.width() / 2, total_rect.bottom() - kBlurRadius)
<< QPoint(total_rect.width() / 2 + kPointWidth, bubble_bottom);
QPolygon inner_pointy;
inner_pointy << QPoint(pointy[0].x() + 1, pointy[0].y() - 1)
<< QPoint(pointy[1].x(), pointy[1].y() - 1)
<< QPoint(pointy[2].x() - 1, pointy[2].y() - 1);
background_cache_ = QPixmap(total_rect.size());
background_cache_.fill(Qt::transparent);
QPainter p(&background_cache_);
p.setRenderHint(QPainter::Antialiasing);
p.setRenderHint(QPainter::HighQualityAntialiasing);
// Draw the shadow to a different image
QImage blur_source(total_rect.size(), QImage::Format_ARGB32);
blur_source.fill(Qt::transparent);
QPainter blur_painter(&blur_source);
blur_painter.setRenderHint(QPainter::Antialiasing);
blur_painter.setRenderHint(QPainter::HighQualityAntialiasing);
blur_painter.setBrush(bg_color_2);
blur_painter.drawRoundedRect(bubble_rect, kBorderRadius, kBorderRadius);
blur_painter.drawPolygon(pointy);
// Fade the shadow out towards the bottom
QLinearGradient fade_gradient(QPoint(0, bubble_bottom),
QPoint(0, bubble_bottom + kPointLength));
fade_gradient.setColorAt(0.0, QColor(255, 0, 0, 0));
fade_gradient.setColorAt(1.0, QColor(255, 0, 0, 255));
blur_painter.setCompositionMode(QPainter::CompositionMode_DestinationOut);
blur_painter.fillRect(total_rect, fade_gradient);
blur_painter.end();
p.save();
qt_blurImage(&p, blur_source, kBlurRadius, true, false);
p.restore();
// Outer bubble
p.setPen(Qt::NoPen);
p.setBrush(bg_color_2);
p.drawRoundedRect(bubble_rect, kBorderRadius, kBorderRadius);
// Outer pointy
p.drawPolygon(pointy);
// Inner bubble
p.setBrush(bg_color_1);
p.drawRoundedRect(bubble_rect.adjusted(1, 1, -1, -1), kBorderRadius,
kBorderRadius);
// Inner pointy
p.drawPolygon(inner_pointy);
}
pixmap_ = QPixmap(total_rect.size());
pixmap_.fill(Qt::transparent);
QPainter p(&pixmap_);
p.setRenderHint(QPainter::Antialiasing);
p.setRenderHint(QPainter::HighQualityAntialiasing);
// Background
p.drawPixmap(total_rect.topLeft(), background_cache_);
// Text
p.setPen(palette().color(QPalette::HighlightedText));
p.setFont(font_);
p.drawText(text_rect1, Qt::AlignHCenter, text_);
p.setFont(small_font_);
p.setOpacity(0.65);
p.drawText(text_rect2, Qt::AlignHCenter, small_text_);
p.end();
resize(pixmap_.size());
UpdatePosition();
update();
}
void TrackSliderPopup::UpdatePosition() {
move(pos_.x() - pixmap_.width() / 2,
pos_.y() - pixmap_.height() + kBlurRadius);
}