C++ Qt開發:自定義Dialog對話框組件

来源:https://www.cnblogs.com/LyShark/archive/2023/12/17/17903549.html
-Advertisement-
Play Games

Qt 是一個跨平臺C++圖形界面開發庫,利用Qt可以快速開發跨平臺窗體應用程式,在Qt中我們可以通過拖拽的方式將不同組件放到指定的位置,實現圖形化開發極大的方便了開發效率,本章將重點介紹自定義`Dialog`組件的常用方法及靈活運用。自定義對話框需要解決的問題是,如何讓父窗體與子窗體進行數據交換,要... ...


Qt 是一個跨平臺C++圖形界面開發庫,利用Qt可以快速開發跨平臺窗體應用程式,在Qt中我們可以通過拖拽的方式將不同組件放到指定的位置,實現圖形化開發極大的方便了開發效率,本章將重點介紹自定義Dialog組件的常用方法及靈活運用。

在之前的文章中筆者已經為大家展示了預設Dialog組件的使用方法,雖然內置組件支持對數據的輸入,但有時候我們需要一次性輸入多個數據,此時如果之使用預設模態對話框似乎不太夠用,此時我們需要自己創建一個自定義對話框,需要說明的是此類對話框也是一種窗體,所以可以在其上面放置任何通用組件,以實現更多複雜的開發需求。

自定義對話框需要解決的問題是,如何讓父窗體與子窗體進行數據交換,要實現數據的交換有兩種方式,第一種方式是通過動態載入模態對話框,當用戶點擊確定後通過GetValue()來拿到數據,而第二種方式則是通過發送信號的方式將數據投遞給父窗體,這兩種方式都可以,讀者可根據自身需求來選擇不同的通信方式。

1.1 使用模態對話框傳值

首先我們需要創建一個自定義對話框,在Qt中創建對話框很容易,具體創建流程如下所示:

  • 選擇項目 -> AddNew -> QT -> Qt設計師界面類 -> 選擇DialogWithoutButtons -> 命名為Dialog保存

此時直接點擊下一步按鈕,並選中Forms/dialog.ui界面編輯菜單,在編輯欄中我們分別增加一個LineEdit編輯框,以及兩個PushButton按鈕組件,將第一個組件命名為BtnOk將第二個組件命名為BtnCancel,界面如下所示;

當做完頁面佈局後,其次我們還需要在Dialog.ui組件上增加兩個信號,分別是點擊關閉,並將信號關聯到兩個槽函數上,其信號應該寫成如下圖所示。

如上圖,accept()QDialog 類的一個公共槽函數。調用這個槽函數會觸發對話框的接受(accept)操作,通常用於模擬用戶點擊對話框的“確定”按鈕。同樣的reject() 也是 QDialog 類的一個公共槽函數。調用這個槽函數會觸發對話框的拒絕(reject)操作,通常用於模擬用戶點擊對話框的“取消”按鈕。

接著我們點開模態對話框的dialog.cpp對話框類,其類內需要定義兩個成員函數,它們的功能如下:

  • 第一個 GetValue() 用來獲取當前編輯框內的數據並將數據返回給父窗體。
  • 第二個 SetValue() 用來接收傳入的參數,並將此參數設置到自身窗體中的編輯框內。
#include "dialog.h"
#include "ui_dialog.h"

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

// 用於MainWindow獲取編輯框中的數據
QString Dialog::GetValue()
{
    return ui->lineEdit->text();
}

// 用於設置當前編輯框中的數據為MainWindow
void Dialog::SetValue(QString x)
{
    ui->lineEdit->setText(x);
}

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

接著我們來看一下MainWindow函數中是如何接收參數的,對於主窗體來說,當用戶點擊on_pushButton_clicked()按鈕時,我們需要動態將自己創建的Dialog載入,讀取出主窗體編輯框內的值並設置到子窗體內,當用戶按下QDialog::Accepted時則是獲取子窗體內的值,此時通過調用ptr->GetValue()子窗體的成員函數來返回一個字元串,並將其設置到父窗體的編輯框內,主函數代碼如下所示;

// 首先要包含Dialog對話框類
#include "dialog.h"

#include <iostream>
#include <QDialog>

MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    ui->lineEdit->setEnabled(false);
    ui->lineEdit->setText("hello lyshark");
}

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

// 按鈕點擊後執行
void MainWindow::on_pushButton_clicked()
{
    // 創建模態對話框
    Dialog *ptr = new Dialog(this);                                 // 創建一個對話框
    Qt::WindowFlags flags = ptr->windowFlags();                     // 需要獲取返回值
    ptr->setWindowFlags(flags | Qt::MSWindowsFixedSizeDialogHint);  // 設置對話框固定大小

    // 讀取MainWindows參數並設置到Dialog
    QString item = ui->lineEdit->text();
    ptr->SetValue(item);

    int ref = ptr->exec();             // 以模態方式顯示對話框
    if (ref==QDialog::Accepted)        // OK鍵被按下,對話框關閉
    {
        // 當BtnOk被按下時,則設置對話框中的數據
        QString the_value = ptr->GetValue();
        std::cout << "value = " << the_value.toStdString().data() << std::endl;
        ui->lineEdit->setText(the_value);
    }

    // 刪除釋放對話框句柄
    delete ptr;
}

至此就實現了參數的子窗體傳遞到父窗體,如下圖所示;

2.1 使用信號傳值

對於信號傳值,我們需要在dialog.h頭文件中增加sendText()信號,以及on_pushButton_clicked()槽函數的聲明部分,如下所示;

// 定義信號(信號只需聲明無需實現)
signals:
    void sendText(QString str);
private slots:
    void on_pushButton_clicked();

而在dialog.cpp實現部分,我們首先需要將子窗體中的按鈕組件綁定到onBtnClick()槽函數上面,當需要發送數據時直接通過調用emit sendText觸發信號,並攜帶子窗體中send_data的數據;

#include "dialog.h"
#include "ui_dialog.h"

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

    // 連接pushButton到onBtnClick上
    connect(ui->pushButton, SIGNAL(clicked()), this, SLOT(onBtnClick()));
}

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

// 發送信號到MainWindow
void Dialog::on_pushButton_clicked()
{
    QString send_data = ui->lineEdit->text();
    emit sendText(send_data);
}

接著是在mainwindow.h頭文件定義中,新增槽函數receiveMsg()函數用來接收信號的傳值。

private slots:
    // 定義槽函數
    void receiveMsg(QString str);
    void on_pushButton_clicked();

mainwindow.cpp實現部分,接收到信號後的槽函數receiveMsg其內部可以直接將參數設置到父類視窗的lineEdit組件上,而當on_pushButton_clicked按鈕被點擊是,我們只需要載入自己的子窗體,並Connect鏈接槽函數receiveMsg上面,當做完這一切之後,再通過subwindow->show()讓子窗體顯示出來。

#include "mainwindow.h"
#include "ui_mainwindow.h"

#include "dialog.h"
#include <QDialog>

MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent),ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    ui->lineEdit->setEnabled(false);
}

// 接收信號並設置到LineEdit上
void MainWindow::receiveMsg(QString str)
{
    ui->lineEdit->setText(str);
}

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

void MainWindow::on_pushButton_clicked()
{
    Dialog *subwindow = new Dialog(this);
    // 當收到sendText信號時使用receiveMsg槽函數處理
    connect(subwindow, SIGNAL(sendText(QString)), this, SLOT(receiveMsg(QString)));
    subwindow->show();
}

當然,此類對話框是非模態的,讀者可以拖動父對話框,而由於是信號控制,所以當發送參數到父窗體後,子窗體並不會立即關閉,如下圖所示;

完整案例下載

文章出處:https://www.cnblogs.com/LyShark/p/17903549.html
本博客所有文章除特別聲明外,均採用 BY-NC-SA 許可協議。轉載請註明出處!
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、JetPack Compose、組合函數與註解和文本修改 1、JetPack Compose:Jetpack Compose 是由 Google 推出的用於構建 Android 用戶界面的現代化工具包。它是一個聲明式的 UI 工具包,用於簡化 Android 應用程式的用戶界面設計和開發。Jet ...
  • Chrome 擴展項目使用前端 html,css,js 基礎技術開發,一大痛點就是改動代碼後的擴展更新問題。仔細想想想,目前前端項目開發已經有 HMR 熱重載技術,在開發 Web 頁面時可以實時查看效果,極大的提升了開發體驗。那麼,能否把這種極致的體驗帶到 Chrome 擴展開發中來呢?經過多番折騰... ...
  • 三、python數據、變數、註釋 1.數據 什麼是數據?在python中像數字,漢字,英文,圖片,音頻都是數據。目前瞭解就可以了 打開命令視窗,win+r 輸入cmd 打開python 2.變數和標識符 什麼是變數? 當我們編寫代碼的時候,我們會找一塊地方存放數據,而數據存放一個的時候還特別好找,當 ...
  • 流暢的orm讓我發現我抵觸的是mybatis而不是java 背景介紹 開發.net 也快10年了,到第三年的時候我已經漸漸瓶頸了,於是我在網上找各種資料但是大部分c#資料全是皮毛資料,稍微深一點點就再講表達式expression,感覺完全沒有那個深度,但是同時期的java講解的都是基本原理,和框架思 ...
  • 2.1、創建Maven工程 2.1.1、創建空項目 2.1.2、設置項目名稱和路徑 2.1.3、設置項目sdk 2.1.4、項目初始狀態 註意:需要關閉項目再重新打開,才能看到SpringBoot-Part文件夾 2.1.5、配置maven 2.1.6、創建module 右擊SpringBoot-P ...
  • 相信大家這幾天被董宇輝事件刷屏了,透徹的說就是幾個關鍵詞: 功高蓋主 去董宇輝 卸磨殺驢 飛鳥盡良弓藏 大家都知道新東方的轉型成功(打著助農的旗號),董宇輝發揮了巨大作用,很多人也是通過董宇輝才認識了東方甄選,讓東方甄選一躍成為抖音前三名的帶貨直播間。就在直播間蒸蒸日上的時候,金主擔心“東方甄選== ...
  • SpringCloud 文章推薦:Eureka:Spring Cloud服務註冊與發現組件(非常詳細) (biancheng.net) 概述 Spring Cloud 是一個服務治理平臺,是若幹個框架的集合,提供了全套的分散式系統解決方案。包含了:服務註冊與發現、配置中心、服務網關、智能路由、負載均 ...
  • 在使用pip安裝Python軟體包時,有時會遇到與 SSL/TLS 相關的問題。一種常見情況是在使用VPN時出現以下錯誤信息 ValueError: check_hostname requires server_hostname: ValueError: check_hostname require ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...