''' cpdctrl_gui/init.py ''' from PyQt6.QtGui import QIcon from PyQt6.QtWidgets import QApplication from .ui.main_window import MainWindow from .utility.config import AppConfig from . import resources # This is necessary to set the taskbar icon import ctypes ctypes.windll.shell32.SetCurrentProcessExplicitAppUserModelID(u'n203.cpdctrl-gui.1') def run() -> int: """ Initializes the application and runs it. Returns: int: The exit status code. """ app: QApplication = QApplication(sys.argv) icon_path = resources.get_resource_path("cpdctrl_gui.resources", "icon.png") app.setWindowIcon(QIcon(icon_path)) AppConfig.initialize() window: MainWindow = MainWindow() window.show() exitcode = app.exec() print("Saving configuration") AppConfig.finalize() return sys.exit(exitcode) import sys import traceback import logging from PyQt6 import QtCore, QtWidgets # This is taken from: # https://timlehr.com/2018/01/python-exception-hooks-with-qt-message-box/index.html # basic logger functionality log = logging.getLogger(__name__) handler = logging.StreamHandler(stream=sys.stdout) log.addHandler(handler) def show_exception_box(log_msg): """Checks if a QApplication instance is available and shows a messagebox with the exception message. If unavailable (non-console application), log an additional notice. """ if QtWidgets.QApplication.instance() is not None: errorbox = QtWidgets.QMessageBox() errorbox.setText("Oops. An unexpected error occured:\n{0}".format(log_msg)) errorbox.exec() else: log.debug("No QApplication instance available.") class UncaughtHook(QtCore.QObject): _exception_caught = QtCore.pyqtSignal(object) def __init__(self, *args, **kwargs): super(UncaughtHook, self).__init__(*args, **kwargs) # this registers the exception_hook() function as hook with the Python interpreter sys.excepthook = self.exception_hook # connect signal to execute the message box function always on main thread self._exception_caught.connect(show_exception_box) def exception_hook(self, exc_type, exc_value, exc_traceback): """Function handling uncaught exceptions. It is triggered each time an uncaught exception occurs. """ if issubclass(exc_type, KeyboardInterrupt): # ignore keyboard interrupt to support console applications sys.__excepthook__(exc_type, exc_value, exc_traceback) else: exc_info = (exc_type, exc_value, exc_traceback) log_msg = '\n'.join([''.join(traceback.format_tb(exc_traceback)), '{0}: {1}'.format(exc_type.__name__, exc_value)]) log.critical("Uncaught exception:\n {0}".format(log_msg), exc_info=exc_info) # trigger message box show self._exception_caught.emit(log_msg) # create a global instance of our class to register the hook qt_exception_hook = UncaughtHook()