334 lines
9.7 KiB
C++
334 lines
9.7 KiB
C++
/*
|
|
* SPDX-FileCopyrightText: 2012-2014 Alexander Turkin
|
|
* SPDX-FileCopyrightText: 2014 William Hallatt
|
|
* SPDX-FileCopyrightText: 2015 Jacob Dawid
|
|
* SPDX-FileCopyrightText: 2018 huxingyi
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
/* Original Work Copyright (c) 2012-2014 Alexander Turkin
|
|
Modified 2014 by William Hallatt
|
|
Modified 2015 by Jacob Dawid
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
this software and associated documentation files (the "Software"), to deal in
|
|
the Software without restriction, including without limitation the rights to
|
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
the Software, and to permit persons to whom the Software is furnished to do so,
|
|
subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
// Own includes
|
|
#include "waitingspinnerwidget.h"
|
|
|
|
// Standard includes
|
|
#include <cmath>
|
|
#include <algorithm>
|
|
|
|
// Qt includes
|
|
#include <QPainter>
|
|
#include <QTimer>
|
|
|
|
static bool isAlignCenter(Qt::AlignmentFlag a)
|
|
{
|
|
return a == Qt::AlignmentFlag::AlignVCenter;
|
|
}
|
|
|
|
WaitingSpinnerWidget::WaitingSpinnerWidget(QWidget *parent,
|
|
bool centerOnParent,
|
|
bool disableParentWhenSpinning)
|
|
: QWidget(parent),
|
|
_centerOnParent(centerOnParent),
|
|
_disableParentWhenSpinning(disableParentWhenSpinning) {
|
|
initialize();
|
|
}
|
|
|
|
WaitingSpinnerWidget::WaitingSpinnerWidget(Qt::WindowModality modality,
|
|
QWidget *parent,
|
|
bool centerOnParent,
|
|
bool disableParentWhenSpinning)
|
|
: QWidget(parent, Qt::Dialog | Qt::FramelessWindowHint),
|
|
_centerOnParent(centerOnParent),
|
|
_disableParentWhenSpinning(disableParentWhenSpinning){
|
|
initialize();
|
|
|
|
// We need to set the window modality AFTER we've hidden the
|
|
// widget for the first time since changing this property while
|
|
// the widget is visible has no effect.
|
|
setWindowModality(modality);
|
|
setAttribute(Qt::WA_TranslucentBackground);
|
|
}
|
|
|
|
void WaitingSpinnerWidget::initialize() {
|
|
_color = Qt::black;
|
|
_textColor = Qt::black;
|
|
_roundness = 100.0;
|
|
_minimumTrailOpacity = 3.14159265358979323846;
|
|
_trailFadePercentage = 80.0;
|
|
_revolutionsPerSecond = 1.57079632679489661923;
|
|
_numberOfLines = 20;
|
|
_lineLength = 10;
|
|
_lineWidth = 2;
|
|
_innerRadius = 10;
|
|
_currentCounter = 0;
|
|
_isSpinning = false;
|
|
|
|
_timer = new QTimer(this);
|
|
connect(_timer, SIGNAL(timeout()), this, SLOT(rotate()));
|
|
updateSize();
|
|
updateTimer();
|
|
hide();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::paintEvent(QPaintEvent *) {
|
|
updatePosition();
|
|
QPainter painter(this);
|
|
painter.fillRect(this->rect(), Qt::transparent);
|
|
painter.setRenderHint(QPainter::Antialiasing, true);
|
|
|
|
if (_currentCounter >= _numberOfLines) {
|
|
_currentCounter = 0;
|
|
}
|
|
|
|
painter.setPen(Qt::NoPen);
|
|
for (int i = 0; i < _numberOfLines; ++i) {
|
|
painter.save();
|
|
painter.translate(_innerRadius + _lineLength,
|
|
_innerRadius + _lineLength);
|
|
painter.translate((width() - _imageSize.width()) / 2, 0);
|
|
qreal rotateAngle =
|
|
static_cast<qreal>(360 * i) / static_cast<qreal>(_numberOfLines);
|
|
painter.rotate(rotateAngle);
|
|
painter.translate(_innerRadius, 0);
|
|
int distance =
|
|
lineCountDistanceFromPrimary(i, _currentCounter, _numberOfLines);
|
|
QColor color =
|
|
currentLineColor(distance, _numberOfLines, _trailFadePercentage,
|
|
_minimumTrailOpacity, _color);
|
|
painter.setBrush(color);
|
|
// TODO improve the way rounded rect is painted
|
|
painter.drawRoundedRect(
|
|
QRect(0, -_lineWidth / 2, _lineLength, _lineWidth), _roundness,
|
|
_roundness, Qt::RelativeSize);
|
|
painter.restore();
|
|
}
|
|
|
|
if (!_text.isEmpty()) {
|
|
painter.setPen(QPen(_textColor));
|
|
if (isAlignCenter(alignment())) {
|
|
painter.drawText(QRect(0, 0, width(), height()),
|
|
Qt::AlignVCenter | Qt::AlignHCenter, _text);
|
|
} else {
|
|
painter.drawText(QRect(0, _imageSize.height(), width(), height() - _imageSize.height()),
|
|
Qt::AlignBottom | Qt::AlignHCenter, _text);
|
|
}
|
|
}
|
|
}
|
|
|
|
void WaitingSpinnerWidget::start() {
|
|
updatePosition();
|
|
_isSpinning = true;
|
|
show();
|
|
|
|
if(parentWidget() && _disableParentWhenSpinning) {
|
|
parentWidget()->setEnabled(false);
|
|
}
|
|
|
|
if (!_timer->isActive()) {
|
|
_timer->start();
|
|
_currentCounter = 0;
|
|
}
|
|
}
|
|
|
|
void WaitingSpinnerWidget::stop() {
|
|
_isSpinning = false;
|
|
hide();
|
|
|
|
if(parentWidget() && _disableParentWhenSpinning) {
|
|
parentWidget()->setEnabled(true);
|
|
}
|
|
|
|
if (_timer->isActive()) {
|
|
_timer->stop();
|
|
_currentCounter = 0;
|
|
}
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setNumberOfLines(int lines) {
|
|
_numberOfLines = lines;
|
|
_currentCounter = 0;
|
|
updateTimer();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setLineLength(int length) {
|
|
_lineLength = length;
|
|
updateSize();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setLineWidth(int width) {
|
|
_lineWidth = width;
|
|
updateSize();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setInnerRadius(int radius) {
|
|
_innerRadius = radius;
|
|
updateSize();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setText(const QString& text) {
|
|
_text = text;
|
|
updateSize();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setAlignment(Qt::AlignmentFlag align)
|
|
{
|
|
_alignment = align;
|
|
updateSize();
|
|
}
|
|
|
|
QColor WaitingSpinnerWidget::color() const {
|
|
return _color;
|
|
}
|
|
|
|
QColor WaitingSpinnerWidget::textColor() const {
|
|
return _textColor;
|
|
}
|
|
|
|
QString WaitingSpinnerWidget::text() const {
|
|
return _text;
|
|
}
|
|
|
|
qreal WaitingSpinnerWidget::roundness() const {
|
|
return _roundness;
|
|
}
|
|
|
|
qreal WaitingSpinnerWidget::minimumTrailOpacity() const {
|
|
return _minimumTrailOpacity;
|
|
}
|
|
|
|
qreal WaitingSpinnerWidget::trailFadePercentage() const {
|
|
return _trailFadePercentage;
|
|
}
|
|
|
|
qreal WaitingSpinnerWidget::revolutionsPersSecond() const {
|
|
return _revolutionsPerSecond;
|
|
}
|
|
|
|
int WaitingSpinnerWidget::numberOfLines() const {
|
|
return _numberOfLines;
|
|
}
|
|
|
|
int WaitingSpinnerWidget::lineLength() const {
|
|
return _lineLength;
|
|
}
|
|
|
|
int WaitingSpinnerWidget::lineWidth() const {
|
|
return _lineWidth;
|
|
}
|
|
|
|
int WaitingSpinnerWidget::innerRadius() const {
|
|
return _innerRadius;
|
|
}
|
|
|
|
bool WaitingSpinnerWidget::isSpinning() const {
|
|
return _isSpinning;
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setRoundness(qreal roundness) {
|
|
_roundness = std::max(0.0, std::min(100.0, roundness));
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setColor(QColor color) {
|
|
_color = color;
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setTextColor(QColor color) {
|
|
_textColor = color;
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setRevolutionsPerSecond(qreal revolutionsPerSecond) {
|
|
_revolutionsPerSecond = revolutionsPerSecond;
|
|
updateTimer();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setTrailFadePercentage(qreal trail) {
|
|
_trailFadePercentage = trail;
|
|
}
|
|
|
|
void WaitingSpinnerWidget::setMinimumTrailOpacity(qreal minimumTrailOpacity) {
|
|
_minimumTrailOpacity = minimumTrailOpacity;
|
|
}
|
|
|
|
void WaitingSpinnerWidget::rotate() {
|
|
++_currentCounter;
|
|
if (_currentCounter >= _numberOfLines) {
|
|
_currentCounter = 0;
|
|
}
|
|
update();
|
|
}
|
|
|
|
void WaitingSpinnerWidget::updateSize() {
|
|
int size = (_innerRadius + _lineLength) * 2;
|
|
_imageSize = QSize(size, size);
|
|
if (_text.isEmpty() || isAlignCenter(alignment())) {
|
|
setFixedSize(size, size);
|
|
} else {
|
|
QFontMetrics fm(font());
|
|
QSize textSize = QSize(fm.width(_text), fm.height());
|
|
setFixedSize(std::max(size, textSize.width()), size + size / 4 + textSize.height());
|
|
}
|
|
}
|
|
|
|
void WaitingSpinnerWidget::updateTimer() {
|
|
_timer->setInterval(1000 / (_numberOfLines * _revolutionsPerSecond));
|
|
}
|
|
|
|
void WaitingSpinnerWidget::updatePosition() {
|
|
if (parentWidget() && _centerOnParent) {
|
|
move(parentWidget()->width() / 2 - width() / 2,
|
|
parentWidget()->height() / 2 - height() / 2);
|
|
}
|
|
}
|
|
|
|
int WaitingSpinnerWidget::lineCountDistanceFromPrimary(int current, int primary,
|
|
int totalNrOfLines) {
|
|
int distance = primary - current;
|
|
if (distance < 0) {
|
|
distance += totalNrOfLines;
|
|
}
|
|
return distance;
|
|
}
|
|
|
|
QColor WaitingSpinnerWidget::currentLineColor(int countDistance, int totalNrOfLines,
|
|
qreal trailFadePerc, qreal minOpacity,
|
|
QColor color) {
|
|
if (countDistance == 0) {
|
|
return color;
|
|
}
|
|
const qreal minAlphaF = minOpacity / 100.0;
|
|
int distanceThreshold =
|
|
static_cast<int>(ceil((totalNrOfLines - 1) * trailFadePerc / 100.0));
|
|
if (countDistance > distanceThreshold) {
|
|
color.setAlphaF(minAlphaF);
|
|
} else {
|
|
qreal alphaDiff = color.alphaF() - minAlphaF;
|
|
qreal gradient = alphaDiff / static_cast<qreal>(distanceThreshold + 1);
|
|
qreal resultAlpha = color.alphaF() - gradient * countDistance;
|
|
|
|
// If alpha is out of bounds, clip it.
|
|
resultAlpha = std::min(1.0, std::max(0.0, resultAlpha));
|
|
color.setAlphaF(resultAlpha);
|
|
}
|
|
return color;
|
|
}
|