strawberry-audio-player-win.../src/widgets/lineedit.cpp

268 lines
7.8 KiB
C++

/*
* Strawberry Music Player
* This file was part of Clementine.
* Copyright 2010, David Sansome <me@davidsansome.com>
* Copyright 2018-2021, Jonas Kvinge <jonas@jkvinge.net>
*
* Strawberry 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.
*
* Strawberry 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 Strawberry. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include <QWidget>
#include <QString>
#include <QFont>
#include <QFontMetrics>
#include <QPainter>
#include <QPalette>
#include <QPaintDevice>
#include <QRect>
#include <QSize>
#include <QStyle>
#include <QStyleOption>
#include <QToolButton>
#include <QLineEdit>
#include <QPlainTextEdit>
#include <QSpinBox>
#include <QFlags>
#include <QPaintEvent>
#include <QResizeEvent>
#include "core/iconloader.h"
#include "lineedit.h"
ExtendedEditor::ExtendedEditor(QWidget *widget, int extra_right_padding, bool draw_hint)
: LineEditInterface(widget),
has_clear_button_(true),
clear_button_(new QToolButton(widget)),
reset_button_(new QToolButton(widget)),
extra_right_padding_(extra_right_padding),
draw_hint_(draw_hint),
font_point_size_(widget->font().pointSizeF() - 1),
is_rtl_(false) {
clear_button_->setIcon(IconLoader::Load(QStringLiteral("edit-clear-locationbar-ltr")));
clear_button_->setIconSize(QSize(16, 16));
clear_button_->setCursor(Qt::ArrowCursor);
clear_button_->setStyleSheet(QStringLiteral("QToolButton { border: none; padding: 0px; }"));
clear_button_->setToolTip(QWidget::tr("Clear"));
clear_button_->setFocusPolicy(Qt::NoFocus);
QStyleOption opt;
opt.initFrom(widget);
reset_button_->setIcon(widget->style()->standardIcon(QStyle::SP_DialogResetButton, &opt, widget));
reset_button_->setIconSize(QSize(16, 16));
reset_button_->setCursor(Qt::ArrowCursor);
reset_button_->setStyleSheet(QStringLiteral("QToolButton { border: none; padding: 0px; }"));
reset_button_->setToolTip(QWidget::tr("Reset"));
reset_button_->setFocusPolicy(Qt::NoFocus);
reset_button_->hide();
if (LineEdit *lineedit = qobject_cast<LineEdit*>(widget)) {
QObject::connect(clear_button_, &QToolButton::clicked, lineedit, &LineEdit::set_focus);
QObject::connect(clear_button_, &QToolButton::clicked, lineedit, &LineEdit::clear);
}
else if (TextEdit *textedit = qobject_cast<TextEdit*>(widget)) {
QObject::connect(clear_button_, &QToolButton::clicked, textedit, &TextEdit::set_focus);
QObject::connect(clear_button_, &QToolButton::clicked, textedit, &TextEdit::clear);
}
else if (SpinBox *spinbox = qobject_cast<SpinBox*>(widget)) {
QObject::connect(clear_button_, &QToolButton::clicked, spinbox, &SpinBox::set_focus);
QObject::connect(clear_button_, &QToolButton::clicked, spinbox, &SpinBox::clear);
}
UpdateButtonGeometry();
}
void ExtendedEditor::set_hint(const QString &hint) {
hint_ = hint;
widget_->update();
}
void ExtendedEditor::set_clear_button(const bool visible) {
has_clear_button_ = visible;
clear_button_->setVisible(visible);
UpdateButtonGeometry();
}
bool ExtendedEditor::has_reset_button() const {
return reset_button_->isVisible();
}
void ExtendedEditor::set_reset_button(const bool visible) {
reset_button_->setVisible(visible);
UpdateButtonGeometry();
}
void ExtendedEditor::UpdateButtonGeometry() {
const int frame_width = widget_->style()->pixelMetric(QStyle::PM_DefaultFrameWidth);
const int left = frame_width + 1 + (has_clear_button() ? clear_button_->sizeHint().width() : 0);
const int right = frame_width + 1 + (has_reset_button() ? reset_button_->sizeHint().width() : 0);
widget_->setStyleSheet(QStringLiteral("QLineEdit { padding-left: %1px; padding-right: %2px; }").arg(left).arg(right));
QSize msz = widget_->minimumSizeHint();
widget_->setMinimumSize(msz.width() + (clear_button_->sizeHint().width() + frame_width + 1) * 2 + extra_right_padding_, qMax(msz.height(), clear_button_->sizeHint().height() + frame_width * 2 + 2));
}
void ExtendedEditor::Paint(QPaintDevice *device) {
if (!widget_->hasFocus() && is_empty() && !hint_.isEmpty()) {
clear_button_->hide();
if (draw_hint_) {
QPainter p(device);
QFont font;
font.setBold(false);
font.setPointSizeF(font_point_size_);
QFontMetrics m(font);
const int kBorder = (device->height() - m.height()) / 2;
p.setPen(widget_->palette().color(QPalette::Disabled, QPalette::Text));
p.setFont(font);
QRect r(5, kBorder, device->width() - 10, device->height() - kBorder * 2);
p.drawText(r, Qt::AlignLeft | Qt::AlignVCenter, m.elidedText(hint_, Qt::ElideRight, r.width()));
}
}
else {
clear_button_->setVisible(has_clear_button_);
}
}
void ExtendedEditor::Resize() {
const QSize sz = clear_button_->sizeHint();
const int frame_width = widget_->style()->pixelMetric(QStyle::PM_DefaultFrameWidth);
const int y = (widget_->rect().height() - sz.height()) / 2;
clear_button_->move(frame_width, y);
if (!is_rtl_) {
reset_button_->move(widget_->width() - frame_width - sz.width() - extra_right_padding_, y);
}
else {
reset_button_->move((has_clear_button() ? sz.width() + 4 : 0) + frame_width, y);
}
}
LineEdit::LineEdit(QWidget *parent) : QLineEdit(parent), ExtendedEditor(this) {
QObject::connect(reset_button_, &QToolButton::clicked, this, &LineEdit::Reset);
QObject::connect(this, &LineEdit::textChanged, this, &LineEdit::text_changed);
}
void LineEdit::text_changed(const QString &text) {
if (text.isEmpty()) {
// Consider empty string as LTR
set_rtl(false);
}
else {
// For some reason Qt will detect any text with LTR at the end as LTR, so instead compare only the first character
set_rtl(QString(text.at(0)).isRightToLeft());
}
Resize();
}
void LineEdit::paintEvent(QPaintEvent *e) {
QLineEdit::paintEvent(e);
Paint(this);
}
void LineEdit::resizeEvent(QResizeEvent *e) {
QLineEdit::resizeEvent(e);
Resize();
}
TextEdit::TextEdit(QWidget *parent)
: QPlainTextEdit(parent),
ExtendedEditor(this) {
QObject::connect(reset_button_, &QToolButton::clicked, this, &TextEdit::Reset);
QObject::connect(this, &TextEdit::textChanged, [this]() { viewport()->update(); }); // To clear the hint
}
void TextEdit::paintEvent(QPaintEvent *e) {
QPlainTextEdit::paintEvent(e);
Paint(viewport());
}
void TextEdit::resizeEvent(QResizeEvent *e) {
QPlainTextEdit::resizeEvent(e);
Resize();
}
SpinBox::SpinBox(QWidget *parent)
: QSpinBox(parent),
ExtendedEditor(this, 14, false) {
QObject::connect(reset_button_, &QToolButton::clicked, this, &SpinBox::Reset);
}
void SpinBox::paintEvent(QPaintEvent *e) {
QSpinBox::paintEvent(e);
Paint(this);
}
void SpinBox::resizeEvent(QResizeEvent *e) {
QSpinBox::resizeEvent(e);
Resize();
}
CheckBox::CheckBox(QWidget *parent)
: QCheckBox(parent), ExtendedEditor(this, 14, false) {
QObject::connect(reset_button_, &QToolButton::clicked, this, &CheckBox::Reset);
}
void CheckBox::paintEvent(QPaintEvent *e) {
QCheckBox::paintEvent(e);
Paint(this);
}
void CheckBox::resizeEvent(QResizeEvent *e) {
QCheckBox::resizeEvent(e);
Resize();
}
QString SpinBox::textFromValue(int val) const {
if (val <= 0 && !hint_.isEmpty()) {
return QStringLiteral("-");
}
return QSpinBox::textFromValue(val);
}
RatingBox::RatingBox(QWidget *parent)
: RatingWidget(parent),
ExtendedEditor(this) {
clear_button_->hide();
reset_button_->hide();
}