Search code examples
pythonpyqtpyqt5qmessageboxqinputdialog

How to center QMessageBox and QInputDialog on the screen?


I have this function to center an object in the middle of the screen.

I want to center a QMainWindow, QInputDialog and a QMessageBox.

This is my MessageBox:

def _Warning(self,_type):
        infoBox = QtWidgets.QMessageBox()
        infoBox.setIcon(QtWidgets.QMessageBox.Warning)
        infoBox.setWindowTitle("Warning")
        if (_type=="File"):
            infoBox.setText("The File Already exist in the current Directory")
        else:
            infoBox.setText("The Folder Already exist in the current Directory")

        self.center(infoBox)

        infoBox.exec_()

This is my QInputDialog:

def AddFile_B(self):
        self.cuadro = QInputDialog()
        self.center(self.cuadro)
        text, okPressed = self.cuadro.getText(self, "New File","File Name:", QLineEdit.Normal, "")
        if okPressed and text != '':
            file = File_Node(text)
            verify = self.bonsai_B.addChild(file)
            if (verify == True):
                item = QtWidgets.QListWidgetItem(None,0)
                self.TreeB.addItem(item)
            else:
                del file
                self._Warning("File")

This is My Center Function:

def center(self,object):
    qtRectangle = object.frameGeometry() 
    centerPoint = QtWidgets.QDesktopWidget().availableGeometry().center()
    qtRectangle.moveCenter(centerPoint)
    object.move(qtRectangle.topLeft())

I can only center the QMainWindow.

The logic is to move the object into the topLeft Point (screenWidth/2 - objectWidth/2, screenHeight/2 - objectHeight/2) but I don't know what I am doing wrong.


Solution

  • - QMessageBox

    In the case of QMessageBox this is resized in the exec_() method, so a possible solution is to use QTimer.singleShot() to change the geometry a moment after being displayed.

    from functools import partial
    from PyQt5 import QtCore, QtWidgets
    
    
    def center(window):
        # https://wiki.qt.io/How_to_Center_a_Window_on_the_Screen
    
        window.setGeometry(
            QtWidgets.QStyle.alignedRect(
                QtCore.Qt.LeftToRight,
                QtCore.Qt.AlignCenter,
                window.size(),
                QtWidgets.qApp.desktop().availableGeometry(),
            )
        )
    
    
    class Widget(QtWidgets.QWidget):
        def __init__(self, parent=None):
            super().__init__(parent)
    
            self.btn_warning = QtWidgets.QPushButton(
                "Open QMessageBox", clicked=self.open_qmessagebox
            )
    
            lay = QtWidgets.QVBoxLayout(self)
            lay.addWidget(self.btn_warning)
    
            center(self)
    
        @QtCore.pyqtSlot()
        def open_qmessagebox(self):
            infoBox = QtWidgets.QMessageBox()
            infoBox.setIcon(QtWidgets.QMessageBox.Warning)
            infoBox.setWindowTitle("Warning")
            infoBox.setText("The XXX Already exist in the current Directory")
            wrapper = partial(center, infoBox)
            QtCore.QTimer.singleShot(0, wrapper)
            infoBox.exec_()
    
    
    if __name__ == "__main__":
        import sys
    
        app = QtWidgets.QApplication(sys.argv)
        w = Widget()
        w.show()
        sys.exit(app.exec_())
    

    - QInputDialog

    In the case of QInputDialog, the QInputDialog::getText() method is static, so the "self.cuadro" object is not the window since the window is created within that method. If you pass a parent to getText() then by default it will be centered with respect to that.

    So if the QMainWindow is centered and assuming that QMainWindow is the self then it is not necessary to modify anything.

    If instead the parent is not centered on the screen then there are 2 possible solutions:

    • Do not use the static method and implement the logic through a QInputDialog instance:
    from functools import partial
    from PyQt5 import QtCore, QtWidgets
    
    
    def center(window):
        # https://wiki.qt.io/How_to_Center_a_Window_on_the_Screen
    
        window.setGeometry(
            QtWidgets.QStyle.alignedRect(
                QtCore.Qt.LeftToRight,
                QtCore.Qt.AlignCenter,
                window.size(),
                QtWidgets.qApp.desktop().availableGeometry(),
            )
        )
    
    
    class Widget(QtWidgets.QWidget):
        def __init__(self, parent=None):
            super().__init__(parent)
    
            self.btn_inputdialog = QtWidgets.QPushButton(
                "Open QInputDialog", clicked=self.open_qinputdialog
            )
    
            lay = QtWidgets.QVBoxLayout(self)
            lay.addWidget(self.btn_inputdialog)
    
            center(self)
    
        @QtCore.pyqtSlot()
        def open_qinputdialog(self):
            dialog = QtWidgets.QInputDialog(self)
            dialog.setWindowTitle("New File")
            dialog.setLabelText("File Name:")
            dialog.setTextEchoMode(QtWidgets.QLineEdit.Normal)
            dialog.setTextValue("")
            wrapper = partial(center, dialog)
            QtCore.QTimer.singleShot(0, wrapper)
            text, okPressed = (
                dialog.textValue(),
                dialog.exec_() == QtWidgets.QDialog.Accepted,
            )
            if okPressed and text:
                print(text)
    
    
    if __name__ == "__main__":
        import sys
    
        app = QtWidgets.QApplication(sys.argv)
        w = Widget()
        w.show()
        sys.exit(app.exec_())
    
    • Continue using the static method and get the window using findChildren()
    from functools import partial
    from PyQt5 import QtCore, QtWidgets
    
    
    def center(window):
        # https://wiki.qt.io/How_to_Center_a_Window_on_the_Screen
    
        window.setGeometry(
            QtWidgets.QStyle.alignedRect(
                QtCore.Qt.LeftToRight,
                QtCore.Qt.AlignCenter,
                window.size(),
                QtWidgets.qApp.desktop().availableGeometry(),
            )
        )
    
    
    class Widget(QtWidgets.QWidget):
        def __init__(self, parent=None):
            super().__init__(parent)
    
            self.btn_inputdialog = QtWidgets.QPushButton(
                "Open QInputDialog", clicked=self.open_qinputdialog
            )
    
            lay = QtWidgets.QVBoxLayout(self)
            lay.addWidget(self.btn_inputdialog)
    
            center(self)
    
        @QtCore.pyqtSlot()
        def open_qinputdialog(self):
            parent = self
            dialogs = parent.findChildren(QtWidgets.QInputDialog)
    
            def onTimeout():
                dialog, *_ = set(parent.findChildren(QtWidgets.QInputDialog)) - set(dialogs)
                center(dialog)
    
            QtCore.QTimer.singleShot(0, onTimeout)
            text, okPressed = QtWidgets.QInputDialog.getText(
                parent, "New File", "File Name:", QtWidgets.QLineEdit.Normal, ""
            )
            if okPressed and text:
                print(text)
    
    
    if __name__ == "__main__":
        import sys
    
        app = QtWidgets.QApplication(sys.argv)
        w = Widget()
        w.show()
        sys.exit(app.exec_())