Categories: Python

python-gtk/

Недавно мне стало интересно, насколько хорошо Python, на котором, напомню, я теперь пишу все свои скрипты , справляется с задачей создания GUI-приложений. И тут язык превзошел все мои ожидания, так как найти нужную библиотеку, разобраться, как ею пользоваться, а потом написать пример, демонстрирующий все интересовавшие меня возможности, заняло ну что-то около одного часа. Пример этот, как вы и сами сейчас убедитесь, показывает не только создание формочек на GTK, но и отображение иконки в трее с меню и нотификациями.

Установка зависимостей (Glade нужен только на время разработки):

sudo apt-get install python3-gi glade

Код самого скрипта:

#!/usr/bin/env python3

# gtk-example.py
# (c) Aleksander Alekseev 2016
# http://remontka.com/

import signal
import os
import gi
gi. require_version ( ‘Gtk’ , ‘3.0’ )
gi. require_version ( ‘Notify’ , ‘0.7’ )
from gi. repository import Gtk
from gi. repository import Notify

APPID = «GTK Test»
CURRDIR = os . path . dirname ( os . path . abspath ( __file__ ) )
# could be PNG or SVG as well
ICON = os . path . join ( CURRDIR , ‘python3.xpm’ )

# Cross-platform tray icon implementation
class TrayIcon:

def __init__ ( self , appid , icon , menu ) :
self . menu = menu

APPIND_SUPPORT = 1
try :
from gi. repository import AppIndicator3
except :
APPIND_SUPPORT = 0

if APPIND_SUPPORT == 1 :
self . ind = AppIndicator3. Indicator . new (
appid , icon ,
AppIndicator3. IndicatorCategory . APPLICATION_STATUS )
self . ind . set_status ( AppIndicator3. IndicatorStatus . ACTIVE )
self . ind . set_menu ( self . menu )
else :
self . ind = Gtk. StatusIcon ( )
self . ind . set_from_file ( icon )
self . ind . connect ( ‘popup-menu’ , self . onPopupMenu )

def onPopupMenu ( self , icon , button , time ) :
self . menu . popup ( None , None , Gtk. StatusIcon . position_menu , icon ,
button , time )

class Handler:

def __init__ ( self ) :
self . window_is_hidden = False

def onShowButtonClicked ( self , button ) :
msg = «Clicked: » + entry. get_text ( )
dialog = Gtk. MessageDialog ( window , 0 , Gtk. MessageType . INFO ,
Gtk. ButtonsType . OK , msg )
dialog. run ( )
dialog. destroy ( )

def onNotify ( self , *args ) :
Notify. Notification . new ( «Notification» , «Hello!» , ICON ) . show ( )

def onShowOrHide ( self , *args ) :
if self . window_is_hidden :
window. show ( )
else :
window. hide ( )

self . window_is_hidden = not self . window_is_hidden

def onQuit ( self , *args ) :
Notify. uninit ( )
Gtk. main_quit ( )

# Handle pressing Ctr+C properly, ignored by default
signal . signal ( signal . SIGINT , signal . SIG_DFL )

builder = Gtk. Builder ( )
builder. add_from_file ( ‘gtk-example.glade’ )
builder. connect_signals ( Handler ( ) )

window = builder. get_object ( ‘window1’ )
window. set_icon_from_file ( ICON )
window. show_all ( )

entry = builder. get_object ( ‘entry1’ )
menu = builder. get_object ( ‘menu1’ )
icon = TrayIcon ( APPID , ICON , menu )
Notify. init ( APPID )

Gtk. main ( )

Примечание: Насколько я смог разобраться, используемый здесь пакет обычно называют «PyGObject bindings». Существует очень похожий пакет PyGTK , но его не рекомендуется использовать в новых проектах даже на официальном сайте самого PyGTK. Поэтому в этой заметке мы его и не используем.

А вот скриншот, показывающий все возможности программы:

Скрипт был проверен на Linux с Unity , Linux с Xfce, а также FreeBSD с i3 . Во FreeBSD для работы приложения требуется установить пакет py34-gobject3.

По-моему, код предельно прост и понятен, и в особых пояснениях не нуждается.

Но на всякий случай все же отмечу, что сам интерфейс описывается в XML-файле с расширением .glade, который был создан в режиме WYSIWYG при помощи программы Glade. Так как XML является текстовым форматом, его очень здорово хранить в Git . Притом XML генерируется очень компактный и отлично читаемый. Пользоваться Glade очень просто, поэтому не стану заострять на нем внимание. Лайоут строится при помощи боксов по тому же принципу, что и в wxWidgets .

Связывание событий и хэндлеров происходит при помощи вот этой строчки:

builder. connect_signals ( Handler ( ) )

При этом имена методов класса Handler пишутся прямым текстом в Glade. Просто старый-добрый Delphi в самом хорошем смысле.

С иконкой в трее все немного непросто. В зависимости от используемого вами десктоп окружения для ее отображения следует использовать либо AppIndicator3, либо Gtk.StatusIcon. Например, в Unity работает только AppIndicator3, а вот в i3 соответствующий .typelib файл просто отсутствует, и нужно использовать Gtk.StatusIcon. А в Xfce работает и так, и так. Интересную сводную таблицу по теме можно найти в комментариях к коду Syncthing-GTK . Кстати, Syncthing недавно обозревался в этом блоге.

Ссылки по теме:

Полную версию исходного кода к посту вы найдете на GitHub .

admin

Recent Posts

После включения диагностического запуска не удается войти в Windows — как исправить?

Некоторые пользователи, экспериментируя с конфигурацией системы в окне msconfig могут столкнуться с ситуацией, когда после…

2 недели ago

WinScript — очистка и настройка Windows 11 и 10

На сайте не раз публиковались обзоры программ, предназначенных для очистки или настройки последних версий Windows.…

2 недели ago

Предварительный просмотр накопительного обновления не устанавливается — решение

При установке некоторых обновлений Windows 11, имеющих в названии «Предварительный просмотр накопительного обновления», многие пользователи…

2 недели ago

FixExec — восстановление ассоциаций .exe, .bat и .com файлов

Некоторые пользователи Windows 11, 10 и предыдущих версий системы могут столкнуться с ситуацией, когда исполняемые…

2 недели ago

Ошибка 0x800705b4 при обновлении Windows 11 и 10 — как исправить?

При установке обновлений Windows 11/10 некоторые пользователи могут столкнуться с ошибкой с кодом 0x800705b4 и…

2 недели ago

Обслуживание вашей версии Windows 11 окончено — что делать?

Пользователи Windows 11 могут столкнуться с сообщением «Обслуживание вашей версии Windows окончено» (Your version of…

2 недели ago