Qt5 C++入门教程-第13章 自定义控件(Custom Widgets)
zhezhongyun 2025-06-23 23:40 3 浏览
大道至简,在 Qt5 C++入门教程的这一部分,我们将创建一个自定义部件。
大多数工具包通常仅提供诸如按钮、文本部件或滑块等最常见的部件。没有哪个工具包能够提供所有可能的部件。程序员必须自行创建此类部件,这可以借助工具包提供的绘图工具来实现。有两种可行的方式:
1.程序员既可以修改或增强现有的部件。
2.也能够从零开始创建自定义部件。
刻录小部件(自定义控件)
在接下来的示例中,我们要从零开始创建一个自定义的刻录部件。这种部件能在 一些视频编辑软件,DVD刻录软件等应用程序中见到,它是从零开始创建的。
application.h
#pragma once
#include <QWidget>
#include <QSlider>
#include <QFrame>
#include "widget.h"
class Application : public QFrame {
Q_OBJECT
public:
Application(QWidget *parent = nullptr);
private:
QSlider *slider;
Widget *widget;
void initUI();
};
application.cpp
#include <QVBoxLayout>
#include <QHBoxLayout>
#include "application.h"
Application::Application(QWidget *parent)
: QFrame(parent) {
initUI();
}
void Application::initUI() {
const int MAX_VALUE = 750;
slider = new QSlider(Qt::Horizontal , this);
slider->setMaximum(MAX_VALUE);
slider->setGeometry(50, 50, 170, 30);
widget = new Widget(this);
connect(slider, &QSlider::valueChanged, widget, &Widget::setValue);
auto *vbox = new QVBoxLayout(this);
auto *hbox = new QHBoxLayout();
vbox->addStretch(1);
hbox->addWidget(widget, 0);
vbox->addLayout(hbox);
setLayout(vbox);
}
我们在这里构建示例的主窗口。
connect(slider, &QSlider::valueChanged, widget, &Widget::setValue);
当我们移动滑块时,valueChanged 槽就会被执行。
widget.h
#pragma once
#include <QFrame>
class Widget : public QFrame {
Q_OBJECT;
public:
Widget(QWidget *parent = nullptr);
public slots:
void setValue(int);
protected:
void paintEvent(QPaintEvent *e);
void drawWidget(QPainter &qp);
private:
int cur_width;
static constexpr int DISTANCE = 19;
static constexpr int LINE_WIDTH = 5;
static constexpr int DIVISIONS = 10;
static constexpr float FULL_CAPACITY = 700;
static constexpr float MAX_CAPACITY = 750;
};
这是自定义刻录部件的头文件。
private:
int cur_width;
cur_width 变量用于存储滑块的当前值,在绘制自定义部件时会用到这个值。
static constexpr int DISTANCE = 19;
static constexpr int LINE_WIDTH = 5;
static constexpr int DIVISIONS = 10;
static constexpr float FULL_CAPACITY = 700;
static constexpr float MAX_CAPACITY = 750;
这些都是重要的常量。DISTANCE 指的是刻度上的数字与父边框顶部的距离;LINE_WIDTH 是垂直线的宽度;DIVISIONS 是刻度的划分数量;FULL_CAPACITY 是介质的容量,达到该容量后就会进行超刻,超刻状态用红色表示;MAX_CAPACITY 则是介质的最大容量。
widget.cpp
#include <QtGui>
#include "widget.h"
const int PANEL_HEIGHT = 30;
Widget::Widget(QWidget *parent)
: QFrame(parent), cur_width(0) {
setMinimumHeight(PANEL_HEIGHT);
}
void Widget::setValue(int value)
{
cur_width = value;
repaint();
}
void Widget::paintEvent(QPaintEvent *e) {
QPainter qp(this);
drawWidget(qp);
QFrame::paintEvent(e);
}
void Widget::drawWidget(QPainter &qp) {
QString num[] = { "75", "150", "225", "300", "375", "450",
"525", "600", "675" };
int asize = sizeof(num)/sizeof(num[1]);
QColor redColor(255, 175, 175);
QColor yellowColor(255, 255, 184);
int width = size().width();
int step = (int) qRound((double)width / DIVISIONS);
int till = (int) ((width / MAX_CAPACITY) * cur_width);
int full = (int) ((width / MAX_CAPACITY) * FULL_CAPACITY);
if (cur_width >= FULL_CAPACITY) {
qp.setPen(yellowColor);
qp.setBrush(yellowColor);
qp.drawRect(0, 0, full, 30);
qp.setPen(redColor);
qp.setBrush(redColor);
qp.drawRect(full, 0, till-full, PANEL_HEIGHT);
} else if (till > 0) {
qp.setPen(yellowColor);
qp.setBrush(yellowColor);
qp.drawRect(0, 0, till, PANEL_HEIGHT);
}
QColor grayColor(90, 80, 60);
qp.setPen(grayColor);
for (int i=1; i <=asize; i++) {
qp.drawLine(i*step, 0, i*step, LINE_WIDTH);
QFont newFont = font();
newFont.setPointSize(8);
setFont(newFont);
QFontMetrics metrics(font());
int w = metrics.horizontalAdvance(num[i-1]);
qp.drawText(i*step-w/2, DISTANCE, num[i-1]);
}
}
我们在这儿绘制自定义部件,具体绘制内容包括矩形、垂直线和数字。
void Widget::setValue(int value)
{
cur_width = value;
repaint();
}
我们把当前选中的值存储到 cur_width 变量中,然后重新绘制部件。
void Widget::paintEvent(QPaintEvent *e) {
QPainter qp(this);
drawWidget(qp);
QFrame::paintEvent(e);
}
自定义部件的绘制工作被委托给了 drawWidget 方法。
QString num[] = { "75", "150", "225", "300", "375", "450",
"525", "600", "675" };
我们使用这些数字来构建刻录部件的刻度。
int width = size().width();
我们获取部件的宽度,自定义部件的宽度是动态的,用户可以对其进行调整。
int till = (int) ((width / MAX_CAPACITY) * cur_width);
int full = (int) ((width / MAX_CAPACITY) * FULL_CAPACITY);
我们利用 width 变量进行转换,在刻度值和自定义部件的尺寸之间建立联系。
qp.setPen(redColor);
qp.setBrush(redColor);
qp.drawRect(full, 0, till-full, PANEL_HEIGHT);
这三行代码绘制红色矩形,用于指示超刻状态。
qp.drawLine(i*step, 0, i*step, LINE_WIDTH);
我们在这里绘制小垂直线。
QFontMetrics metrics(font());
int w = metrics.horizontalAdvance(num[i-1]);
qp.drawText(i*step-w/2, DISTANCE, num[i-1]);
我们在这里绘制刻度数字。为了精确地定位这些数字,我们需要获取字符串的宽度。
main.cpp
#include <QApplication>
#include "application.h"
int main(int argc, char *argv[]) {
QApplication app(argc, argv);
Application window;
window.resize(370, 200);
window.setWindowTitle("刻录部件");
window.show();
return app.exec();
}
相关推荐
- 自助机网络监控(自助设备监控)
-
今日领导打电话说上次的自助机修好了,今天又不行了,这是第二次打电话了。目前也没有其他人反馈这个机器的问题。于是我就想能不能根据IP去ping这个自助机的网络是否正常,如果不正常,关机了就自动给我发到科...
- C# 打字练习(WinForm)快速打字小游戏,输入正确自动爆炸效果
-
打字游戏窗体初始化:publicintk;PictureBoxpic=newPictureBox();//生成飞机控件privatev...
- 这三款颜值极高的工具完美契合win10风格,逼格满满
-
今天介绍三款开源软件让你的Windows10逼格再飞跃一个等级,而且功能与颜值兼具!ModernFlyouts这是一款Windows10弹出UI的替代品,比原生UI更漂亮、更现代。提示框的位置可以随意...
- VC界面开发组件Xtreme Toolkit Pro全新发布v17.0.0
-
Codejock软件公司的XtremeToolkitPro是屡获殊荣的VC界面库,是MFC开发中最全面界面控件套包,它提供了Windows开发所需要的11种主流的VisualC++MFC控件,...
- 字体|好看的英文字体合集(四)(最好看的十种英文字体)
-
电脑里的字体不在于多而在精,拥有10000个字体不用的话也是浪费,找起来也不容易,只有经常使用的才是好字体。获取方式:私31.潮流赛博朋克金属字体32.潮流创意可扩展英文字体StretchPro3...
- Qt5 C++入门教程-第13章 自定义控件(Custom Widgets)
-
大道至简,在Qt5C++入门教程的这一部分,我们将创建一个自定义部件。大多数工具包通常仅提供诸如按钮、文本部件或滑块等最常见的部件。没有哪个工具包能够提供所有可能的部件。程序员必须自行创建此类部件...
- 艾灸的美容养生功效(艾灸美容养颜)
-
艾灸,属于医外治物理疗法,通过百草之王艾草和13种名贵中草药特制成香条状,点燃以温热刺激患处四周或相关穴位,帮助人体全面温通经络,从而加速皮肤的血液循环,淡化色斑。艾灸疗法-温和灸1.四白穴:四白...
- 你可能不知道的10个CSS3中的隐藏特性
-
CSS3为web设计增添了许多令人惊叹的特性,这其中你经常会用到box-shadow(图层阴影),border-radius(边框圆角),transform(变形)这一类受欢迎的常用特性。但是还有一些...
- 不会才学-4 Tkinter 控件1一Label、Button、Entry
-
一、标签控件(Label)序号可选项&描述1anchor文本或图像在背景内容区的位置,默认为center,可选值为(n,s,w,e,ne,nw,sw,se,center)eswn是东南西北英文...
- 从0到1建立一张评分卡之可视化分析
-
上一篇文章介绍了如何进行数据预处理,接下来介绍如何进行探索性数据分析。探索性数据分析又叫EDA,即ExploratoryDataAnalysis。其实数据预处理也属于EDA的一部分,EDA的目...
- 高温致日本冲绳最大珊瑚群90%白化褪色
-
来源:新华网高温致日本冲绳最大珊瑚群90%白化褪色A+A-分享2016-09-0412:50:53 来源:新华网查看原文<divclass="video-box"data-vid="...
- 学习笔记(十七)|PPT考点总结(ppt.1)
-
分享兴趣,传播快乐,增长见闻,留下美好!亲爱的您,这里是LearningYard学苑。今天小编为大家带来“学习笔记(十七)|PPT考点总结”,欢迎您的访问。Shareyourinterests,...
- 史上最全的浏览器兼容性问题和解决方案
-
微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●浏览器兼容性问题是指因为不同的浏览器对同一段CSS代码解析不同导致页面显示效果不统一...
- 304071纽郎缝包机送料牙架(缝包机送料牙更换)
-
portant;white-space:normal;font-size-adjust:none;font-stretch:normal;background-color:rgb(25...
- 安卓手机微信设置字体大小导致页面错位的解决方案
-
切图网专注网页切图、小程序切图服务,在最近的客户反应中,当安卓手机设置字体大小以后,会导致从微信公众号进入的页面也会字体放大,导致页面的不同程度的错位,要想解决这个问题,可以通过代码解决,ios相对简...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 教程 (33)
- HTML 简介 (35)
- HTML 实例/测验 (32)
- HTML 测验 (32)
- JavaScript 和 HTML DOM 参考手册 (32)
- HTML 拓展阅读 (30)
- HTML常用标签 (29)
- HTML文本框样式 (31)
- HTML滚动条样式 (34)
- HTML5 浏览器支持 (33)
- HTML5 新元素 (33)
- HTML5 WebSocket (30)
- HTML5 代码规范 (32)
- HTML5 标签 (717)
- HTML5 标签 (已废弃) (75)
- HTML5电子书 (32)
- HTML5开发工具 (34)
- HTML5小游戏源码 (34)
- HTML5模板下载 (30)
- HTTP 状态消息 (33)
- HTTP 方法:GET 对比 POST (33)
- 键盘快捷键 (35)
- 标签 (226)
- HTML button formtarget 属性 (30)
- CSS 水平对齐 (Horizontal Align) (30)