Evgenii Legotckoi
Evgenii Legotckoi27 листопада 2016 р. 12:13

PyQt5 - Урок 003. QSystemTrayIcon - Як згорнути програму в трей

Аналогічна стаття на Qt/C++

Однією з перших статей роботи з Qt була стаття роботи з QSystemTrayIcon . А тепер настала черга аналогічного матеріалу, але вже на Python із застосуванням бібліотеки PyQt5.

У цій статті буде розглянуто, як написати найпростіший додаток на PyQt5, який згортатиметься в системний трей як при натисканні пунктів меню в іконці системного трею, так і при натисканні кнопки закриття вікна.

Структура проекту у разі мінімальна. Буде лише один файл, який і буде запускатися. Тобто файл main .py .

Сама програма буде виглядати наступним чином.

Якщо чекбокс відзначений, то програма буде згортатися в трей, інакше програма буде завершуватися. Крім цього є можливість згортатися програму в трей за допомогою контекстного меню іконки програми в треї.


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

main .py

from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel, QGridLayout, QWidget, QCheckBox, QSystemTrayIcon, \
    QSpacerItem, QSizePolicy, QMenu, QAction, QStyle, qApp
from PyQt5.QtCore import QSize


class MainWindow(QMainWindow):
    """
        Объявление чекбокса и иконки системного трея.
        Инициализироваться будут в конструкторе.
    """
    check_box = None
    tray_icon = None

    # Переопределяем конструктор класса
    def __init__(self):
        # Обязательно нужно вызвать метод супер класса
        QMainWindow.__init__(self)

        self.setMinimumSize(QSize(480, 80))             # Устанавливаем размеры
        self.setWindowTitle("System Tray Application")  # Устанавливаем заголовок окна
        central_widget = QWidget(self)                  # Создаём центральный виджет
        self.setCentralWidget(central_widget)           # Устанавливаем центральный виджет

        grid_layout = QGridLayout(self)         # Создаём QGridLayout
        central_widget.setLayout(grid_layout)   # Устанавливаем данное размещение в центральный виджет
        grid_layout.addWidget(QLabel("Application, which can minimize to Tray", self), 0, 0)

        # Добавляем чекбокс, от которого будет зависеть поведение программы при закрытии окна
        self.check_box = QCheckBox('Minimize to Tray')
        grid_layout.addWidget(self.check_box, 1, 0)
        grid_layout.addItem(QSpacerItem(0, 0, QSizePolicy.Expanding, QSizePolicy.Expanding), 2, 0)

        # Инициализируем QSystemTrayIcon
        self.tray_icon = QSystemTrayIcon(self)
        self.tray_icon.setIcon(self.style().standardIcon(QStyle.SP_ComputerIcon))

        '''
            Объявим и добавим действия для работы с иконкой системного трея
            show - показать окно
            hide - скрыть окно
            exit - выход из программы
        '''
        show_action = QAction("Show", self)
        quit_action = QAction("Exit", self)
        hide_action = QAction("Hide", self)
        show_action.triggered.connect(self.show)
        hide_action.triggered.connect(self.hide)
        quit_action.triggered.connect(qApp.quit)
        tray_menu = QMenu()
        tray_menu.addAction(show_action)
        tray_menu.addAction(hide_action)
        tray_menu.addAction(quit_action)
        self.tray_icon.setContextMenu(tray_menu)
        self.tray_icon.show()

    # Переопределение метода closeEvent, для перехвата события закрытия окна
    # Окно будет закрываться только в том случае, если нет галочки в чекбоксе
    def closeEvent(self, event):
        if self.check_box.isChecked():
            event.ignore()
            self.hide()
            self.tray_icon.showMessage(
                "Tray Program",
                "Application was minimized to Tray",
                QSystemTrayIcon.Information,
                2000
            )


if __name__ == "__main__":
    import sys

    app = QApplication(sys.argv)
    mw = MainWindow()
    mw.show()
    sys.exit(app.exec())

Відеоурок

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Стабільний хостинг, на якому розміщується соціальна мережа EVILEG. Для проектів на Django радимо VDS хостинг.

Вам це подобається? Поділіться в соціальних мережах!

f
  • 04 жовтня 2017 р. 21:20

Congratulations on the article, it was a great help.

Evgenii Legotckoi
  • 05 жовтня 2017 р. 03:01

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
  • 24 червня 2018 р. 09:51

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

QLayout: Attempting to add QLayout "" to MainWindow "", which already has a layout
Evgenii Legotckoi
  • 25 червня 2018 р. 03:51

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

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

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

Е
  • 16 серпня 2018 р. 23:00

PyQt5 установлен

Е
  • 17 серпня 2018 р. 11:47

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

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

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

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

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

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

b
  • 21 листопада 2019 р. 08:55

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

Evgenii Legotckoi
  • 22 листопада 2019 р. 02:43

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

b
  • 24 листопада 2019 р. 05:54

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

АБ
  • 15 грудня 2019 р. 11:31

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

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

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

Evgenii Legotckoi
  • 17 грудня 2019 р. 04:43

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

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

self.tray_icon.activated.connect(self.myfunc)
АБ
  • 17 грудня 2019 р. 12:23

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

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

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

Evgenii Legotckoi
  • 18 грудня 2019 р. 04:19

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

def myfunc(reason):
    if reason == QSystemTrayIcon.DoubleClick:
        # Todo Something
АБ
  • 18 грудня 2019 р. 05:21

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

RK
  • 13 серпня 2020 р. 01:05

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

Коментарі

Only authorized users can post comments.
Please, Log in or Sign up
AD

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:50бали,
  • Рейтинг балів-4
m
  • molni99
  • 26 жовтня 2024 р. 01:37

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:80бали,
  • Рейтинг балів4
m
  • molni99
  • 26 жовтня 2024 р. 01:29

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:20бали,
  • Рейтинг балів-10
Останні коментарі
ИМ
Игорь Максимов22 листопада 2024 р. 11:51
Django - Підручник 017. Налаштуйте сторінку входу до Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
Evgenii Legotckoi
Evgenii Legotckoi31 жовтня 2024 р. 14:37
Django - Урок 064. Як написати розширення для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
A
ALO1ZE19 жовтня 2024 р. 08:19
Читалка файлів fb3 на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
ИМ
Игорь Максимов05 жовтня 2024 р. 07:51
Django - Урок 064. Як написати розширення для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
d
dblas505 липня 2024 р. 11:02
QML - Урок 016. База даних SQLite та робота з нею в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
Тепер обговоріть на форумі
Evgenii Legotckoi
Evgenii Legotckoi24 червня 2024 р. 15:11
добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
t
tonypeachey115 листопада 2024 р. 06:04
google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
NSProject
NSProject04 червня 2022 р. 03:49
Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
9
9Anonim25 жовтня 2024 р. 09:10
Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

Слідкуйте за нами в соціальних мережах