PyQt4 свернуть в лоток - PullRequest
       31

PyQt4 свернуть в лоток

22 голосов
/ 17 апреля 2009

Есть ли способ свернуть трей в PyQt4? Я уже работал с классом QSystemTrayIcon, но теперь я хотел бы свернуть или «спрятать» окно моего приложения и показать только значок в трее.

Кто-нибудь делал это? Любое направление будет оценено.

Использование Python 2.5.4 и PyQt4 в Windows XP Pro

Ответы [ 7 ]

29 голосов
/ 17 апреля 2009

Это довольно просто, если вспомнить, что на самом деле нет способа свести к минимуму системный трей .

Вместо этого вы притворяетесь, делая это:

  1. Поймайте событие свертывания в вашем окне
  2. В обработчике минимизации событий создайте и отобразите QSystemTrayIcon
  3. Также в обработчике событий свернуть вызовите hide () или setVisible (false) для вашего окна
  4. Поймать щелчок / двойной щелчок / пункт меню на значке в системном трее
  5. В обработчике событий значка в системном трее вызовите show () или setVisible (true) в своем окне и, при необходимости, скройте значок в панели задач.
12 голосов
/ 17 апреля 2009

Код помогает, поэтому вот что я написал для приложения, за исключением closeEvent вместо события минимизации.

Примечания:

"closeEvent (event)" - переопределенное событие Qt, поэтому его нужно поместить в класс, реализующий окно, которое вы хотите скрыть.

"okayToClose ()" - это функция, которую вы, возможно, захотите реализовать (или логический флаг, который вы можете захотеть сохранить), поскольку иногда вы действительно хотите выйти из приложения, а не сворачивать в systray.

Существует также пример того, как снова показать () ваше окно.

def __init__(self):
  traySignal = "activated(QSystemTrayIcon::ActivationReason)"
  QtCore.QObject.connect(self.trayIcon, QtCore.SIGNAL(traySignal), self.__icon_activated)

def closeEvent(self, event):
  if self.okayToClose(): 
    #user asked for exit
    self.trayIcon.hide()
    event.accept()
  else:
    #"minimize"
    self.hide()
    self.trayIcon.show() #thanks @mojo
    event.ignore()

def __icon_activated(self, reason):
  if reason == QtGui.QSystemTrayIcon.DoubleClick:
    self.show()
7 голосов
/ 15 ноября 2009

Просто добавлю к примеру Криса:

Очень важно , что вы используете нотацию Qt при объявлении сигнала, т.е.

правильный

self.connect(self.icon, SIGNAL("activated(QSystemTrayIcon::ActivationReason)"), self.iconClicked)

а не PyQt

неверно и не будет работать:

self.connect(self.icon, SIGNAL("activated(QSystemTrayIcon.ActivationReason)"), self.iconClicked)

Обратите внимание на :: в строке сигнала. Мне потребовалось около трех часов, чтобы понять.

4 голосов
/ 13 января 2011

Вот рабочий код .. Спасибо Матзе за Решающее значение , СИГНАЛ занял у меня больше часов любопытства ... но занимался другими делами. так что за #! момент: -)

def create_sys_tray(self):
    self.sysTray = QtGui.QSystemTrayIcon(self)
    self.sysTray.setIcon( QtGui.QIcon('../images/corp/blip_32.png') )
    self.sysTray.setVisible(True)
    self.connect(self.sysTray, QtCore.SIGNAL("activated(QSystemTrayIcon::ActivationReason)"), self.on_sys_tray_activated)

    self.sysTrayMenu = QtGui.QMenu(self)
    act = self.sysTrayMenu.addAction("FOO")

def on_sys_tray_activated(self, reason):
    print "reason-=" , reason
3 голосов
/ 04 мая 2015

Это был ответ vzades, но он был отклонен по ряду причин. Он делает то же самое, что и их код, но также подчиняется событию минимизации (и работает без синтаксических ошибок / пропущенных значков).

import sys
from PyQt4 import QtGui, QtCore


class Example(QtGui.QWidget):
    def __init__(self):
        super(Example, self).__init__()
        self.initUI()

    def initUI(self):
        style = self.style()

        # Set the window and tray icon to something
        icon = style.standardIcon(QtGui.QStyle.SP_MediaSeekForward)
        self.tray_icon = QtGui.QSystemTrayIcon()
        self.tray_icon.setIcon(QtGui.QIcon(icon))
        self.setWindowIcon(QtGui.QIcon(icon))

        # Restore the window when the tray icon is double clicked.
        self.tray_icon.activated.connect(self.restore_window)

    def event(self, event):
        if (event.type() == QtCore.QEvent.WindowStateChange and 
                self.isMinimized()):
            # The window is already minimized at this point.  AFAIK,
            # there is no hook stop a minimize event. Instead,
            # removing the Qt.Tool flag should remove the window
            # from the taskbar.
            self.setWindowFlags(self.windowFlags() & ~QtCore.Qt.Tool)
            self.tray_icon.show()
            return True
        else:
            return super(Example, self).event(event)

    def closeEvent(self, event):
        reply = QtGui.QMessageBox.question(
            self,
            'Message',"Are you sure to quit?",
            QtGui.QMessageBox.Yes | QtGui.QMessageBox.No,
            QtGui.QMessageBox.No)

        if reply == QtGui.QMessageBox.Yes:
            event.accept()
        else:
            self.tray_icon.show()
            self.hide()
            event.ignore()

    def restore_window(self, reason):
        if reason == QtGui.QSystemTrayIcon.DoubleClick:
            self.tray_icon.hide()
            # self.showNormal will restore the window even if it was
            # minimized.
            self.showNormal()

def main():
    app = QtGui.QApplication(sys.argv)
    ex = Example()
    ex.show()
    sys.exit(app.exec_())


if __name__ == '__main__':
    main()
1 голос
/ 28 сентября 2017

Это правильный способ обработки двойного щелчка по иконке в трее для PyQt5.

def _create_tray(self):
    self.tray_icon = QSystemTrayIcon(self)
    self.tray_icon.activated.connect(self.__icon_activated)

def __icon_activated(self, reason):
    if reason in (QSystemTrayIcon.Trigger, QSystemTrayIcon.DoubleClick):
        pass
1 голос
/ 06 ноября 2011

Это код, и он помогает мне поверить, покажи мне код

import sys
from PyQt4 import QtGui, QtCore
from PyQt4.QtGui import QDialog, QApplication, QPushButton, QLineEdit, QFormLayout, QSystemTrayIcon


class Example(QtGui.QWidget):

    def __init__(self):
        super(Example, self).__init__()
        self.initUI()

    def initUI(self):
        self.icon = QSystemTrayIcon()
        r = self.icon.isSystemTrayAvailable()
        print r
        self.icon.setIcon(QtGui.QIcon('/home/vzades/Desktop/web.png'))
        self.icon.show()
        # self.icon.setVisible(True)
        self.setGeometry(300, 300, 250, 150)
        self.setWindowIcon(QtGui.QIcon('/home/vzades/Desktop/web.png'))
        self.setWindowTitle('Message box')
        self.show()
        self.icon.activated.connect(self.activate)
        self.show()

    def closeEvent(self, event):

        reply = QtGui.QMessageBox.question(self, 'Message', "Are you sure to quit?", QtGui.QMessageBox.Yes |
                                           QtGui.QMessageBox.No, QtGui.QMessageBox.No)

        if reply == QtGui.QMessageBox.Yes:
            event.accept()
        else:
            self.icon.show()

            self.hide()

            event.ignore()

    def activate(self, reason):
        print reason
        if reason == 2:
            self.show()

    def __icon_activated(self, reason):
        if reason == QtGui.QSystemTrayIcon.DoubleClick:
            self.show()


def main():

    app = QtGui.QApplication(sys.argv)
    ex = Example()
    sys.exit(app.exec_())


if __name__ == '__main__':
    main()
...