Своими корнями Tk уходит в далекий 1988 год, если считать и предварительные версии. Долгое время эта система считалась спутником языка Tcl, но вот уже несколько лет как она используется и с другими языками, в том числе Perl и Ruby.
Если бы у Ruby был «родной» графический интерфейс, наверное, им стал бы Tk. В настоящее время он все еще широко распространен, а в некоторые дистрибутивы Ruby входит в более или менее готовом виде.
Я упомянул о Perl не зря. Привязки Tk к Ruby и Perl похожи настолько, что любая информация о Perl/Tk применима и к Ruby/Tk. В этой связи стоит упомянуть книгу Нэнси Уолш (Nancy Walsh) «Learning Perl/Tk».
12.1.1. Обзор
В 2001 году Tk был,
наверное, самым популярным графическим интерфейсом для Ruby. Он был первым и долгое время входил в состав стандартного дистрибутива Ruby. Сейчас он, пожалуй, не так распространен, но все еще широко применяется.
Кто-то скажет, что Tk уже устарел. Те, кому нравятся объектно-ориентированные интерфейсы, будут им немного разочарованы. Но есть и достоинства: широкая известность, переносимость и стабильность.
Любое приложение Ruby/Tk должно загрузить расширение
tk
, выполнив директиву
load tk
. Далее интерфейс приложения строится поэтапно, начиная с того или иного контейнера, в который помещаются отдельные элементы управления. В конце выполняется вызов метода
Tk.mainloop
, в котором обрабатываются события: перемещения мыши, нажатия кнопок и т. д.
require "tk"
# Подготовка интерфейса приложения...
Tk.mainloop
Как и в большинстве оконных систем, графические элементы управления в Tk называются виджетами. Виджеты группируются с помощью контейнеров. Контейнер верхнего уровня называется корневым. Явно задавать корневой контейнер не всегда обязательно, но лучше это сделать.
Классы виджетов именуются так, как принято в мире Tk (в начале идет префикс
Tk
). Например, виджету
Frame
соответствует класс
TkFrame
.
Понятно, что виджеты создаются методом
new
. Первый параметр определяет контейнер, в который помещается виджет; если он опущен, подразумевается корневой контейнер.
Дополнительные параметры виджета можно задавать двумя способами. Первый (в духе Perl) — передать хэш, содержащий названия и значения атрибутов. (Напомним, что в Ruby при передаче хэша последним или единственным параметром фигурные скобки можно опускать).
Другой способ — передать конструктору блок, который будет вычислен методом
instance_eval
. Внутри блока можно вызывать методы для установки атрибутов виджета (такие методы называются так же, как сами атрибуты). Имейте в виду, что блок вычисляется в контексте вызываемого объекта, а не вызывающей программы. Это означает, например, что к переменным экземпляра вызывающего объекта в блоке обращаться нельзя.
my_widget = TkSomewidget.new do
borderwidth 2
height 40
justify "center"
end
Tk предоставляет три геометрических менеджера для управления относительным размером и расположением виджетов на экране. Наиболее распространенный из них —
pack
, остальные два —
grid
и
place
. Менеджер
grid
обладает богатыми возможностями, но не свободен от ошибок;
place
— самый простой из трех, он требует, чтобы были заданы абсолютные координаты всех расположенных внутри него виджетов. В примерах ниже мы будем пользоваться только менеджером
pack
.
12.1.2. Простое оконное приложение
Продемонстрируем очень простое приложение — окно,
в котором выводится текущая дата. Начнем с явного создания корневого контейнера root и поместим в него виджет
Label
.
require "tk"
root = TkRoot.new { title "Today's Date" }
str = Time.now.strftime("Today is \n%B %d, %Y")
lab = TkLabel.new(root) do
text str
pack("padx" => 15, "pady" => 10, "side" => "top")
end
Tk.mainloop
Здесь мы создали корневой контейнер, сформировали строку даты и создали метку. В качестве текста, изображаемого на метке, мы задали строку
str
, а чтобы все выглядело аккуратно, вызвали метод
pack
, которому сказали, что отступ по горизонтали должен составлять 15 пикселей, а по вертикали — 10 пикселей. Текст мы попросили отцентрировать в границах метки.
На рис. 12.1 показано, как выглядит окно приложения.
Рис. 12.1. Простое приложение Tk
Как было сказано выше, создать метку можно было бы и так:
lab = TkLabel.new(root) do
text str
pack("padx" => 15, "pady" => 10,
"side" => "top")
end
Экранные единицы измерения (в примере выше мы их использовали для указания
padx
и
pady
) — по умолчанию пиксели. Можно применять и другие единицы, если добавить к числу суффикс. Тогда значение становится строковым, но поскольку Ruby/Tk на это наплевать, то и мы не станем беспокоиться. Допустимы следующие единицы измерения: сантиметры (
с
), миллиметры (
m
), дюймы (
i
) и пункты (
р
). Все показанные ниже способы указания
padx
правильны:
pack("padx" => "80m")
pack("padx" => "8с")
pack("padx" => "3i")
pack("padx" => "12p")
Атрибут
side
в данном случае не дает ничего, поскольку мы установили его значение по умолчанию. Если вы измените размер окна, то текст останется «приклеенным» к верхней части той области, которой принадлежит. Другие возможные значения
side
:
right
,
left
и
bottom
.
У метода
pack
есть и другие атрибуты, управляющие размещением виджетов на экране. Мы рассмотрим не все.
Атрибут
fill
указывает, должен ли виджет заполнять весь выделенный для него прямоугольник (по горизонтали и/или по вертикали). Допустимые значения:
x
,
у
,
both
и
none
(по умолчанию
none
).
Атрибут
anchor
«скрепляет» виджет с теми или иными сторонами его прямоугольника; при этом применяется нотация «сторон света». По умолчанию подразумевается значение