Qt从入门到入土(八) -打包Qt程序

前言

当你写完一个有趣的Qt程序时,想发给朋友或者家人,但是他们的电脑又没有安装Qt,那么如何直接在他们电脑上运行又不需要安装Qt呢?本篇文章会告诉你答案,本文详细的介绍了界面设计和功能实现。读完本文你不仅可以学会打包部署Qt程序而且还可以制作一个Qt打包程序的软件。

前期准备

创建Qt项目,CMakeLists.txt文件配置Qt。

cpp 复制代码
#使用此项目版本不能低于3.20
cmake_minimum_required(VERSION 3.20)
#项目名称 版本 语言    
project(WindePloyGui VERSION 0.1 LANGUAGES CXX)   
#查找QT包
find_package(Qt6 REQUIRED COMPONENTS Widgets)
    
#设置变量
set(PROJECT_SOURCES
   main.cpp
   )    
#添加可执行文件
add_executable(WindePloyGui ${PROJECT_SOURCES})   
#添加Qt链接库
target_link_libraries(WindePloyGui Qt6::Widgets)  

创建main.cpp,构建项目查看是否配置错误。

cpp 复制代码
#include <QApplication>
#include <QWidget>

int main(int argc, char* argv[]) 
{
    QApplication app(argc, argv);

    QWidget window;
    window.show();

    return app.exec();
}

创建WindePloyGui.h

cpp 复制代码
#ifndef PACKAGEQTGUI_H
#define PACKAGEQTGUI_H

#include <QWidget>

QT_BEGIN_NAMESPACE
namespace Ui {
class PackageQtGui;
}
QT_END_NAMESPACE

class PackageQtGui : public QWidget
{
    Q_OBJECT

public:
    PackageQtGui(QWidget *parent = nullptr);
    ~PackageQtGui();

public:
    void initUI();
private:
    Ui::PackageQtGui *ui;
};
#endif // PACKAGEQTGUI_H

创建WindePloyGui.cpp

cpp 复制代码
#include "packageqtgui.h"
#include "./ui_packageqtgui.h"

PackageQtGui::PackageQtGui(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::PackageQtGui)
{
    ui->setupUi(this);
}

PackageQtGui::~PackageQtGui()
{
    delete ui;
}

界面初始化

本次界面使用到了QFormLayout,QComboBox等控件。

**QFormLayout:QFormLayout**是 Qt 提供的一种布局工具,专门用来生成类似表单的界面。它允许您将控件以"标签-控件"对的形式排列,通常用于输入表单、设置界面等场景。

**QComboBox:**创建下拉列表,用户可以从列表中选择一个选项。

this->m_qtVersionListCmb= new QComboBox;

this->m_qtKitsListCmb = new QComboBox;

//给QComboBox添加项

//this->m_qtVersionListCmb->addItems({"one","two","three"});

auto flayout = new QFormLayout;

flayout->addRow("选择Qt版本",this->m_qtVersionListCmb);

flayout->addRow("选择Qt编译套件",this->m_qtKitsListCmb);

当前初始化UI代码

cpp 复制代码
void PackageQtGui::initUI()
{
    this->m_qtVersionListCmb= new QComboBox;
    this->m_qtKitsListCmb = new QComboBox;

    auto selectExeBtn = new QPushButton("选择exe文件");
    auto createBtn = new QPushButton("生成");
    auto testBtn = new QPushButton("测试");
    auto aboutBtn = new QPushButton("关于");

    auto flayout = new QFormLayout;
    flayout->addRow("选择Qt版本",this->m_qtVersionListCmb);
    flayout->addRow("选择Qt编译套件",this->m_qtKitsListCmb);

    auto hlayout = new QHBoxLayout;
    hlayout->addWidget(testBtn);
    hlayout->addWidget(aboutBtn);

    auto vlayout = new QVBoxLayout(this);
    vlayout->addLayout(flayout);
    vlayout->addWidget(selectExeBtn);
    vlayout->addWidget(createBtn);
    vlayout->addLayout(hlayout);

}

当前效果

调整控件大小

使用setSizePolicy函数来调整控件大小。

selectExeBtn->setSizePolicy(QSizePolicy::Policy::Expanding,QSizePolicy::Policy::Expanding);

美化QComboBox中的drop-down

有时直接给子控件设置图片可能会因为QComboBox::drop-down 的显示区域可能太小,导致图片无法完全显示。可以通过设置 widthheight 来调整显示区域。

QComboBox::drop-down{

image:url(':/Recourses/Icons/drop-down.png');

subcontrol-origin: padding;

subcontrol-position: right center;

width: 20px; /* 调整箭头宽度 */

height: 20px; /* 调整箭头高度 */

}

QComboBox::drop-down:hover{

image:url(':/Recourses/Icons/drop-down-hover.png');

}

QComboBox::drop-down:checked{

image:url(':/Recourses/Icons/drop-down-on.png');

}

QComboBox::drop-down:checked:hover{

image:url(':/Recourses/Icons/drop-down-on-hover.png');

}

初始化UI代码

cpp 复制代码
void PackageQtGui::initUI()
{
    this->setWindowTitle("QT程序打包工具");
    this->setFixedSize(640,510);
    this->m_qtVersionListCmb= new QComboBox;
    this->m_qtKitsListCmb = new QComboBox;

    auto selectExeBtn = new QPushButton("选择exe文件");
    auto createBtn = new QPushButton("生成");
    auto testBtn = new QPushButton("测试");
    auto aboutBtn = new QPushButton("关于");

    selectExeBtn->setObjectName("selectExeBtn");
    selectExeBtn->setSizePolicy(QSizePolicy::Policy::Expanding,QSizePolicy::Policy::Expanding);

    auto flayout = new QFormLayout;
    flayout->addRow("选择 Qt 版本",this->m_qtVersionListCmb);
    flayout->addRow("选择编译套件",this->m_qtKitsListCmb);

    auto hlayout = new QHBoxLayout;
    hlayout->addWidget(testBtn);
    hlayout->addWidget(aboutBtn);

    auto vlayout = new QVBoxLayout(this);
    vlayout->addLayout(flayout);
    vlayout->addWidget(selectExeBtn);
    vlayout->addWidget(createBtn);
    vlayout->addLayout(hlayout);
}

css界面美化代码

cpp 复制代码
/*通用的样式*/
*{
background-color:white;
font:normal 15px "楷体";
}

QPushButton,QComboBox{
border:1px solid rgb(213,213,213);
border-radius:8px;
}

QComboBox:hover{
border-color:rgb(100,100,100);
}

QComboBox::drop-down{
image:url(':/Recourses/Icons/drop-down.png');
subcontrol-origin: padding;
subcontrol-position: right center;
width: 20px; /* 调整箭头宽度 */
height: 20px; /* 调整箭头高度 */
}

QComboBox::drop-down:hover{
image:url(':/Recourses/Icons/drop-down-hover.png');
}

QComboBox::drop-down:checked{
image:url(':/Recourses/Icons/drop-down-on.png');
}

QComboBox::drop-down:checked:hover{
image:url(':/Recourses/Icons/drop-down-on-hover.png');
}

QPushButton:hover{
background:rgb(220,220,220);
}

/*设置特定样式*/
QPushButton#selectExeBtn{
border-width:2px;
font:italic 20px "楷体";
color:rgb(174,174,174);
}

功能实现

首先,找到Qt版本编译的套件

构建QtEnvSearch类用于获取系统的应用程序可写路径,因为不同的电脑路径不同但可以通过QStandardPaths来获取到当前电脑的路径

QString path = QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation);

查找Qt版本的文件夹,获取一个目录中所有以 "Qt" 开头的子目录的信息列表

QFileInfoList info_list = dir.entryInfoList({"Qt*"},QDir::Dirs);

筛选出不为空的文件夹

for(auto& info:info_list)

{

//筛选出系统路径中名称带有Qt的文件夹

// 检查文件夹是否为空

QDir subDir(info.absoluteFilePath());

if (!subDir.isEmpty()) {

qDebug() << "Non-empty Qt version folder:" << info.absoluteFilePath();

infoList.append(info.absoluteFilePath());

}

}

再次筛选Qt目录中Qt版本号文件夹,并排除空文件夹

QStringList resultList;

for(auto &info:infoList)

{

//QDir::Dirs 表示只获取子目录。

//QDir::NoDotAndDotDot 是一个常用的过滤器,用于排除 . 和 .. 这两个特殊目录。

QFileInfoList sub_info_list = QDir(info).entryInfoList(QDir::Dirs | QDir::NoDotAndDotDot);

for(auto& i:sub_info_list)

{

QDir subDir(i.absoluteFilePath());

if(!subDir.isEmpty())

{

resultList.append(i.absoluteFilePath());

}

}

}

获取到所选版本目录中的Qt编译套件

QStringList QtEnvSearch::m_qtKitsLists()

{

if(m_currentVersion.isEmpty())

{

m_currentVersion = m_versionMap.firstKey();

}

auto path = m_versionMap.value(m_currentVersion);

QDir dir(path);

if(!dir.exists())

{

qWarning()<<path<<"not exists!";

return QStringList{};

}

return dir.entryList(QDir::Dirs|QDir::NoDotAndDotDot);

}

把获取到的版本号和套件列表添加到QComboBox的item中

this->m_qtVersionListCmb->addItems({m_envSearch.m_qtVersionLists()});

this->m_qtKitsListCmb->addItems({m_envSearch.m_qtKitsLists()});

当选择的Qt版本改变时需要更新所对应的选择编译套件

connect(this->m_qtVersionListCmb,&QComboBox::currentTextChanged,[=](const QString& ver)

{

this->m_envSearch.setCurrentVersion(ver);

this->m_qtKitsListCmb->clear();

this->m_qtKitsListCmb->addItems(this->m_envSearch.m_qtKitsLists());

});

当选择编译套件发生改变的时候需要更新对应的Qt版本

connect(this->m_qtKitsListCmb,&QComboBox::currentTextChanged,&this->m_envSearch,&QtEnvSearch::setCurrentKits);

当前效果

接下来要实现selectExeBtn,点击该按钮即可打开文件(这里使用QFileDialog),并将按钮文本修改成获取到的文件名。

connect(selectExeBtn,&QPushButton::clicked,[=](){

auto filename = QFileDialog::getOpenFileName(this,"选择exe文件",QStandardPaths::writableLocation(QStandardPaths::DesktopLocation)

,"exe(*.exe);;all(*.*)");

if(filename.isEmpty())

return;

selectExeBtn->setText(QFileInfo(filename).fileName());

qDebug()<<filename;

});

之后,点击生成按钮完成部署。如何完成部署呢?首先进入到套件目录,根据快捷方式找到指定的bin目录。

bool QtEnvSearch::generate()

{

//进入套件目录

QDir dir(m_versionMap.value(m_currentVersion));

if(!dir.cd(m_currentKits))

{

qWarning()<<"cd"<<m_currentKits<<"failed~";

return false;

}

//找到一个快捷方式,并进入指向的目标

auto all_entry = dir.entryInfoList(QDir::NoDotAndDotDot | QDir::AllEntries);

if(all_entry.isEmpty())

{

qWarning()<<"生成失败,套件无效";

return false;

}

//qDebug()<<all_entry;

//获取qt的bin目录

auto qtBin = QFileInfo(all_entry.first().symLinkTarget()).path();

//qDebug()<<qtBin;

return true;

}

使用system函数进行部署

// 传入qtbin目录中的windeployqt.exe和待部署的exe文件的路径

system(QString(qtBin + "/windeployqt.exe " + m_exeFile).toStdString().data());

使用QProcess::startDetached进行部署

bool success = QProcess::startDetached(QString(qtBin + "/windeployqt.exe"),{m_exeFile});

点击测试按钮打开程序

// 同样使用QProcess::startDetached来打开EXE程序

return QProcess::startDetached(m_exeFile);

至此,打包Qt程序的软件就已经完成了,当然还有一个关于按钮你可以写上你的信息表明你是创作者或是对这个软件的介绍,这里就不一一赘述了。

相关推荐
mengzhi啊12 分钟前
QTreeWidget的右键菜单,展开和折叠,或者其他操作
qt
不爱学习的小枫25 分钟前
scala的集合
开发语言·scala
梦醒沉醉25 分钟前
Scala的初步使用
开发语言·后端·scala
小白学大数据30 分钟前
Fuel 爬虫:Scala 中的图片数据采集与分析
开发语言·爬虫·scala
十年编程老舅36 分钟前
用Qt手搓AI助手,挑战24小时开发DeepSeek Assistant!
qt·计算机毕设·c++项目·qt项目·deepseek·计算机毕设项目
贩卖纯净水.1 小时前
《React 属性与状态江湖:从验证到表单受控的实战探险》
开发语言·前端·javascript·react.js
JouJz1 小时前
Java基础系列:深入解析反射机制与代理模式及避坑指南
java·开发语言·代理模式
香菇滑稽之谈1 小时前
代理模式的C++实现示例
c++·设计模式·系统安全·代理模式
Dante7981 小时前
【数据结构】二叉搜索树、平衡搜索树、红黑树
数据结构·c++·算法
march of Time1 小时前
go注册rpc接口
qt·rpc·golang