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

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类
利用文件的后缀名,可以做过滤文件的作用(这里不展示)
在这里插入图片描述

相关内容

热门资讯

埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
应用未安装解决办法 平板应用未... ---IT小技术,每天Get一个小技能!一、前言描述苹果IPad2居然不能安装怎么办?与此IPad不...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
猫咪吃了塑料袋怎么办 猫咪误食... 你知道吗?塑料袋放久了会长猫哦!要说猫咪对塑料袋的喜爱程度完完全全可以媲美纸箱家里只要一有塑料袋的响...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...
埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
应用未安装解决办法 平板应用未... ---IT小技术,每天Get一个小技能!一、前言描述苹果IPad2居然不能安装怎么办?与此IPad不...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
猫咪吃了塑料袋怎么办 猫咪误食... 你知道吗?塑料袋放久了会长猫哦!要说猫咪对塑料袋的喜爱程度完完全全可以媲美纸箱家里只要一有塑料袋的响...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...