ui控件库——PushButton系列

作者:小 琛
欢迎转载,请标明出处

PushButton在ui库中的使用

QT中提供了QPushButton,基于它我们可以实现button的绝大多数功能,同时配合样式修改,即可实现自定义控件库PushButton的效果。

一般的Button在产品中可能有的问题

在实际的产品中,针对button往往需要考虑以下几个问题

  • 多语言问题。通常情况下,我们button在确定好大小后,将不再改变。而button里的内容,由于多语言则可能导致各种显示问题。(例如:中文确定退出,俄文Определение выхода )
  • 文本更迭问题。很多场景下,button中的文字不是固定的,随着业务场景而改变。
  • 按钮的状态问题。按钮通常有:normal、hover、click状态,需要对他们进行定义。

设计一个PushButton控件

现在我们使用QT,来设计一个自己的PushButton

  • 实现控件类,继承自QPushButton
  • 通常来说,我们自定义的QPushButton,高度是固定的(若不固定,在构建时添加高度参数即可),那么在构造函数时,接口参数可定义为:width、text、parent
  • 重写event事件,实现normal、hover、click三种状态;重写paintEvent事件,实现控件样式的绘制
  • 当外界使用时,直接将该类当作普通QPushButton使用,同时可以使用connect进行绑定等

example

#include <QPushButton>
class MyPushButton : public QPushButton {
    Q_OBJECT
public:
    MyPushButton (int width, const QString& text, QWidget* parent = nullptr);
    ~MyPushButton () = default;

    void updateText(const QString& text); // 提供给构建后,修改文本的接口

protected:
    bool event(QEvent* event);
    void paintEvent(QPaintEvent* event);

private:
    QColor getBackgroundColor() const;
    QColor getTextColor() const;
    QFont getTextFont() const;
    void adjustText();
    QString elideText(const QString& text, int width, const QFont& font);
    QColor opacityToAlphaColor(int red, int green, int blue, double opacity)private:
    int textMargin_{8};
    QString text_;
    QString displayText_;
    ButtonStatus status_{ButtonStatus::Status_Normal};
};
#include "MyPushButton.h"
#include <QPaintEvent>
#include <QSvgRenderer>
#include <QStyleOption>
#include <QFontMetrics>
MyPushButton::MyPushButton (int width, const QString& text, QWidget* parent) : QPushButton(parent), text_(text) {
		setFlat(true); // 取消边框和背景色
		setFocusPolicy(Qt::ClickFocus); // 设置点击获取焦点
		const int fixedHeight = 32;
		setFixedSize(width, fixedHeight);
		adjustText(); // 格式化输入文本
	}

	void MyPushButton::updateText(const QString& text) {
		text_ = text;
		adjustText();
		update();
	}

	bool MyPushButton::event(QEvent* event) {
		switch (event->type()) {
		case QEvent::Enter:
			status_ = ButtonStatus::Status_Hover;
			update();
			break;
		case QEvent::Leave:
			status_ = ButtonStatus::Status_Normal;
			update();
			break;
		case QEvent::MouseButtonPress:
			status_ = ButtonStatus::Status_Press;
			update();
			break;
		case QEvent::MouseButtonRelease:
			if (geometry().contains(parentWidget()->mapFromGlobal(QCursor::pos()))) {
				status_ = ButtonStatus::Status_Hover;
			}
			else {
				status_ = ButtonStatus::Status_Normal;
			}

			update();
			break;
		default:
			break;
		}

		return QPushButton::event(event);
	}

	void MyPushButton::paintEvent(QPaintEvent* event) {
		QPainter painter(this);
		painter.save();
		painter.setRenderHint(QPainter::Antialiasing, true);
		painter.setPen(Qt::NoPen);

		const auto radiusValue{ 3 };
		painter.setBrush(QBrush(getBackgroundColor()));
		painter.drawRoundedRect(rect(), radiusValue, radiusValue);

		const auto textX = textMargin_;
		const auto textY = 0;
		const auto textWidth = width() - textMargin_ * 2;
		const auto textHeight = height();

		painter.setFont(getTextFont());
		painter.setPen(QPen(getTextColor(), 1));
		painter.drawText(QRect{ textX, textY, textWidth, textHeight }, Qt::AlignCenter, displayText_);
		painter.restore();
	}

	QColor MyPushButton::getBackgroundColor() const {
		QColor color = Qt::transparent;
		switch (status_) {
		case ButtonStatus::Status_Normal:
			color = opacityToAlphaColor(2, 191, 255, 1);
			break;
		case ButtonStatus::Status_Hover:
			color = return opacityToAlphaColor(0, 209, 255, 1);
			break;
		case ButtonStatus::Status_Press:
			color = opacityToAlphaColor(2, 191, 255, 1);
			break;
		default:
			break;
		}

		return color;
	}

	QColor MyPushButton::getTextColor() const {
		QColor color = Qt::transparent;
		switch (status_) {
		case ButtonStatus::Status_Normal:
			color = opacityToAlphaColor(0, 0, 0, 1);
			break;
		case ButtonStatus::Status_Hover:
			color = opacityToAlphaColor(0, 0, 0, 1);
			break;
		case ButtonStatus::Status_Press:
			color = opacityToAlphaColor(0, 0, 0, 0.6);
			break;
		default:
			break;
		}

		return color;
	}

	QFont MyPushButton::getTextFont() const { return NemuUiLib::NemuFont::getFont14(); }

	void MyPushButton::adjustText() {
		const auto textWidth = width() - textMargin_ * 2;
		displayText_ = TextUtil::elideText(text_, textWidth, getTextFont());

		if (text_ != displayText_) {
			setToolTip(text_);
		}
	}
   
   QString MyPushButton::elideText(const QString& text, int width, const QFont& font) {
        QFontMetrics fontMetrics(font);

        if (fontMetrics.width(text) > width) {
            return QFontMetrics(font).elidedText(text, Qt::ElideRight, width);
        }

        return text;
    }

   QColor MyPushButton::opacityToAlphaColor(int red, int green, int blue, double opacity) {
    	const int alpha = round(opacity * 255);
    	return QColor(red, green, blue, alpha);
   }
    auto testButton = new MyPushButton7(240, tr("testing"), this);
    connect(testButton , &QPushButton::clicked, this, [=](){});

代码解析:

  • 类继承QPushButton,可自定义构建宽度和文本,同时将输入的文本结合输入的宽度以及间隔,实现文本格式化-超出范围显示…
  • opacityToAlphaColor是根据R G B A计算得到的QColor
  • 核心是重写了event和paintEvent,event实现了对当前状态的更迭,paintEvent结合当前状态,重绘button
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值