Qt自定义汽车速度表控件

Qt自定义汽车速度表控件,支持自适应缩放,效果如下图:

 speedBoard.h:

#ifndef SPEED_BOARD_H
#define SPEED_BOARD_H

#include <QWidget>

class Speedboard : public QWidget
{
    Q_OBJECT

public: Speedboard(QWidget* parent = nullptr);
    void setValue(qreal ivalue);

private:
    void paintEvent(QPaintEvent* event) override;

private:
    int radius = 80;
    int maxv = 100;
    int minv = 0;
    qreal value;
};
#endif // SPEED_BOARD_H

SpeedBoard.cpp:

#include "speedboard.h"
#include <qpainter.h>
#include <QtMath>
#include <QDateTime>
#include <stdlib.h>

Speedboard::Speedboard(QWidget* parent) :
    QWidget(parent)
{
    value = 0;
}

void Speedboard::setValue(qreal ivalue)
{
    value = ivalue;
    update();
}

void Speedboard::paintEvent(QPaintEvent*)
{
    radius = qMin(width(), height()) / 2 - 20;
    QPainter painter(this);
    painter.setRenderHint(QPainter::Antialiasing);
    QPoint center(width() / 2, height() / 2);
    painter.save();

    /* 绘制背景圆形 */
    painter.translate(center);
    painter.setPen(Qt::NoPen);
    painter.setBrush(QColor(143, 143, 143));
    const int bkRadius = radius + 3;
    painter.drawEllipse(QPoint(), bkRadius, bkRadius);

    /* 绘制长短刻度线 */
    painter.rotate(-135);
    const int longStep = 10;
    const int shortStep = 50;
    for(int i = 0; i <= shortStep; i++)
    {
        /* 前80%绿色刻度线,后20%红色 */
        painter.setPen((i < shortStep * 0.8) ? QColor(32, 243, 32) : QColor(243, 32, 32));
        if(i % (shortStep / longStep) != 0)
        {
            QPoint p1(0, -(radius - 8));
            QPoint p2(0, -radius);
            painter.drawLine(p1, p2);
        }
        else
        {
            QPoint pl(0, -(radius - 12));
            QPoint p2(0, -radius);
            painter.drawLine(pl, p2);
        }
        painter.rotate(270.0 / shortStep);
    }

    /* 绘制表盘名字和它的背景矩形 */
    painter.restore();
    QFontMetrics fm = painter.fontMetrics();
    int tx = center.x();
    int ty = int(center.y() + (radius / 1.4142));
    QString title = u8"仪表盘";
    QSize tsz = fm.size(0, title);
    QRect trect(QPoint(tx - tsz.width() / 2, ty - tsz.height() / 2), tsz);
    painter.setPen(QColor(127, 127, 127));
    painter.setBrush(QColor(173, 163, 163));
    painter.drawRoundedRect(trect.adjusted(-6, -4, 6, 4), 3, 3);
    painter.setPen(QColor(243, 243, 243));
    painter.setBrush(Qt::NoBrush);
    painter.drawText(trect, title);/* 绘制每个长刻度线对应的文字 */
    painter.setPen(QColor(243, 243, 243));
    const int hand = radius - 21;
    for(int i = 0; i <= longStep; i++)
    {
        qreal angle = qDegreesToRadians(-135.0 + i * (270.0 / longStep));
        QString etext = QString::number(minv + i * (maxv - minv) / longStep); /* 注意这里整形运算 */
        QPointF dirVec(qSin(angle), -qCos(angle));
        QPointF ecenter(center.x() + hand * dirVec.x(), center.y() + hand * dirVec.y());
        QSize esz = fm.size(0, etext);
        QRectF erect(QPointF(ecenter.x() - esz.width() / 2, ecenter.y() - esz.height() / 2), esz);
        painter.drawText(erect, etext);
    }

    /* 绘制表盘指针 */
    painter.save();
    QPoint triangle[] = /* 向上的三角形 */
    {
        { -5, 0 },
        { 0, 25 - radius },
        { 5, 0 },
    };
    qreal degree = -135.0 + 270.0 * (value - minv) / (maxv - minv);
    painter.translate(center);
    painter.rotate(degree);
    painter.setPen(Qt::NoPen);
    painter.setBrush(Qt::red);
    painter.drawPolygon(triangle, 3);
    painter.setPen(QPen(QColor(213, 0, 0), 2));
    painter.drawEllipse(QPoint(), 7, 7);
    painter.restore();
}

测试代码main.cpp:

#include "speedboard.h"
#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Speedboard w;
    w.show();

    return a.exec();
}

工程文件speedBoard.pro:

QT       += core gui

greaterThan(QT_MAJOR_VERSION, 4): QT += widgets

TARGET = SpeedBoard
TEMPLATE = app

SOURCES += \
        main.cpp \
    speedboard.cpp

HEADERS += \
    speedboard.h

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
QT是一种跨平台的图形用户界面开发框架,允许开发者创建各种自定义控件。其中的一个自定义控件就是表盘,可以用来展示时间、进度或其他指示器。 QT自定义控件表盘的实现可以通过继承QAbstractSlider类来实现。首先,我们需要定义一个继承自QWidget的类,这个类将成为我们自定义表盘控件的主窗口。在这个类中,我们可以添加各种用于展示表盘的元素,比如指针、刻度、数字等。 接下来,在主窗口类的构造函数中,我们可以设置一些控件的基本属性,比如大小、位置、样式等。然后,可以创建QPainter对象并使用它来绘制各种表盘元素。绘制过程中,我们可以使用QPainter的API来画出指针、刻度线、数字等等。 为了实现表盘的交互功能,我们可以重写主窗口类的一些事件处理函数,比如鼠标按下、移动、释放事件等。通过这些事件处理函数,我们可以改变表盘的显示状态,比如旋转指针、更新数值等等。 除了绘制和交互功能,我们还可以通过添加信号和槽来实现表盘和外部代码的交互。比如,我们可以在表盘被点击时发送一个信号,让外部的槽函数来处理相应的逻辑。 总之,通过QT自定义控件功能,我们可以方便地创建各种定制化的表盘控件。通过继承QAbstractSlider类,定义绘制元素、处理事件和信号、槽函数等,我们可以实现一个完整的、交互性强的表盘控件

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

qq_21239475

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值