Evgenii Legotckoi
Nov. 27, 2016, 11:13 p.m.

PyQt5 - Lesson 003. QSystemTrayIcon - How to minimize application to tray

Content

Similar article on Qt/C++

One of the first articles on working with Qt has an article on working with QSystemTrayIcon . Now came the turn of the same material, but in Python using PyQt5 library.

This article will discuss how to write a simple application to PyQt5, which will be minimized to the system tray when you click a menu item in the system tray icon, or by pressing the close button.

Project structure in this case is minimal. There is only one file, which will be run. That is main .py file.

The application itself will be as follows.

If the checkbox is checked, the application will minimize to the system tray, otherwise the application will be terminated. In addition, it is possible to fold the application to the system tray using the context menu of the application icon in the system tray.


When the application is minimized to the system tray on the close button of the application window, it will display a message about it.

main .py

  1. from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel, QGridLayout, QWidget, QCheckBox, QSystemTrayIcon, \
  2. QSpacerItem, QSizePolicy, QMenu, QAction, QStyle, qApp
  3. from PyQt5.QtCore import QSize
  4.  
  5.  
  6. class MainWindow(QMainWindow):
  7. """
  8. Сheckbox and system tray icons.
  9.          Will initialize in the constructor.
  10. """
  11. check_box = None
  12. tray_icon = None
  13.  
  14. # Override the class constructor
  15. def __init__(self):
  16. # Be sure to call the super class method
  17. QMainWindow.__init__(self)
  18.  
  19. self.setMinimumSize(QSize(480, 80)) # Set sizes
  20. self.setWindowTitle("System Tray Application") # Set a title
  21. central_widget = QWidget(self) # Create a central widget
  22. self.setCentralWidget(central_widget) # Set the central widget
  23.  
  24. grid_layout = QGridLayout(self) # Create a QGridLayout
  25. central_widget.setLayout(grid_layout) # Set the layout into the central widget
  26. grid_layout.addWidget(QLabel("Application, which can minimize to Tray", self), 0, 0)
  27.  
  28. # Add a checkbox, which will depend on the behavior of the program when the window is closed
  29. self.check_box = QCheckBox('Minimize to Tray')
  30. grid_layout.addWidget(self.check_box, 1, 0)
  31. grid_layout.addItem(QSpacerItem(0, 0, QSizePolicy.Expanding, QSizePolicy.Expanding), 2, 0)
  32.  
  33. # Init QSystemTrayIcon
  34. self.tray_icon = QSystemTrayIcon(self)
  35. self.tray_icon.setIcon(self.style().standardIcon(QStyle.SP_ComputerIcon))
  36.  
  37. '''
  38. Define and add steps to work with the system tray icon
  39. show - show window
  40. hide - hide window
  41. exit - exit from application
  42. '''
  43. show_action = QAction("Show", self)
  44. quit_action = QAction("Exit", self)
  45. hide_action = QAction("Hide", self)
  46. show_action.triggered.connect(self.show)
  47. hide_action.triggered.connect(self.hide)
  48. quit_action.triggered.connect(qApp.quit)
  49. tray_menu = QMenu()
  50. tray_menu.addAction(show_action)
  51. tray_menu.addAction(hide_action)
  52. tray_menu.addAction(quit_action)
  53. self.tray_icon.setContextMenu(tray_menu)
  54. self.tray_icon.show()
  55.  
  56. # Override closeEvent, to intercept the window closing event
  57. # The window will be closed only if there is no check mark in the check box
  58. def closeEvent(self, event):
  59. if self.check_box.isChecked():
  60. event.ignore()
  61. self.hide()
  62. self.tray_icon.showMessage(
  63. "Tray Program",
  64. "Application was minimized to Tray",
  65. QSystemTrayIcon.Information,
  66. 2000
  67. )
  68.  
  69.  
  70. if __name__ == "__main__":
  71. import sys
  72.  
  73. app = QApplication(sys.argv)
  74. mw = MainWindow()
  75. mw.show()
  76. sys.exit(app.exec())

Video

f
  • Oct. 5, 2017, 7:20 a.m.

Congratulations on the article, it was a great help.

Evgenii Legotckoi
  • Oct. 5, 2017, 1:01 p.m.

No problem. If you need help with another questions with PyQt5. Then you can ask me on the forum in this site .

PyQt5 is not my main field, therefore I have a little number of articles, but it is not problem for me to answer about PyQt5 on the forum.
a
  • June 24, 2018, 7:51 p.m.

В консоль выдается предупреждение. Почему и как исправить? Может ошибка в коде или QGridLayout как-нибудь не так добавляется?

QLayout: Attempting to add QLayout "" to MainWindow "", which already has a layout
Evgenii Legotckoi
  • June 25, 2018, 1:51 p.m.

Если не ошибаюсь, можно просто удалить вот эту строку

central_widget.setLayout(grid_layout)
Там указатель на парента передаётся в само размещение, что автоматически заменяет в размещение в central_widget и всё нормально работает.

Добрый вечер Евгений и форумчане! Не подскажите почему в при запуске данного кода в PyCharm выдаётся сообщение: "ModuleNotFoundError: No module named 'PyQt5'"?

PyQt5 установлен

Решение проблемы нашел в интернете)) Лечится так: File - Settings - Project:{name_my_project} - Project Interpreter - устанавливаем нужную нам версию интерпретатора(python 3.6.2 например) - Install(плюсик сбоку) - Available Packages - набираем в поиске PyQt5 - нажимаем Install Package

Хорошо, что нашли решение. Получается что через виртуальную среду проекта настроили, если правильно понимаю

На этот вопрос Вам ответить затрудняюсь, так как не силён такими познаниями в программировании. А что за виртуальная среда?

для питона можно использовать виртуальную среду, в которой будут установлены требуемые версии пакетов. Это позволяет держать на одной системе для каждого проекта свой набор пакетов с требуемыми версиями.

Не знаю, или вы просто доустановили нужные пакеты, или там автоматически была создана своя виртуальная среда для проекта, теоретически не должна сама настраиваться, хотя, возможно я не в курсе, как сейчас создаётся по умолчанию проект в PyCharm, я так часто новые проекты не создаю в нём.

Спасибо за подсказку, запомню на будущее.

b
  • Nov. 21, 2019, 7:55 p.m.

Не горит чекбокс
Добрый день, Евгений,
Подскажите, почему может не гореть чекбокс?
С уважением,
Сергей

Evgenii Legotckoi
  • Nov. 22, 2019, 1:43 p.m.

Добрый день. Вы полностью скопировали код?

b
  • Nov. 24, 2019, 4:54 p.m.

Евгений,
Разобрался, спасибо. вместо check_box я писал kek_box, вот и не работало.

АБ
  • Dec. 15, 2019, 10:31 p.m.

Спасибо, статья полезная. Непонятен один момент, как повесить действие на двойное нажатие?
У TrayIcon есть триггер Activated, но он срабатывает при одинарном нажатии.
У Activated есть Activation reason, среди которых есть QSystemTrayIcon::DoubleClick. Но как это использовать?
Пробовал по-разному, в том числе и:

  1. self.tray_icon.activated(QSystemTrayIcon.DoubleClick).connect(self.myfunc)
  2. self.tray_icon.activated[QSystemTrayIcon.DoubleClick].connect(self.myfunc)

Работать не хочет

Evgenii Legotckoi
  • Dec. 17, 2019, 3:43 p.m.

Передавать значение не нужно, вам нужно событие обрабатывать в self.myfunc

Попробуйте подключить слот так

  1. self.tray_icon.activated.connect(self.myfunc)

Спасибо за ответ.
Нет, еслли передавать без уточенения, тоесть как вы сказали, тогда триггер срабатывает на любые нажатия. На одинарное нажатие, на одинарное нажатие правой кнопкой тоже (таким образом чтобы вызвать меню - надо нажать правой кнопкой, сработает триггер на функцию, нажать правой кнопкой ещё раз - и только тогда появится меню).

Нужно как то уточнить, чтобы сигнал activated появлялся только при даблклике. Я вотт вычитал что для этого надо передавать Activation reason, о никак не пойму как.

А надо это, для того чтобы двойным щелчком по иконке в трее открывалось окно

Evgenii Legotckoi
  • Dec. 18, 2019, 3:19 p.m.

Ваша функция должна принимать аргумент, событие, которое вы потом через if else прогоните и определитесь, какое именно событие прилетело.

  1. def myfunc(reason):
  2. if reason == QSystemTrayIcon.DoubleClick:
  3. # Todo Something

Спасибо. Вот теперь это работает

RK
  • Aug. 13, 2020, 11:05 a.m.

Thank you... Thank you.... comrade

Comments

Only authorized users can post comments.
Please, Log in or Sign up
  • Last comments
  • Evgenii Legotckoi
    March 9, 2025, 9:02 p.m.
    К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
  • VP
    March 9, 2025, 4:14 p.m.
    Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…
  • ИМ
    Nov. 22, 2024, 9:51 p.m.
    Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
  • Evgenii Legotckoi
    Oct. 31, 2024, 11:37 p.m.
    Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
  • A
    Oct. 19, 2024, 5:19 p.m.
    Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html