gpt4 book ai didi

c++ - 如何在注销时优雅地退出 QApplication?

转载 作者:IT王子 更新时间:2023-10-29 01:21:53 25 4
gpt4 key购买 nike

我有一个带有通知区域图标的应用程序,因此主窗口可能会忽略关闭事件。我正在尝试在应用程序退出时保存首选项和历史记录。我还想在应用程序运行时处理注销。虽然该应用程序是跨平台的,但它在 GNU/Linux 中最为方便/有用,因此 Windows 的注销行为无关紧要。这是一个最小的可编译示例,用于测试不同桌面环境/窗口管理器的行为:

// main.cpp
# include <QApplication>
# include <QMainWindow>
# include <QCloseEvent>
# include <QTimer>
# include <iostream>

class M : public QMainWindow
{
Q_OBJECT
public:
~M();
public slots:
void onAboutToQuit();
private:
void closeEvent(QCloseEvent *);
};

M::~M()
{
std::cout << "M::~M()" << std::endl;
}

void M::onAboutToQuit()
{
std::cout << "onAboutToQuit()" << std::endl;
}

void M::closeEvent(QCloseEvent * e)
{
std::cout << "closeEvent()" << std::endl;
hide();
QTimer::singleShot(5000, this, SLOT(show()));
e->ignore();
}

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

M m;
m.setWindowModality(Qt::NonModal);
m.connect(& app, SIGNAL(aboutToQuit()),
SLOT(onAboutToQuit()));
m.show();

return app.exec();
}

# include "main.moc"

// CMakeLists.txt
cmake_minimum_required(VERSION 2.6)
project(closeeventbug)

option(QT5 "Use Qt5" OFF)

if(QT5)
find_package(Qt5Core REQUIRED)
find_package(Qt5Widgets REQUIRED)
else()
find_package(Qt4 REQUIRED)
include_directories(${QT_INCLUDES})
endif()

include_directories(${CMAKE_CURRENT_BINARY_DIR})
set(CMAKE_AUTOMOC ON)

add_executable(closeeventbug main.cpp)

if(QT5)
qt5_use_modules(closeeventbug Core Widgets)
else()
target_link_libraries(closeeventbug ${QT_QTCORE_LIBRARY} ${QT_QTGUI_LIBRARY})
endif()

大多数成熟的桌面环境在注销时会尝试关闭可见窗口。但是由于测试应用程序在关闭时不会退出,因此注销会被中断/取消。如果在窗口不可见时执行注销,它会优雅地退出(就像我想要的那样)。功能较少的桌面环境/窗口管理器会忽略仍在运行的应用程序并退出。它们中的大多数甚至不警告应用程序注销,因此程序输出到的文件中既没有“closeEvent()”,也没有“onAboutToQuit()”,也没有“M::~M()”被重定向。

详细结果(所有非Windows结果均来自Manjaro GNU/Linux):

  1. 成熟的桌面环境,如果可见窗口拒绝退出,则取消注销,优雅地完成不可见的应用程序:

    closeEvent()
    onAboutToQuit()
    M::~M()
    { KDE, XFCE, Mate, GNOME, Cinnamon }

    所有其他人都没有取消退出,但其中一些人试图警告应用程序。

  2. 我不知道为什么 onAboutToQuit() 会出现在日志中,但 M::~M() 不在这种情况下...

    closeEvent()
    onAboutToQuit()
    { Windows7 }

3.

   closeEvent()
{ icewm, E17 }

4.

   <nothing in the log>
{ RazorQt, LxDE, LxQt, i3, budgie, fluxbox, awesome, openbox,
wmii, E16, pekWM, uwm, fvwm, xmonad, spectrwm, windowmaker,
herbstluftwm, WindowsXP }

对于(GCC 4.9.1 或 Clang 3.4.2)和(Qt 4.8.6 或 Qt 5.3.1)的任意组合,行为完全相同。然而,当我在 Xubuntu 上尝试 Qt 4.8 和 Qt 5.2 时,结果有些不同:XFCE 中的 Qt 5.2 没有阻塞——无论主窗口可见性如何,应用程序都能正常完成。但是 Qt 4.8 中存在阻塞(与 Manjaro 中一样)。

我知道可以正确处理注销(不阻塞),因为有几个应用程序可以很好地做到这一点。它们都有通知区域图标,关闭通知区域,但不阻止注销。

  • 基于 Qt:GoldenDict、Transmission-Qt、Kopete;
  • 基于 GTK:Audacious、Pidgin。

我检查了基于 Qt 的源代码,发现在处理 closeEvent 方面没有什么特别之处:

https://github.com/goldendict/goldendict/blob/master/mainwindow.cc

void MainWindow::closeEvent( QCloseEvent * ev )
{
if ( cfg.preferences.enableTrayIcon && cfg.preferences.closeToTray )
{
ev->ignore();
hide();
}
else
{
ev->accept();
qApp->quit();
}
}


https://github.com/bfleischer/transmission/blob/master/qt/mainwin.cc

void
TrMainWindow :: closeEvent( QCloseEvent * event )
{
// if they're using a tray icon, close to the tray
// instead of exiting
if( !myPrefs.getBool( Prefs :: SHOW_TRAY_ICON ) )
event->accept( );
else {
toggleWindows( false );
event->ignore( );
}
}

void
TrMainWindow :: toggleWindows( bool doShow )
{
if( !doShow )
{
hide( );
}
else
{
if ( !isVisible( ) ) show( );
if ( isMinimized( ) ) showNormal( );
//activateWindow( );
raise( );
QApplication::setActiveWindow( this );
}
}

git clone git://anongit.kde.org/kopete

void KopeteWindow::closeEvent ( QCloseEvent *e )
{
// if we are not ok to exit on close and we are not shutting down then just do what needs to be done if a
// window is closed.
KopeteApplication *app = static_cast<KopeteApplication *> ( kapp );
if ( !shouldExitOnClose() && !app->isShuttingDown() && !app->sessionSaving() ) {
// BEGIN of code borrowed from KMainWindow::closeEvent
// Save settings if auto-save is enabled, and settings have changed
if ( settingsDirty() && autoSaveSettings() )
saveAutoSaveSettings();

if ( queryClose() ) {
e->accept();
}
// END of code borrowed from KMainWindow::closeEvent
kDebug ( 14000 ) << "just closing because we have a system tray icon";
}
else
{
kDebug ( 14000 ) << "delegating to KXmlGuiWindow::closeEvent()";
KXmlGuiWindow::closeEvent ( e );
}
}

所以问题:

  1. 即使主窗口可见,如何确保我的应用程序不会阻止注销?

  2. 如何确保在尽可能多的桌面环境/窗口管理器中注销时调用 onAboutToQuit() 和 ~M()?

我怀疑应该听一些系统信号,但我不知道到底是哪个...

最佳答案

QApplication 有 signal与操作系统 session 操作相关 - 您可以轻松处理它。有关详细信息,请查看 Qt 文档 Session Management页面

关于c++ - 如何在注销时优雅地退出 QApplication?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25108520/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com