c++学习27qt(三)Qt自定义控件封装,事件处理,绘画和文件读写
创始人
2024-03-30 12:19:19
0

Qt自定义控件封装

步骤

  1. 添加类 Qt-设计师界面类 .h .cpp .ui
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    此时就添加成功
  2. 选择一个QWidget控件,添加到主界面,右击选择提升为
    . 在这里插入图片描述
    此时,我们要将1步骤的Qt界面设计类添加进去,选择提升
    如果

在这里插入图片描述
则说明替换成功

  1. 设置自定义控件封装内组件之间的联动效果,在.cpp文件中设置
smallWiget::smallWiget(QWidget *parent) :QWidget(parent),ui(new Ui::smallWiget)
{ui->setupUi(this);//QSpinBox移动 QSlider跟着移动void(QSpinBox:: *spinSignal)(int) = &QSpinBox::valueChanged;connect(ui->spinBox,spinSignal,ui->horizontalSlider,&QSlider::setValue);//QSlider移动 QSpinBox 跟着改变connect(ui->horizontalSlider,&QSlider::valueChanged,ui->spinBox,&QSpinBox::setValue);
}
  1. 设置一些接口,供外部去使用 :(在.h文件中声明,在.cpp文件中实现)
//smallWiget.h文件中
class smallWiget : public QWidget
{Q_OBJECTpublic:explicit smallWiget(QWidget *parent = 0);//定义的方法:void setData(int val);int getData();//end~smallWiget();private:Ui::smallWiget *ui;
};
//smallWiget.cpp文件
void smallWiget::setData(int val){ui->spinBox->setValue(val);
}int smallWiget::getData(){return ui->spinBox->value();
}
  1. 外部的简单使用
 //获取按钮connect(ui->btn_getValue,&QPushButton::clicked,[=](){qDebug()<<"当前的数值为:"<< ui->widget->getData();});//设置一半按钮connect(ui->btn_setHalf,&QPushButton::clicked,[=](){ui->widget->setData(50);});
  1. 效果展示
    请添加图片描述

Qt中的事件处理

捕获QLabel中的鼠标事件

如果我们想捕获QLabel的鼠标事件,而控件本身的事件无法满足需求,就需要我们重写其方法来满足需求
这样我们就需要创建一个类去继承QLabel,再在界面中添加一个空的容器,使其提升,再在其头文件中声明需要的方法,在.cpp文件中对具体需求做实现

鼠标的一些事件:

  • 鼠标进入enterEvent
  • 鼠标离开leaveEvent
  • 鼠标按下mousePressEvent
  • 鼠标释放mouseReleaseEvent
  • 鼠标移动mouseMoveEvent
  • 设置鼠标追踪 this->setMouseTracking(true); 默认是true

定时器

定时器事件

 void timerEvent(QTimerEvent *e);

可以通过e获取定时器的id

e->timerId();

通过定时器的id,我们可以在开启不同的定时器,做出不同的实现
如:

void Widget::timerEvent(QTimerEvent *e){//使用QTimerEvent的timeId()方法拿到具体的开启定时器的值if(e->timerId()==id1){static int num =1;//加上static 使其就初始化一次,如果不加static那么num会在timerEvent()方法运行时一直走,导致显示的一直是1ui->label->setText(QString::number(num++));}else{static int num2 =10;ui->label_2->setText(QString::number(num2+=10));}
}

当我们有多个定时器时,我们可以使用定时器的类QTimer去创建定时器,使代码更便于管理
如:

//利用定时器的类QTimer实现定时功能QTimer * timer = new QTimer(this);//启动定时器timer->start(500);connect(timer,&QTimer::timeout,[=](){//定时功能static int num3 = 9;ui->label_3->setText(QString::number(num3+=9));});
//关闭定时器(这里没有用槽和信号,尽量用事件或方法去包裹使用,再不济加个判断
timer->stop();

注意:两个实现定时器的方法没有优劣之分,按照需求选择合适的就可

事件分发器

  1. bool event(QEvent *e)
  2. e->type()所有Qt中的事件
  3. 事件分发器可以做事件的拦截操作,如果用户进行了拦截,在对应事件里返回true代表自己处理事件

事件过滤器

使用事件过滤器可以做更高级拦截

使用事件过滤器有两步:

  1. 给对应的控件安装事件过滤器
ui->label->installEventFilter(this);
  1. 重写eventFilter
//.h
bool eventFilter(QObject *watched, QEvent *event);//.cpp
bool Widget::eventFilter(QObject *watched, QEvent *event){//参数1 对比控件  参数2对比事件//对比控件if(obj==ui->label){//对比事件if(event->type()==QEvent::MouseButtonPress){//处理方式return true;   //如果返回true 代表用户自己处理 }}    return QWidget::eventFilter(obj,event);//默认处理    
}

绘画

基本绘画

  1. 绘画方法
//绘画事件void paintEvent(QPaintEvent *event);
  1. 声明画家
 //声明一个画家的对象 在窗口设备中绘制图片QPainter painter(this);
  1. 设置画笔
//指定画笔QPen pen(QColor(255,0,0));//设置笔宽pen.setWidth(3);//设置笔风格pen.setStyle(Qt::DotLine);//给画家拿个笔painter.setPen(pen);
  1. 设置画刷
//画刷 用来填充画家画出的封闭图形QBrush brush(Qt::cyan);//设置画刷风格brush.setStyle(Qt::Dense6Pattern);//给画家设置画刷painter.setBrush(brush);
  1. 绘制一些基础图形
 //画线painter.drawLine(QPoint(0,0),QPoint(100,100));//画圆(椭圆)当rx=ry时,画出的就是一个圆painter.drawEllipse(QPoint(100,100),50,50);//画矩形 QRect()四参数说明,前面两个代表坐标,后面两个代表宽高painter.drawRect(QRect(20,20,50,50));//画文字 前面的参数代表一个矩形范围,后面的参数是目标文字painter.drawText(QRect(10,200,120,100),"好好学习,天天向上");
  1. 效果展示
    在这里插入图片描述

高级绘画设置

1.设置抗锯齿

   painter.drawEllipse(QPoint(100,100),50,50);
//   //设置抗锯齿 效率更低,但图片更清晰painter.setRenderHint(QPainter::Antialiasing);painter.drawEllipse(QPoint(250,100),50,50);

在这里插入图片描述
2.移动画家的位置

 painter.drawRect(QRect(20,20,50,50));//移动画家的位置,画家初始位置为(0,0) 下面将(100,0)当作(0,0)去使用painter.translate(QPoint(100,0));painter.save();//保存画家的位置painter.drawRect(QRect(20,20,50,50));//移动画家的位置,画家初始位置为(0,0) 下面将(100,0)当作(0,0)去使用painter.translate(QPoint(100,0));//将画家的位置还原到上一次painter.restore();painter.drawRect(QRect(20,20,50,50));

在这里插入图片描述
3. 利用画家绘制资源图片

#include "widget.h"
#include "ui_widget.h"
#include
Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);posX =10;//点击按钮实现图片移动connect(ui->pushButton,&QPushButton::clicked,[=](){//手动调用paintEvent函数,利用updateposX+=20;update();});
}
void Widget::paintEvent(QPaintEvent *event){QPainter painter(this);//如果超过屏幕的范围,将posX设置回来if(posX>this->width()){posX =10;}//利用画家绘制图片painter.drawPixmap(posX,100,QPixmap(":/image/page.png"));
}Widget::~Widget()
{delete ui;
}

效果图
请添加图片描述

绘图设备

绘图设备有一下几种:

QPixmap , QBitmap , QImage , QPicture , QWidget

QPixmap

QPixmap做绘图设备
在不同平台做了显示的优化

QPixmap简单使用案例

//QPixmap做绘图设备//在不同平台做了显示的优化QPixmap pix(300,300);pix.fill(Qt::white); //设置QPixmap的填充色,默认是黑色QPainter painter(&pix);painter.setPen(QPen(Qt::blue));painter.drawEllipse(QPoint(150,150),100,100);//保存pix.save("E:\\pix.png");

在这里插入图片描述

QImage

QImage做绘图设备
可以对像素进行访问

 QImage img(300,300,QImage::Format_RGB32);img.fill(Qt::white);QPainter painter(&img);painter.setPen(QPen(Qt::green));painter.drawEllipse(QPoint(150,150),100,100);img.save("E:\\img.png");

QPicture

QPicture做绘图设备
记录和重现QPainter的各条命令
这个就比较有意思了,我们可以设置生成文件的格式,并且如果不是常规默认的后缀名,那么就只有我们的Qt程序可以去打开它了

#include "widget.h"
#include "ui_widget.h"
#include
#include
Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);//QPicture做绘图设备//记录和重现QPainter的各条命令QPicture pic;QPainter painter;painter.begin(&pic);//开始执行命令,且绑定绘图设备painter.setPen(QPen(Qt::blue));painter.drawEllipse(QPoint(150,150),100,100);painter.end();//结束执行命令pic.save("E:\\pic.wjl");//保存//这个格式后缀可以随便写,只要不是常规默认,是没办法直接打开预览的
}void Widget::paintEvent(QPaintEvent *event){//重现QPicture 的绘图命令QPainter painter(this);QPicture pic;pic.load("E:\\pic.wjl");painter.drawPicture(0,0,pic);}
Widget::~Widget()
{delete ui;
}

在这里插入图片描述
QBitmap 色深限定为1 ,只有黑白色

QFile 文件读写

规律

和c++的方式一样,只是更换了类,还是需要路径作为文件的参数,且需要指定的打开方式,且其有默认的编码格式为utf-8

案例

#include "widget.h"
#include "ui_widget.h"
#include
#include
#include
Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);// 实现效果: 点击按钮 选择文件connect(ui->pushButton,&QPushButton::clicked,[=](){//拿到文件路径QString filePath =  QFileDialog::getOpenFileName(this,"选择文件","E:\\weixintool\\微信web开发者工具") ;//将文件路径设置到lineEdit中if(filePath.isEmpty()){QMessageBox::warning(this,"打开失败","文件路径为空");return;}ui->lineEdit->setText(filePath);//设置编码格式QTextCodec *code = QTextCodec::codecForName("gbk");//默认的编码格式是utf-8//将文件中的内容放入到textEdit中QFile file(filePath);//设置目标文件//设置打开方式file.open(QIODevice::ReadOnly);//以只读的方式打开QByteArray arr;//读取全部的方式://arr = file.readAll();//按行读取while(!file.atEnd()){arr += file.readLine();}//显示arr数据ui->textEdit->setText(arr);//使用设置好的编码格式
//      ui->textEdit->setText(code->toUnicode(arr));//用完后要关闭文件对象file.close();//写文件file.open(QIODevice::Append);//利用追加的方式写入file.write("这是添加的条目");file.close();});
}Widget::~Widget()
{delete ui;
}

请添加图片描述

QFileInfo类

直接使用:

//文件信息类QFileInfo info(filePath);qDebug()<<"大小"<

如果使用与日期有关的,需要导入QDate类
利用文件的后缀名,可以做过滤文件的作用(这里不展示)
在这里插入图片描述

相关内容

热门资讯

落户政策居然考虑放开! 怎么,我能落户北京了? 大家好,我是孙少睡,这是我的第467篇楼市评论。 很多人的第一反应肯定是有没...
股市必读:ST泉为股东因涉嫌违... 截至2025年12月26日收盘,ST泉为(300716)报收于9.96元,下跌0.8%,换手率0.9...
日本公布犯罪白皮书:2024年... 日本法务省19日公布的2025年版犯罪白皮书显示,日本2024年刑事犯罪案件数量明显上升,其中性犯罪...
中央广电总台副台长王晓真,黑龙... 据央视新闻报道,12月28日,中央广播电视总台《2026年春节联欢晚会》分会场发布。黑龙江哈尔滨、浙...
聚焦全国财政工作会议丨明年财政... (央视财经《经济信息联播》)明年是“十五五”规划的开局之年,财政政策将聚焦哪些关键领域精准发力? ...
原创 中... 12月26日,中国对美国实施了一次重磅反制,针对美国政府前不久批准的111亿美元对台军售,中方决定出...
徐杰11分王少杰遭驱逐 张宁缺... [搜狐体育战报]北京时间12月28日消息,2025-26赛季CBA常规赛继续第7轮角逐。王少杰第三节...
《今日说法》主持人李晓东买茶叶... 12月28日,《今日说法》栏目主持人李晓东发布视频称,此前“被骗1000元买茶叶”事件迎来新进展:该...
3-0领先终于能休息了!莫德里... 在意甲第17轮的一场焦点战中,AC米兰迎战维罗纳。比赛进行到第70分钟时,AC米兰在3-0领先的情况...