АНДРЕЙ УВАРОВ
PHP-GTK
Наверное, не стоит говорить о том, какое большое распространение получил язык PHP в настоящее время. Преимущественно он используется в веб-программировании (собственно, как и задумывалось его создателем). Надёжно укрепившись на серверах, теперь не спеша переходит и на десктопы. Об этом свидетельствуют как минимум две вещи: появление поддержки ncurses и проект PHP-GTK, о чём и поговорим сегодня.
Данная статья ориентирована на достаточно широкий круг читателей, так как в ней не рассматриваются какие-либо сложные аспекты. И может оказаться полезной как для людей, желающих использовать PHP для написания приложений с графическим интерфейсом пользователя, выполняющихся на клиентской стороне, так и для тех, кто просто желает начать изучение GTK.
PHP-GTK представляет собой расширение (модуль), позволяющее создавать приложения с оконным графическим интерфейсом. Стоит заметить, что данное расширение является кроссплатформенным, речь идёт не только о *nix-подобных системах, но и Win. Всю документацию, исходные коды или же откомпилированное расширение вы найдёте на сайте проекта – gtk.php.net.
Что такое GTK?
Итак, что же такое GTK? На самом деле в данном случае правильней было бы говорить о GTK+. GTK представляет собой набор библиотек, используемых для создания оконного графического интерфейса пользователя. Основным отличием GTK+ от GTK является объектная ориентированность и наличие механизма сигналов. Аббревиатура GTK означает – GIMP Toolkit, так как изначально разрабатывалась для GIMP (в рамках одноименного проекта).
GTK+ основывается на двух основных понятиях: виджет и сигнал.
Виджетом называется любой компонент графического интерфейса, будь то кнопка, поле ввода или окно. Все виджеты наследуются от абстрактного класса GtkWidget. Виджеты, которые могут содержать другие виджеты, называются контейнерами. Предком контейнеров является Gtk Container (сам же GtkContainer наследуется от GtkWidget).
Сигналами называются сообщения, создаваемые виджетами. Сигналы используются для того, чтобы реагировать на действия пользователя. Например, при появлении курсора над кнопкой – экземпляром класса GtkButton, генерируется сигнал «enter», а при нажатии – сигнал «clicked». Сигналы наследуются. Для обработки сигналов используются callback-фукции. Не стоит путать сигналы с событиями. Сигналы существуют в рамках GTK. А события являются отражением механизма событий в оконной системе.
Итак, несколько «оглядевшись», можно приступить и к примерам.
Напишем простейшее приложение.
<?php
if( !extension_loaded('gtk')) {
dl( 'php_gtk.' . PHP_SHLIB_SUFFIX);
}
$window = &new GtkWindow();
$window->show();
gtk::main();
?>
Разберём нашу программу «по косточкам».
Осуществляем проверку, загружено ли уже расширение, если нет, то загружаем его.
if( !extension_loaded("gtk")) {
dl( "php_gtk." . PHP_SHLIB_SUFFIX);
}
Создаём экземпляр класса GtkWindow – окно. И осуществляем его отображение вызовом метода show(). Если размер окна не устанавливается в программе, как в нашем случае, то используется размер по умолчанию – 200х200. То же касается и заголовка окна, по умолчанию это имя скрипта.
$window = &new GtkWindow();
$window->show();
Вызовом gtk::main() мы «включаем» основной цикл программы, то есть программа начинает слушать события. Без этой строки наше приложение просто отобразило бы окно и завершило на этом свою работу.
Если вы попробуете закрыть приложение, то оно всё равно не завершит работу, так как сигнал о завершении выполнения игнорируется
Для нормального завершения мы должны были обрабатывать сигнал «destroy» объекта window.
Итак, овладев основами, можно приступить к рассмотрению чуть более сложного примера.
<?php
if( !extension_loaded('gtk')) {
dl( 'php_gtk.' . PHP_SHLIB_SUFFIX);
}
function button_click(&$text){
$window = &new GtkWindow();
// устанавливаем заголовок окна
$window->set_title('Message');
// устанавливаем размер окна
$window->set_usize(200, 30);
$label = &new GtkLabel($text->get_text());
// добавляем объект в контейнер
$window->add($label);
$window->show_all();
}
$mainwindow = &new GtkWindow();
// устанавливаем заголовок окна
$mainwindow->set_title('Hello world');
$mainwindow->connect_object('destroy', Array('gtk', ї
'main_quit'));
// Этот класс является специальным контейнером, который позволяет размещать по горизонтали
// добавленные в него объекты.
$box = &new GtkHBox();
$button = &new GtkButton('press me');
$entry = &new GtkEntry();
$button->connect_object('clicked', 'button_click', $entry);
$box->pack_start($entry);
$box->pack_end($button);
$mainwindow->add($box);
$mainwindow->show_all();
gtk::main();
?>
Особый интерес для нас в данном случае представляет метод connect_object(). Данный метод осуществляет присоединение callback-функции к нужному сигналу. По своему назначению метод connect_object() сходен с connect() (оба метода наследуются от GtkObject). Строку:
$mainwindow->connect_object("destroy", Array("gtk", "main_quit"));
Можно было бы заменить вызовом метода connect(), но при этом нам было бы необходимо определять функцию для обработки сигнала «destroy».
...
function mainDestroy(){
gtk::main_quit();
}:
...
$mainwindow->connect("destroy", "mainDestroy");
...
Первым параметром connect() является имя сигнала, вторым – имя callback-функции.
Напишем похожую программу, только уже на языке Python.
sample.py:
import gtk
def button_click(text):
window = gtk.Window()
window.set_title("Message")
window.set_usize(200, 30)
label = gtk.Label(text.get_text())
window.add(label)
window.show_all()
def main_quit(object):
gtk.main_quit()
mainwindow = gtk.Window()
mainwindow.set_title("Hello world")
mainwindow.connect("destroy", main_quit)
box = gtk.HBox()
button = gtk.Button("press me")
entry = gtk.Entry()
button.connect_object("clicked", button_click, entry)
box.pack_start(entry)
box.pack_end(button)
mainwindow.add(box)
mainwindow.show_all()
gtk.main()
Как видите, все различия заключаются в синтаксисе. Таким образом, освоив GTK, вам не составит труда писать приложения практически на любом языке (разумеется, имеющем поддержку GTK).
На данный момент PHP-GTK существует только для PHP4, но сейчас ведётся разработка следующей версии, в которой будет реализована поддержка GTK2 и PHP5, что выглядит весьма перспективно.