====== Программирование в X-Window средствами Free Pascal ======
[[index|Перейти к содержанию]]
===== 1.1.8. Атрибуты окна =====
Многие атрибуты окна задаются при его создании с помощью процедуры ''XCreateWindow()'' или ''XCreateSimpleWindow()''. Впоследствии параметры можно изменить, обратившись к процедуре ''XChangeWindowAttributes()''.
Характеристики окна описываются структурами типа ''TXSetWindowAttributes'' и ''TXWindowAttributes''. Получить их можно с помощью процедуры ''XGetWindowAttributes()''.
Все они делятся на две группы. В первую входят параметры, доступные "на чтение" и "на запись". Вторая группа представляет собой внутренние данные. Программа может прочитать их, но не может менять.
Сначала перечислим поля этих структур, которые относятся к "изменяемым" параметрам.
Фон окна определяется атрибутами ''background_pixmap'' и ''background_pixel''. Первый из них задает картинку (карту пикселей), которая используется для заливки фона окна. При необходимости картина повторяется слева направо и сверху вниз. Если параметр ''background_pixmap'' равен ''None'' (задается по умолчанию), то он игнорируется. Если же при этом поле ''background_pixel'' не задано (установлено по умолчанию), то окно считается "прозрачным", в противном случае его фон заливается цветом ''background_pixel''. Атрибуты ''background_pixmap'' и ''background_pixel'' могут также принимать значение ''ParentRelative''. В этом случае характеристики фона заимствуются у родительского окна.
Вид края окна определяется полями ''border_pixmap'' и ''border_pixel''. Первый атрибут определяет карту пикселей, используемую для заполнения края. Если он равен ''None'', то край заполняется цветом ''border_pixel''. Если же и поле ''border_pixel'' не задано, то для изображения края используются соответствующие характеристики родителя. То же самое происходит, если параметр ''border_pixmap'' равен ''CopyFromParent'' (взять у родителя). Последнее значение есть значение по умолчанию.
На перерисовку окна после изменения его размеров влияют атрибуты ''bit_gravity'' и ''win_gravity''. Когда окно меняет размер, например, увеличивается или уменьшается, то, в принципе, нет необходимости перерисовывать все его содержимое. Часть окна остается неизменной. Правда, эта часть может поменять свое положение: переместиться вправо, влево, вверх или вниз.
Поле ''bit_gravity'' говорит серверу, что делать с оставшейся частью изображения. Возможные значения параметра следующие:
* ''ForgetGravity'' - содержимое окна перерисовывается (считается значением по умолчанию);
* ''StaticGravity'' - остающаяся часть не должна менять положение по отношению к главному (корневому (root)) окну сервера;
* ''NorthWestGravity'' - остающаяся часть смещается к левому верхнему углу;
* ''NorthGravity'' - остающаяся часть смещается к верху окна;
* ''NorthEastGravity'' - остающаяся часть смещается к правому верхнему углу;
* ''WestGravity'' - остающаяся часть смещается к левому краю окна;
* ''CenterGravity'' - остающаяся часть смещается к центру окна;
* ''EastGravity'' - остающаяся часть смещается к правому краю окна;
* ''SouthWestGravity'' - остающаяся часть смещается к левому нижнему углу;
* ''SouthGravity'' - остающаяся часть смещается к нижнему краю окна;
* ''SouthEastGravity'' - остающаяся часть смещается к правому нижнему углу.
Параметр ''win_gravity'' говорит о том, что делать с подокнами окна после изменения размеров последнего. Возможные значения параметра следующие (при перечислении используются следующие обозначения:
* ''H'' - изменение размеров окна по горизонтали,
* ''V'' - изменение размеров по вертикали,
* ''(H, V)'' - смещение подокна на ''H'' пикселей по горизонтали и на ''V'' пикселей по вертикали):
* ''UnmapGravity'' - подокна удаляются с экрана; окну посылается событие ''UnmapNotify'', в ответ на которое оно может переместить свои подокна и показать их с помощью процедуры ''XMapSubWindow()'';
* ''StaticGravity'' - подокна остаются на месте по отношению к главному (корневому) окну сервера;
* ''NorthWestGravity'' - устанавливается по умолчанию; соответствует смещению ''(0, 0)'';
* ''NorthGravity'' - смещение ''(H/2, 0)'';
* ''NorthEastGravity'' - смещение ''(H, 0)'';
* ''WestGravity'' - смещение ''(0, V/2)'';
* ''CenterGravity'' - смещение ''(H/2, V/2)'';
* ''EastGravity'' - смещение ''(H, V/2)'';
* ''SouthWestGravity'' - смещение ''(0, V)'';
* ''SouthGravity'' - смещение ''(H/2, V)'';
* ''SouthEastGravity'' - смещение ''(H, V)'';
Автоматическое сохранение содержимого окна, когда его часть перекрывается другими окнами, или, когда окно удаляется с экрана, определяется параметрами ''backing_store'', ''backing_planes'' и ''backing_pixel''. Сохраненные данные могут использоваться для восстановления окна, что значительно быстрее, чем его перерисовка программой в ответ на событие ''Expose''.
Параметр ''backing_store'' имеет следующие возможные значения:
* ''NotUseful'' (устанавливается по умолчанию) - серверу не рекомендуется сохранять содержимое окна;
* ''WhenMapped'' - серверу рекомендуется спасти содержимое невидимых частей окна, когда окно показывается на экране;
* ''Always'' - серверу рекомендуется сохранить содержимое окна даже, если оно не показано на экране.
Сохранение изображений требует, как правило, довольно большого расхода памяти. Атрибуты ''backing_planes'' и ''backing_pixel'' призваны уменьшить этот расход. Первый из указанных параметров говорит серверу, какие плоскости изображения надо сохранять; ''backing_pixel'' означает, какой цвет использовать при восстановлении изображения в тех плоскостях, которые не сохранялись. По умолчанию ''backing_planes'' - маска, состоящая из единиц, а ''backing_pixel'' равно 0.
Иногда при показе окна полезно сохранить содержимое экрана под окном. Если окно невелико, и показывается не на долго, то это позволяет экономить время, которое надо будет затратить на перерисовку экрана после того, как окно будет закрыто. Если атрибут ''save_under'' равен ''True'', то сервер будет пытаться сохранить изображение под окном. Если же он равен ''False'' (по умолчанию), то сервер ничего не предпринимает.
Когда обрабатывает (или не обрабатывает) событие, последнее может быть передано его родительскому окну. Атрибут ''do_not_propagate_mask'' (по умолчанию 0) говорит и о том, какие события не должны доходить до родителей.
Изменение размеров окна и его положения на экране контролируется атрибутом ''override_redirect''. Если он равен ''False'', то размер окна и его положение меняются с помощью менеджера окон. Если же он равен ''True'', то окно само решает, где ему быть, и какую ширину и высоту иметь.
Цветовую гамму окна задает параметр ''colormap''. Значение по умолчанию - ''CopyFromPatent'', которое говорит, что окно использует палитру своего непосредственного родителя.
Теперь рассмотрим "неизменяемые" параметры окна. Строго говоря, атрибуты, о которых пойдет речь, нельзя назвать неизменяемыми. Некоторые из них могут меняться сервером или менеджером окон. Но для обычных программ-клиентов они действительно являются таковыми.
Положение окна и его размеры сообщают поля ''x'', ''y'', ''width'' и ''height''. Они дают координаты левого верхнего угла, ширину и высоту окна соответственно. Координаты измеряются в пикселях по отношению к родительскому окну.
Ширина края окна определяется параметром ''border_width''.
Маска, говорящая о том, какие события выбраны для передачи окну породившим его клиентом, содержится в поле флагов ''your_event_mask''. Значение параметра образуется комбинацией флагов, идентифицирующих события.
Информация о дисплее, на котором показано окно, содержится в структуре ''Visual'', на которую показывает поле ''visual''. Эти данные, как правило, не обрабатываются обычными программами-клиентами (заметим, что для получения информации о дисплее, в системе предусмотрена процедура ''XGetVisualInfo()'').
Класс окна сообщает поле ''class''. Возможные значения: ''InputOutput'' и ''InputOnly''.
Число цветовых плоскостей дисплея (число бит-на-пиксел) помещается в поле ''depth''.
На информацию об экране, на котором помещается окно, указывает поле ''screen''. Она, как правило, не используется обычными программами.
Идентификатор главного (корневого) окна экрана, на котором помещается окно, находится в поле ''root''.
Если окно имеет палитру, и она в настоящее время активна, то поле ''map_installed'' равно ''True'', в противном случае - ''False''.
Видно в настоящее время окно на экране или нет, сообщает атрибут ''map_state''.
Маска всех событий, выбранных всеми программами для данного окна, содержится в атрибуте ''all_event_mask''. Дело в том, что окно обрабатывается не только порождающим его клиентом, но, возможно, и другими приложениями, например, менеджером окон.
Мы рассказали о том, как получить атрибуты окна, и что они означают. Теперь рассмотрим, как их изменить. Для этого можно использовать несколько процедур X Window, основной из которых является ''XChangeWindowAttributes()'', имеющая следующий прототип:
Function XChangeWindowAttributes(prDisplay : PDisplay;
nWnd : TWindow; nValueMask : cardinal;
prWinAttr : PXSetWindowAttributes) : longint; cdecl;external;
Требуемые установки атрибутов передаются через аргумент ''prWinAttr''. Он указывает на переменную типа ''TXSetWindowAttributes''. Ее поля те же, что и соответствующие поля ''TXWindowAttributes''. Разница заключается лишь в разных именах некоторых из них. Так, поле ''your_event_mask'' в ''TXWindowAttributes'' соответствует полю ''event_mask'' в ''TXSetWindowAttributes''.
Структура ''TXSetWindowAttributes'' содержит дополнительное поле ''cursor''. Оно определяет вид курсора мыши, когда последний находится в окне. Если поле равно ''None'' (значение по умолчанию), то используется курсор родительского окна, в противном случае значением параметра должен быть идентификатор того или иного курсора.
Параметр ''nValueMask'' при вызове указанной процедуры представляет комбинацию флагов, говорящих о том, какие из полей переменной ''prWinAttr'' принимать во внимание.
В следующем примере приведен фрагмент кода, в котором изменяются параметры ''border_pixmap'' и ''win_gravity'' некоторого окна:
.......
var
prDisplay : PDisplay;
prWnd : TWindow;
rWndAttr : TXSetWindowAttributes;
nValMask : cardinal;
const
nPixmap : TPixmap =0;
.......
nValMask := CWBorderPixmap or CWWinGravity;
rWndAttr.border_pixmap := nPixmap;
rWndAttr.win_gravity := StaticGravity;
.......
XChangeWindowAttributes (prDisplay, prWnd, nValMask, @rWndAttr);
.......
Отдельные атрибуты окна можно изменить более просто с помощью специальных процедур. Так, функция ''XSetWindowBackground()'' меняет фон окна, ''XSetWindowBorder()'' - его край.