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

前言

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

前期准备

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

#使用此项目版本不能低于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,构建项目查看是否配置错误。

#include <QApplication>
#include <QWidget>

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

    QWidget window;
    window.show();

    return app.exec();
}

创建WindePloyGui.h

#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

#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代码

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代码

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界面美化代码

/*通用的样式*/
*{
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程序的软件就已经完成了,当然还有一个关于按钮你可以写上你的信息表明你是创作者或是对这个软件的介绍,这里就不一一赘述了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/984179.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

使用OpenCV和MediaPipe库——实现人体姿态检测

目录 准备工作如何在Windows系统中安装OpenCV和MediaPipe库&#xff1f; 安装Python 安装OpenCV 安装MediaPipe 验证安装 代码逻辑 整体代码 效果展示 准备工作如何在Windows系统中安装OpenCV和MediaPipe库&#xff1f; 安装Python 可以通过命令行运行python --versio…

React:Axios

axios可以在浏览器和node.js两边跑&#xff0c;可以向服务端发起ajax请求&#xff0c;也可以在node.js里运行&#xff0c;向远端服务发送http请求 Axios中文文档 | Axios中文网 <!DOCTYPE html> <html lang"en"> <head><meta charset"UT…

数据结构第八节:红黑树(初阶)

【本节要点】 红黑树概念红黑树性质红黑树结点定义红黑树结构红黑树插入操作的分析 一、红黑树的概念与性质 1.1 红黑树的概念 红黑树 &#xff0c;是一种 二叉搜索树 &#xff0c;但 在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是 Red和 Black 。 通过对 任何…

使用 vxe-table 导出 excel,支持带数值、货币、图片等带格式导出

使用 vxe-table 导出 excel&#xff0c;支持带数值、货币、图片等带格式导出&#xff0c;通过官方自动的导出插件 plugin-export-xlsx 实现导出功能 查看官网&#xff1a;https://vxetable.cn gitbub&#xff1a;https://github.com/x-extends/vxe-table gitee&#xff1a;htt…

C# Unity 唐老狮 No.7 模拟面试题

本文章不作任何商业用途 仅作学习与交流 安利唐老狮与其他老师合作的网站,内有大量免费资源和优质付费资源,我入门就是看唐老师的课程 打好坚实的基础非常非常重要: 全部 - 游习堂 - 唐老狮创立的游戏开发在线学习平台 - Powered By EduSoho 如果你发现了文章内特殊的字体格式,…

【够用就好008】开新坑自学esb32烧录进军物联网和嵌入式

见字如面&#xff0c;这里是AKA AIGC创意人竹相左边。 学习使用了三年的AI工具&#xff0c;现在最大的自信就是业余时间可以学习任何自己感兴趣的事&#xff0c;感觉手搓火箭也不是梦。 今天开个新坑&#xff0c;也是逐步探索想要进入的新世界。物联网&#xff08;IoT&#…

51单片机Proteus仿真速成教程——P1-软件与配置+Proteus绘制51单片机最小系统+新建程序模版

前言&#xff1a;本文主要围绕 51 单片机最小系统的绘制及程序模板创建展开。首先介绍了使用 Proteus 绘制 51 单片机最小系统的详细步骤&#xff0c;包括软件安装获取途径、工程创建、器件添加&#xff08;如单片机 AT89C51、晶振、电容、电阻、按键等&#xff09;、外围电路&…

MacOS Big Sur 11 新机安装brew wget python3.12 exo

MacOS Big Sur 11,算是很老的系统了&#xff0c;所以装起来brew有点费劲。 首先安装brew 官网&#xff1a; /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)" 官网加速&#xff1a; 按照官网的方法&#xff0…

C++算法——差分

1.差分 差分与前缀和的核心思想相同&#xff0c;是预处理&#xff0c;可以在暴力枚举的过程中&#xff0c;快速给出查询的结果&#xff0c;从而优化时间复杂度。 是经典的用空间替换时间的做法。 补充&#xff1a;使得最短跳跃距离尽可能长&#xff0c;遇到类似这样的问题时…

【VBA】WPS/PPT设置标题字体

通过VBA&#xff0c;配合左上角的快速访问工具栏&#xff0c;实现自动化调整 选中文本框的 字体位置、大小、颜色。 配合quicker更加便捷 Sub DisableAutoWrapAndFormat()Dim shp As Shape 检查是否选中了一个形状&#xff08;文本框&#xff09;If ActiveWindow.Selection.Typ…

YOLO 各系列结构整理

目录 2016 You Only Look Once: Unified, Real-Time Object Detection(CVPR) 2017 YOLO9000: Better, Faster, Stronger CVPR 2018 YOLOv3:AnIncrementalImprovemen CVPR YOLO V3-SPP 2020 YOLOv4: Optimal Speed and Accuracy of Object Detection 2021 YOLOV5 2021 YOL…

六十天前端强化训练之第十四天之深入理解JavaScript异步编程

欢迎来到编程星辰海的博客讲解 目录 一、异步编程的本质与必要性 1.1 单线程的JavaScript运行时 1.2 阻塞与非阻塞的微观区别 1.3 异步操作的性能代价 二、事件循环机制深度解析 2.1 浏览器环境的事件循环架构 核心组件详解&#xff1a; 2.2 执行顺序实战分析 2.3 Nod…

Git基础之工作原理

基础概念 git本地有三个工作区域&#xff0c;工作目录 Working Directory&#xff0c;暂存区Stage/Index和资源区Repository/Git Directory&#xff0c;如果在加上远程的git仓库就是四个工作区域 四个区域与文件交换的命令之间的关系 WorkSpace&#xff1a;工作区&#xff0c;就…

【计算机网络】计算机网络的性能指标——时延、时延带宽积、往返时延、信道利用率

计算机网络的性能指标 导读 大家好&#xff0c;很高兴又和大家见面啦&#xff01;&#xff01;&#xff01; 在上一篇内容中我们介绍了计算机网络的三个性能指标——速率、带宽和吞吐量。用大白话来说就是&#xff1a;网速、最高网速和实时网速。 相信大家看到这三个词应该就…

测试大语言模型在嵌入式设备部署的可能性-ollama本地部署测试

前言 当今各种大语言模型百花齐放&#xff0c;为了方便使用者更加自由的使用大模型&#xff0c;将大模型变成如同棒球棍一样每个人都能用&#xff0c;并且顺手方便的工具&#xff0c;本地私有化具有重要意义。 本次测试使用ollama完成模型下载&#xff0c;过程简单快捷。 1、进…

【实战篇】【DeepSeek 全攻略:从入门到进阶,再到高级应用】

凌晨三点,某程序员在Stack Overflow上发出灵魂拷问:“为什么我的DeepSeek会把财务报表生成成修仙小说?” 这个魔性的AI工具,今天我们就来场从开机键到改造人类文明的硬核教学。(文末含高危操作集锦,未成年人请在师父陪同下观看) 一、萌新村任务:把你的电脑变成炼丹炉 …

【Linux学习笔记】Linux基本指令分析和权限的概念

【Linux学习笔记】Linux基本指令分析和权限的概念 &#x1f525;个人主页&#xff1a;大白的编程日记 &#x1f525;专栏&#xff1a;Linux学习笔记 文章目录 【Linux学习笔记】Linux基本指令分析和权限的概念前言一. 指令的分析1.1 alias 指令1.2 grep 指令1.3 zip/unzip 指…

Unity DOTS从入门到精通之 自定义Authoring类

文章目录 前言安装 DOTS 包什么是Authoring1. 实体组件2. Authoring类 前言 DOTS&#xff08;面向数据的技术堆栈&#xff09;是一套由 Unity 提供支持的技术&#xff0c;用于提供高性能游戏开发解决方案&#xff0c;特别适合需要处理大量数据的游戏&#xff0c;例如大型开放世…

linux如何判断进程对磁盘是随机写入还是顺序写入?

模拟工具&性能测试工具&#xff1a;fio fio参数说明&#xff1a; filename/dev/sdb1&#xff1a;测试文件名称&#xff0c;通常选择需要测试的盘的data目录。 direct1&#xff1a;是否使用directIO&#xff0c;测试过程绕过OS自带的buffer&#xff0c;使测试磁盘的结果更真…

olmOCR:高效精准的 PDF 文本提取工具

在日常的工作和学习中&#xff0c;是否经常被 PDF 文本提取问题困扰&#xff1f;例如&#xff1a; 想从学术论文 PDF 中提取关键信息&#xff0c;却发现传统 OCR 工具识别不准确或文本格式混乱&#xff1f;需要快速提取商务合同 PDF 中的条款内容&#xff0c;却因工具不给力而…