PyQt窗口焦点

10

我试图在用户单击另一个窗口时使窗口获得焦点。

现在我有两个窗口:窗口A在后面,窗口B在前面。当窗口B出现时,它会禁用窗口A。现在我想要的是,无论何时用户在窗口B外单击,都应将焦点返回给窗口B。

这是窗口B的代码:

class window_b(QtGui.QDialog):
    def __init__(self,parent=None):
        super(window_b, self).__init__(parent)
        window_a.setEnabled(False)
        self.ui = Ui_Form_window_b()
        self.ui.setupUi(self)
        self.setFocusPolicy(QtCore.Qt.StrongFocus)

    def focusOutEvent(self,event):
        self.setFocus(True)
        self.activateWindow()
        self.raise_()
        self.show()

我尝试了setFocusactivateWindow,但它没有将焦点返回到B窗口。

有什么建议吗?


您想将此行为应用于所有其他窗口,还是仅适用于A窗口?如果您想要后者,您可以将B窗口设置为模态窗口http://en.wikipedia.org/wiki/Modal_window。然后,B窗口是A窗口的子窗口,只要B窗口打开,A窗口就无法获得焦点。 - halex
4个回答

14

要让window_b始终保持在最上层,您需要添加窗口标志QtCore.Qt.WindowStaysOnTopHint。在您的__init__中添加调用即可。

self.setWindowFlags(PyQt4.QtCore.Qt.WindowStaysOnTopHint)

我必须补充说,这只是向窗口管理器提供的提示,并不能保证成功。


1
它对我有用,但它没有将焦点还回给窗口B,现在窗口B始终保持在最上层,但没有焦点。 - Uahmed
@user1224233 哦,好的。我读成了“总是在焦点”,但我的大脑理解成了“总是置顶”。抱歉。 - halex
有什么建议可以将焦点放在窗口上吗? - Uahmed

8

self.raise_()跟随一个self.activateWindow()应该就是你要寻找的命令,不过在我的Debian操作系统上似乎存在一些问题。例如,如果我点击一个最大化的窗口,窗口将获得焦点,但它也会消失,看起来像是某种bug。在setTopLevelWindow方法中的顺序将规避这种行为:

#!/usr/bin/env python
#-*- coding:utf-8 -*-

import sip
sip.setapi('QString', 2)
sip.setapi('QVariant', 2)

from PyQt4 import QtGui, QtCore, QtWebKit, QtNetwork

class myWindow(QtGui.QMainWindow):
    def __init__(self, parent=None):
        super(myWindow, self).__init__(parent)

        self.button = QtGui.QPushButton(self)
        self.button.setText("Show Dialog")

        self.dialog = QtGui.QDialog(self)
        self.dialog.setFocusPolicy(QtCore.Qt.StrongFocus)
        self.dialog.installEventFilter(self)

        self.button.clicked.connect(self.dialog.show)

        self.setCentralWidget(self.button)

    def eventFilter(self, obj, event):
        if event.type() == QtCore.QEvent.WindowDeactivate:
            self.setTopLevelWindow()
            self.dialog.close()

            return True

        return False

    def setTopLevelWindow(self):    
        if self.windowState() != QtCore.Qt.WindowMaximized:
            self.showMaximized()
            self.showNormal()

        else:
            self.showNormal()
            self.showMaximized()

        self.raise_()
        self.activateWindow()


if __name__ == "__main__":
    import sys

    app = QtGui.QApplication(sys.argv)
    app.setApplicationName('myWindow')

    main = myWindow()
    main.show()

    sys.exit(app.exec_())

0
class window_b(QtGui.QDialog):
    def __init__(self,parent=None):
        super(window_b, self).__init__(parent)
        window_a.setEnabled(False)
        self.ui = Ui_Form_window_b()
        self.ui.setupUi(self)
        self.setWindowModality(QtCore.Qt.ApplicationModal)
        self.setFocusPolicy(QtCore.Qt.StrongFocus)

    def focusOutEvent(self,event):
        self.setFocus(True)
        self.activateWindow()
        self.raise_()
        self.show()


1
在问题的代码中,我们需要将您发布的那行代码放在哪里才能解决问题? - Ophir Carmi

0

我在那里发现了一个简单的技巧,只需要使用self.showNormal()就可以解决问题,将焦点返回到窗口,代码如下:

class window_b(QtGui.QDialog):
    def __init__(self,parent=None):
        super(window_b, self).__init__(parent)
        window_a.setEnabled(False)
        self.ui = Ui_Form_window_b()
        self.ui.setupUi(self)
        self.setFocusPolicy(QtCore.Qt.StrongFocus)

    def focusOutEvent(self,event):
        self.showNormal()

就是这样! 愉快地编程吧


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接