dde-dock/widgets/slidercontainer.cpp

327 lines
10 KiB
C++
Raw Normal View History

// Copyright (C) 2022 ~ 2022 Deepin Technology Co., Ltd.
// SPDX-FileCopyrightText: 2018 - 2023 UnionTech Software Technology Co., Ltd.
//
// SPDX-License-Identifier: LGPL-3.0-or-later
#include "slidercontainer.h"
#include <DStyle>
#include <DGuiApplicationHelper>
#include <DPaletteHelper>
#include <QPainterPath>
#include <QMouseEvent>
#include <QGridLayout>
#include <QLabel>
DWIDGET_USE_NAMESPACE
// 用于绘制图标的窗体,此窗体不想让其在外部调用,因此,将其作为一个私有类
class SliderIconWidget : public QWidget
{
public:
explicit SliderIconWidget(QWidget *parent)
: QWidget(parent)
, m_iconSize(QSize(24, 24))
, m_shadowSize(QSize())
, m_isEnter(false)
{}
void updateData(const QIcon &icon, const QSize &iconSize, const QSize &shadowSize)
{
m_icon = icon;
m_iconSize = iconSize;
m_shadowSize = shadowSize;
update();
}
void updateIcon(const QIcon &icon)
{
m_icon = icon;
update();
}
void enterEvent(QEvent *event) override
{
m_isEnter = true;
QWidget::enterEvent(event);
update();
}
void leaveEvent(QEvent *event) override
{
m_isEnter = false;
QWidget::leaveEvent(event);
update();
}
protected:
void paintEvent(QPaintEvent *e) override;
private:
QIcon m_icon;
QSize m_iconSize;
QSize m_shadowSize;
bool m_isEnter;
};
void SliderIconWidget::paintEvent(QPaintEvent *e)
{
if (m_iconSize.isNull() || m_icon.isNull())
return QWidget::paintEvent(e);
QPainter painter(this);
painter.setRenderHint(QPainter::Antialiasing);
if (m_shadowSize.isValid() && !m_shadowSize.isNull() && !m_shadowSize.isEmpty()) {
// 绘制圆形背景
painter.setPen(Qt::NoPen);
// 获取阴影部分背景颜色
QColor backColor = (DGuiApplicationHelper::instance()->themeType() == DGuiApplicationHelper::ColorType::LightType ? Qt::black : Qt::white);
if (DGuiApplicationHelper::instance()->themeType() == DGuiApplicationHelper::ColorType::LightType)
backColor.setAlphaF(m_isEnter ? 0.2 : 0.1);
else
backColor.setAlphaF(m_isEnter ? 0.1 : 0.2);
painter.setBrush(backColor);
int x = (rect().width() - m_shadowSize.width() ) / 2;
int y = (rect().height() - m_shadowSize.height() ) / 2;
painter.drawEllipse(QRect(x, y, m_shadowSize.width(), m_shadowSize.height()));
}
// 绘制图标
QPixmap pixmap = m_icon.pixmap(m_iconSize);
int iconWidth = static_cast<int>(m_iconSize.width() / qApp->devicePixelRatio());
int iconHeight = static_cast<int>(m_iconSize.height() / qApp->devicePixelRatio());
int x = (rect().width() - iconWidth) / 2;
int y = (rect().height() - iconHeight) / 2;
painter.drawPixmap(x, y, iconWidth, iconHeight, pixmap);
}
SliderContainer::SliderContainer(QWidget *parent)
: QWidget(parent)
, m_leftIconWidget(new SliderIconWidget(this))
, m_slider(new QSlider(Qt::Orientation::Horizontal, this))
, m_titleLabel(new QLabel(this))
, m_rightIconWidget(new SliderIconWidget(this))
, m_spaceLeftWidget(new QWidget(this))
, m_spaceRightWidget(new QWidget(this))
{
QVBoxLayout *mainLayout = new QVBoxLayout(this);
mainLayout->setContentsMargins(0, 0, 0, 0);
mainLayout->setSpacing(0);
QHBoxLayout *sliderLayout = new QHBoxLayout(this);
sliderLayout->setContentsMargins(0, 0, 0, 0);
sliderLayout->setSpacing(0);
sliderLayout->addWidget(m_leftIconWidget);
sliderLayout->addWidget(m_spaceLeftWidget);
sliderLayout->addWidget(m_slider);
sliderLayout->addWidget(m_spaceRightWidget);
sliderLayout->addWidget(m_rightIconWidget);
mainLayout->addWidget(m_titleLabel);
mainLayout->addLayout(sliderLayout);
m_titleLabel->setVisible(false);
m_leftIconWidget->installEventFilter(this);
m_slider->installEventFilter(this);
m_rightIconWidget->installEventFilter(this);
connect(m_slider, &QSlider::valueChanged, this, &SliderContainer::sliderValueChanged);
}
SliderContainer::~SliderContainer()
{
}
void SliderContainer::setTitle(const QString &text)
{
m_titleLabel->setText(text);
m_titleLabel->setVisible(!text.isEmpty());
}
QSize SliderContainer::getSuitableSize(const QSize &iconSize, const QSize &bgSize)
{
if (bgSize.isValid() && !bgSize.isNull() && !bgSize.isEmpty())
return bgSize;
return iconSize;
}
void SliderContainer::setIcon(const SliderContainer::IconPosition &iconPosition, const QPixmap &icon,
const QSize &shadowSize, int space)
{
if (icon.isNull()) {
return;
}
switch (iconPosition) {
case IconPosition::LeftIcon: {
m_leftIconWidget->setFixedSize(getSuitableSize(icon.size(), shadowSize));
m_leftIconWidget->updateData(icon, icon.size(), shadowSize);
m_spaceLeftWidget->setFixedWidth(space);
break;
}
case IconPosition::RightIcon: {
m_rightIconWidget->setFixedSize(getSuitableSize(icon.size(), shadowSize));
m_rightIconWidget->updateData(icon, icon.size(), shadowSize);
m_spaceRightWidget->setFixedWidth(space);
break;
}
}
}
void SliderContainer::setIcon(const SliderContainer::IconPosition &iconPosition, const QIcon &icon)
{
switch (iconPosition) {
case IconPosition::LeftIcon: {
m_leftIconWidget->updateIcon(icon);
break;
}
case IconPosition::RightIcon: {
m_rightIconWidget->updateIcon(icon);
break;
}
}
}
void SliderContainer::setPageStep(int step)
{
return m_slider->setPageStep(step);
}
void SliderContainer::setRange(int min, int max)
{
return m_slider->setRange(min, max);
}
bool SliderContainer::eventFilter(QObject *watched, QEvent *event)
{
if (event->type() == QEvent::MouseButtonRelease) {
if (watched == m_leftIconWidget)
Q_EMIT iconClicked(IconPosition::LeftIcon);
else if (watched == m_rightIconWidget)
Q_EMIT iconClicked(IconPosition::RightIcon);
}
return QWidget::eventFilter(watched, event);
}
void SliderContainer::updateSliderValue(int value)
{
m_slider->blockSignals(true);
m_slider->setValue(value);
m_slider->blockSignals(false);
}
int SliderContainer::getSliderValue()
{
return m_slider->value();
}
void SliderContainer::setSliderProxyStyle(QProxyStyle *proxyStyle)
{
proxyStyle->setParent(m_slider);
m_slider->setStyle(proxyStyle);
}
SliderProxyStyle::SliderProxyStyle(StyleType drawSpecial, QStyle *style)
: QProxyStyle(style)
, m_drawSpecial(drawSpecial)
{
}
SliderProxyStyle::~SliderProxyStyle()
{
}
void SliderProxyStyle::drawComplexControl(QStyle::ComplexControl control, const QStyleOptionComplex *option, QPainter *painter, const QWidget *widget) const
{
if (control != ComplexControl::CC_Slider)
return;
// 绘制之前先保存之前的画笔
painter->save();
painter->setRenderHint(QPainter::RenderHint::Antialiasing);
// 获取滑动条和滑块的区域
const QStyleOptionSlider *sliderOption = static_cast<const QStyleOptionSlider *>(option);
QRect rectGroove = subControlRect(CC_Slider, sliderOption, SC_SliderGroove, widget);
QRect rectHandle = subControlRect(CC_Slider, sliderOption, SC_SliderHandle, widget);
if (m_drawSpecial == RoundHandler)
drawRoundSlider(painter, rectGroove, rectHandle, widget);
else
drawNormalSlider(painter, rectGroove, rectHandle, widget);
painter->restore();
}
// 绘制通用的滑动条
void SliderProxyStyle::drawNormalSlider(QPainter *painter, QRect rectGroove, QRect rectHandle, const QWidget *wigdet) const
{
DPalette dpa = DPaletteHelper::instance()->palette(wigdet);
QColor color = dpa.color(DPalette::Highlight);
QColor rightColor(Qt::gray);
if (!wigdet->isEnabled()) {
color.setAlphaF(0.8);
rightColor.setAlphaF(0.8);
}
QPen penLine = QPen(color, 2);
// 绘制上下的竖线一根竖线的宽度是2+4个像素刚好保证中间也是间隔2个像素
for (int i = rectGroove.x(); i < rectGroove.x() + rectGroove.width(); i = i + 4) {
if (i < rectHandle.x())
painter->setPen(penLine);
else
painter->setPen(QPen(rightColor, 2));
painter->drawLine(i, rectGroove.y() + 2, i, rectGroove.y() + rectGroove.height() - 2);
}
// 绘制滚动区域
painter->setBrush(color);
painter->setPen(Qt::NoPen);
QPainterPath path;
path.addRoundedRect(rectHandle, 6, 6);
painter->drawPath(path);
}
// 绘制设计师定义的那种圆形滑块,黑色的滑条
void SliderProxyStyle::drawRoundSlider(QPainter *painter, QRect rectGroove, QRect rectHandle, const QWidget *wigdet) const
{
// 深色背景下,滑块和滑动条白色,浅色背景下,滑块和滑动条黑色
QColor color = wigdet->isEnabled() ? (DGuiApplicationHelper::DarkType == DGuiApplicationHelper::instance()->themeType() ? Qt::white : Qt::black) : Qt::gray;
// 此处中绘制圆形滑动条,需要绘制圆角,圆角大小为其高度的一半
int radius = rectGroove.height() / 2;
// 此处绘制滑条的全长
QColor allBrush = (DGuiApplicationHelper::instance()->themeType() == DGuiApplicationHelper::ColorType::LightType ? Qt::black : Qt::white);
allBrush.setAlphaF( 0.15);
QPainterPath allPathGroove;
allPathGroove.addRoundedRect(rectGroove, radius, radius);
painter->fillPath(allPathGroove, allBrush);
// 已经滑动过的区域
if (DGuiApplicationHelper::instance()->themeType() == DGuiApplicationHelper::ColorType::DarkType) {
color.setAlphaF(0.6);
}
QPainterPath pathGroove;
int handleSize = qMin(rectHandle.width(), rectHandle.height());
rectGroove.setWidth(rectHandle.x() + (rectHandle.width() - handleSize) / 2);
pathGroove.addRoundedRect(rectGroove, radius, radius);
painter->fillPath(pathGroove, color);
// 绘制滑块,因为滑块是正圆形,而它本来的区域是一个长方形区域,因此,需要计算当前
// 区域的正中心区域,将其作为一个正方形区域来绘制圆形滑块
color.setAlphaF(1.0);
int x = rectHandle.x() + (rectHandle.width() - handleSize) / 2;
int y = rectHandle.y() + (rectHandle.height() - handleSize) / 2;
rectHandle.setX(x);
rectHandle.setY(y);
rectHandle.setWidth(handleSize);
rectHandle.setHeight(handleSize);
QPainterPath pathHandle;
pathHandle.addEllipse(rectHandle);
painter->fillPath(pathHandle, color);
}