方法一、
窗口右上角的X按鍵會導致其在不給出任何提示的情況下直接退出,
當點擊右上角的x按鍵時,會觸發Qt中的一個事件處理函數:void QWidget::closeEvent ( QCloseEvent * e ),默認情況該函數無任何提示性消息,如果需提示則需要重寫該函數
void Dialog::closeEvent ( QCloseEvent * e ) { if( QMessageBox::question(this, tr("Quit"), tr("Are you sure to quit this application?"), QMessageBox::Yes, QMessageBox::No ) == QMessageBox::Yes){ e->accept();//不會將事件傳遞給組件的父組件
qDebug()<<"ok"; } else e->ignore(); }
說明:
- 事件的ignore函數表示忽略事件,將其傳到父組件進行處理
- 事件的accept函數表示接受事件,組件自己進行處理,不會將事件傳遞給父組件
- 但是在這個closeEvent函數中,這兩個函數尤其不一樣的理解:對於窗口關閉QCloseEvent事件,
- 調用accept()意味着 Qt 會停止事件的傳播,窗口關閉;
- 調用ignore()則意味着事件繼續傳播,即阻止窗口關閉。
此外如果需要直接退出某個應用程序,可以直接調用Qt中的一個全局指針:qApp ,全局指針qApp指向全局的QApplication對象。
//在需要停止程序運行的位置直接加入如下代碼
#include <QtGui/QApplication> ... qApp->quit();
說明:
- qApp是一個全局的函數,可以在Qt說明文檔中直接找到,不是main函數定義的QApplication對象(QApplication對象也可以退出,使用 app->exit(0);或者 app->quit();具體可以參考【1】)
- qApp->quit()關閉的是整個應用程序,不只是窗口,
- 一般情況下關閉窗口應用程序會直接退出,如果要求關閉窗口時,應用程序不退出,則需要設置QApplication的屬性:
QApplication::setQuitOnLastWindowClosed(false);
方法二、
QWidget的close槽函數是像widget發送QCloseEvent,如果widget未設置Qt::WA_DeleteOnClose標志的話,將隱藏widget,並不會銷毀相關資源。如果設置了該標志,那么會再發送destroy信號,銷毀相關資源。
(多說一句:QWindow的close槽是調用destroy來銷毀窗口資源的。不過我們一般不直接使用QWindow。)
Qt幫助文檔中的介紹: 調用close()方法后首先它會向widget發送一個關閉事件(QCloseEvent)。如果widget接受了關閉事件(QCloseEvent),窗口將會隱藏(實際上調用hide())。
如果widget不接受關閉事件,那么窗口將什么也不做。默認情況下widget會接受關閉事件,我們可以重寫QCloseEvent事件,可以選擇接受或者不接受。 如果widget設置了Qt::WA_DeleteOnClose屬性,widget將會被釋放。不管widget是否可見,關閉事件都會傳遞給widget。即接收到QCloseEvent事件后,
除了調用hide()方法將窗口隱藏,同時會調用deleteLater()方法將窗口釋放掉,不會再占用資源。 所以說調用close()並不一定就會將窗口對象銷毀。而只有設置了 Qt::WA_DeleteOnClose屬性才會刪除銷毀。如果這個屬性沒有設置,close()的作用和hide(),
setvisible(false)一樣,只會隱藏窗口對象而已,並不會銷毀該對象。
這樣的話我們可以給QLabel(拿它當一個例子)設置Qt::WA_DeleteOnClose
QLabel *label = new QLabel; label->setAttribute(Qt::WA_DeleteOnClose);
之后用connect使用label的destroyed信號綁定一個槽函數即可
connect(label, &QLabel::destroyed, this, [=]() { QString file_path = QFileDialog::getSaveFileName(this, "save file", "tmp.png", "(*.png);;all files(*.*)"); if(!file_path.isEmpty())
...... });