[ << ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
In this chapter we will describe the basics of defining forms. Not all possible classes of objects are described here, only the most common ones. Also, for most classes only a subset of the available types are described. See Part III for a complete overview of all object classes currently available.
Normally you will almost never have to write the code to define forms yourself because the package includes a Form Designer that does this for you (see Part II). Still it is useful to read through this chapter because it explains what some of the different object classes are and how to work with them.
A form consists of a collection of objects. A form definition is started with the routine
FL_FORM *fl_bgn_form(int type, FL_Coord w, FL_Coord h); |
w
and h
indicate the width and height of the form (in
pixels by default). Positions in the form will be indicated by
integers between 0 and w-1
or h-1
. The actual size of
the form when displayed on the screen can still be varied. type
indicates the type of the background drawn in the form. The background
of each form is a box. See the next section for the different types
available. The routine returns a pointer to the form just defined.
This pointer must be used, for example, when drawing the form or doing
interaction with it. The form definition ends with
void fl_end_form(void); |
Between these two calls objects are added to the form. The following sections describe some of the more common classes of objects that can be added to a form.
Many different forms can be defined and displayed when required. Normally you probably will first define all your forms before starting the actual work but it's no problem to define new forms also later on.
The first type of objects are boxes. Boxes are simply used to give the forms and objects a nicer appearance. They can be used to visually group other objects together. The background of each form is a box. To add a box to a form you use the routine
FL_OBJECT *fl_add_box(int type, FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, const char *label); |
where type
indicates the shape of the box. The Forms
Library at the moment supports the following types of boxes:
FL_NO_BOX
FL_UP_BOX
FL_DOWN_BOX
FL_BORDER_BOX
FL_SHADOW_BOX
FL_FRAME_BOX
FL_ROUNDED_BOX
FL_EMBOSSED_BOX
FL_FLAT_BOX
A flat box without a border (normally invisible unless given a different color than the surroundings)
FL_RFLAT_BOX
A rounded box without a border (normally invisible unless given a different color than the surroundings)
FL_RSHADOW_BOX
FL_OVAL_BOX
FL_ROUNDED3D_UPBOX
FL_ROUNDED3D_DOWNBOX
FL_OVAL3D_UPBOX
FL_OVAL3D_DOWNBOX
An oval box going into the screen
x
and y
indicate the upper left corner of the box in the
form. w
and h
are the width and height of the box.
label
is a text that is placed in the center of the box. If you
don't want a label in the box, use an empty string or a NULL
pointer. The label can be either one line or multiple lines. To obtain
multi-line labels, insert newline characters (\n
) in the label
string. It is also possible to underline the label or one of the
characters in the label. This is accomplished by embedding
<CNTRL> H
(\010
) after the letter that needs to be
underlined. If the very first character of the label is
<Ctrl>H
, the entire label is underlined.
The routine fl_add_box()
returns a pointer to the box object.
(All routines that add objects return a pointer to the object.) This
pointer can be used for later references to the object.
It is possible to change the appearance of a box in a form. First of all, it is possible to change the color of the box and secondly, it is possible to change color, size and position of the label inside the box. Details on changing attributes of objects can be found in Changing Attributes. Just a simple example has to suffice here. Assume we want to create a red box, coming out of the screen with the large words "I am a Box" in green in the center:
FL_OBJECT *thebox; thebox = fl_add_box(FL_UP_BOX, 20, 20, 100, 100, "I am a Box"); fl_set_object_color(thebox, FL_RED, 0 ); /* make box red */ fl_set_object_lcol(thebox, FL_GREEN ); /* make label green */ fl_set_object_lsize(thebox, FL_LARGE_SIZE); /* make label large */ |
Of course, this has to be placed inside a form definition.
A second type of object is text. Text can be placed at any place on the form in any color you like. Placing a text object is done with the routine
FL_OBJECT *fl_add_text(int type, FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, const char *label); |
where type
indicates the shape of the text. The Forms Library at
the moment supports only one type of text: FL_NORMAL_TEXT
.
The text can be placed inside a box using the routine
fl_set_object_boxtype()
to be described in Changing Attributes. Again, the text can be multi-lined or underlined by
embedding respectively the newline (\n
) or <Ctrl>H
(\010
) in the label. The style, size and color of the text can
be controlled and changed in many ways. See section Label Attributes and Fonts.
Note that there is almost no difference between a box with a label and a text. The only difference lies in the position where the text is placed. Text is normally placed inside the box at the left side. This helps you put different lines of text below each other. Labels inside boxes are by default centered in the box. You can change the position of the text inside the box using the routines in Label Attributes and Fonts. Note that, when not using any box around the text there is no need to specify a width and height of the box; they can both be 0.
A very important class of objects are buttons. Buttons are placed on the form such that the user can push them with the mouse. Different types of buttons exist: buttons that return to their normal position when the user releases the mouse, buttons that stay pushed until the user pushes them again and radio buttons that make other buttons be released. Adding a button to a form can be done using the following routine
FL_OBJECT *fl_add_button(int type, FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, const char *label); |
label
is the text placed inside (or next to) the button.
type
indicates the type of the button. The Forms Library at the
moment supports a number of types of buttons. The most important ones
are:
FL_NORMAL_BUTTON FL_PUSH_BUTTON FL_TOUCH_BUTTON FL_RADIO_BUTTON |
They all look the same on the screen but their functions are quite different. Each of these buttons get pushed down when the user presses the mouse on top of them. What actually happens when the user does so depends on the type of button.
A normal button returns to its normal position when the user releases the mouse button.
A push button remains pushed and is only released when the user pushes it again.
A touch button is like a normal button except that as long as the user keeps the mouse pressed it is returned to the application program. (See section Doing Interaction, on the details of interaction.)
A radio button is a push button with additional extra property: Whenever the user pushes a radio button, all other pushed radio buttons in the form (or at least in the group, see below) they belong to are released. In this way the user can make a choice among some mutually exclusive possibilities.
Whenever the user pushes a button and then releases the mouse, the
interaction routine fl_do_forms()
is interrupted and
returns a pointer to the button that was pushed and released. If a
callback routine is present for the object being pushed, this routine
will be invoked. In either case, the application program knows that
the button was pushed and can take action accordingly. In the first
case, control will have to be returned to fl_do_forms()
again after the appropriate action is performed; and in the latter,
fl_do_forms()
would never return. See section Doing Interaction, for details on the interaction with
forms.
Different types of buttons are used in all the example programs provided. The application program can also set a button to be pushed or not itself without a user action. (This is of course only useful for push buttons and radio buttons. Setting a radio button does not mean that the currently set radio button is reset. The application program has to do this.) To set or reset a button use the routine
void fl_set_button(FL_OBJECT *obj, int pushed); |
pushed
indicates whether the button should be pushed (1) or
released (0). To figure out whether a button is pushed or not use
int fl_get_button(FL_OBJECT *obj); |
See the program `pushbutton.c' for an example of the use of push buttons and setting and getting button information.
The color and label of buttons can again be changed using the routines in Changing Attributes.
There are other classes of buttons available that behave the same way as buttons but only look different.
have a small "light" (colored area) in the button. Pushing the button
switches the light on, and releasing the button switches it off. To
add a light button use fl_add_lightbutton()
with the same
parameters as for normal buttons. The other routines are exactly the
same as for normal buttons. The color of the light can be controlled
with the routine fl_set_object_color()
. See section Changing Attributes.
are buttons that are round. Use fl_add_roundbutton()
to
add a round button to a form.
are buttons that are round and 3D-ish looking. Round and light buttons are nice as radio and push buttons.
are buttons that have a small checkbox the user can push. To add a
check button, use fl_add_checkbutton()
. More stylish for
a group of radio buttons.
are buttons that have a bitmap on top of the box. Use routine
fl_add_bitmapbutton()
to add a bitmap button to a form.
are buttons that have a pixmap on top of the box. Use routine
fl_add_pixmapbutton()
to add a pixmap button to a form.
Playing with different boxtypes, colors, etc., you can make many different types of buttons. See `buttonall.c' for some examples. Fig. 16.1 shows all buttons in their default states.
Sliders are useful in letting the user indicate a value between some fixed bounds. A slider is added to a form using the routine
FL_OBJECT *fl_add_slider(int type, FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, const char *label); |
The two most important types of sliders are FL_VERT_SLIDERM
and
FL_HOR_SLIDER
. The former displays a slider that can be moved
vertically and the latter gives a slider that moves horizontally. In
both cases the label is placed below the slider. Default value of the
slider is 0.5 and can vary between 0.0 and 1.0. These values can be
changed using the routines:
void fl_set_slider_value(FL_OBJECT *obj, double val); void fl_set_slider_bounds(FL_OBJECT *obj, double min, double max); |
Whenever the value of the slider is changed by the user, it results in the slider being returned to the application program or the callback routine invoked. The program can read the slider value using the call
double fl_get_slider_value(FL_OBJECT *obj); |
and take action accordingly. See the example program `demo05.c' for the use of these routines.
A valslider is almost identical with a normal slider. The only difference is the way the slider is drawn. For valsliders, in addition to the slider itself, its current value is also shown.
FL_OBJECT *fl_add_valslider(int type, FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, const char *label); |
For all other interaction with a valslider the same function as for normal sliders can be used.
It is often required to obtain textual input from the user, e.g. a file name, some fields in a database, etc. To this end input fields exist in the Forms Library. An input field is a field that can be edited by the user using the keyboard. To add an input field to a form use
FL_OBJECT *fl_add_input(int type, FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, const char *label); |
The main type of input field available is FL_NORMAL_INPUT
. The
input field normally looks like an FL_DOWN_BOX
. This can be
changed using the routine fl_set_object_boxtype()
to be
described in Changing Attributes.
Whenever the user presses the mouse inside an input field a cursor will
appear in it (and it will change color). Further input will appear
inside this field. Full emacs(1) style editing is supported. When the
user presses <Return>
or <Tab>
the input field is returned
to the application program and further input is directed to the next
input field. (The <Return>
key only works if there are no default
buttons in the form. See the overview of object classes. The
<Tab>
key always works.)
The user can use the mouse to select parts of the input field which will be removed when the user types the erase character or replaced by any new input the user types in. Also the location of the cursor can be moved in the input field using the mouse.
The input field is fully integrated with the X Selection mechanism. Use the left button to cut from and the middle button to paste into an input field.
The application program can direct the focus to a particular object using the call
void fl_set_focus_object(FL_FORM *form, FL_OBJECT *obj); |
It puts the input focus in the form form onto object obj
. To
obtain the focus object, the following routine is available
FL_OBJECT *fl_get_focus_object(FL_FORM *form); |
Note that the label is not the default text in the input field. The label is (by default) placed in front of the input field. To set the contents of the input field use the routine
void fl_set_input(FL_OBJECT *obj, const char *str); |
To change the color of the input text or the cursor use
void fl_set_input_color(FL_OBJECT *obj, int tcol, int ccol); |
Here tcol
indicates the color of the text
and ccol
is the color of the cursor. To obtain the
string in the field (when the user has changed it) use:
[const] char *fl_get_input(FL_OBJECT *obj); |
Notice the bracket around the qualifier const. This indicates although the function is not declared to return a pointer to const string, it should be used as one. This is done mainly for compilation on machines whose string library header is buggy. Modifying the string returned by this function can produce unpredictable results.
See the program `demo06.c' for an example of the use of input fields.
Objects inside a form definition can be grouped together. To this end we place them in between the routines
FL_OBJECT *fl_bgn_group(void); |
void fl_end_group(void); |
The first function returns a pointer to a pseudo-object that
represents the start of the group (its class is
FL_BEGIN_GROUP
). It can be used in a number of functions to
work on the whole group at once. Also the second creates a
pseudo-object (of class FL_END_GROUP
), marking the groups end,
but since this object can't be used its address isn't returned.
Groups can't be nested. Groups are useful for two reasons. First of all it is possible to hide groups of objects. (See section Hiding and Showing, below.) This is often very handy. We can, for example, display part of a form only when the user asks for it (see demo program `group.c'. Some attributes are naturally multi-objects, e.g., to glue several objects together using the gravity attribute. Instead of setting the gravity for each object, you can place all related objects inside a group and set the resize/gravity attribute of the group.
The second reason is for using radio buttons. As indicated in section 3.4 pushing a radio button makes the currently pushed radio button released. In fact, this happens only with radio buttons in the particular group. So to make two pairs (or more) of radio buttons, simply put each pair in a different group so that they won't interfere with each other. See, e.g., the example program `buttonall.c'. It is a good idea to always put radio buttons in a group, even if you have only one set of them.
It is possible to add objects to an existing group
FL_OBJECT *fl_addto_group(FL_OBJECT *group); |
where group
is the object returned by
fl_bgn_group()
. After this call, you can start adding
objects to the group (e.g., fl_add_button()
etc.). The
newly added objects are appended at the end of the group. When through
with adding, use fl_end_group()
as before.
It is possible to temporarily hide certain objects or groups of objects. To this end, use the routine
void fl_hide_object(FL_OBJECT *obj); |
obj
is the object to hide or the group of objects to hide. Hidden
objects don't play any role anymore. All routines on the form act as if
the object does not exist. To make the object or group of objects
visible again use
void fl_show_object(FL_OBJECT *obj); |
Hiding and showing (groups of) objects are useful to change the appearance of a form depending on particular information provided by the user. You can also make overlapping groups in the form and take care that only one of them is visible.
If you want to know if an object is shown you can use
int fl_object_is_visible(FL_OBJECT *obj); |
Please note for an object to be visible also the form it belongs to must be shown, which isn't factored into the return value.
Sometimes you might want a particular object to be temporarily inactive, e.g., you want to make it impossible for the user to press a particular button or to type input in a particular field. For this you can use the routine
void fl_deactivate_object(FL_OBJECT *obj); |
obj
is the object to be deactivated.
When obj
is a group the whole group is deactivated.
To reactivate the group or button use the routine
void fl_activate_object(FL_OBJECT *obj); |
To find out if an object is in active state use the function
int fl_object_is_active(FL_OBJECT *obj); |
Normally you also want to give the user a visual indication that the object is not active. This can, for example, be done by changing the label color to grey (see below). This is not done automatically, so unless you set e.g. a different color the objects appearance won't change on deactivation (or re-activation).
It is possible to simulate the action of an object being triggered from within the program by using the following routine
void fl_trigger_object(FL_OBJECT *obj); |
Calling this routine on an object obj
results in the object
returned to the application program or its callback being called if it
exists. Note however, there is no visual feedback, i.e.
fl_trigger_object(button)
will not make the button object
named button
appear to be pushed.
There are a number of general routines that can be used to alter the appearance of any object.
3.11.1 Color | ||
3.11.2 Bounding Boxes | ||
3.11.3 Label Attributes and Fonts | ||
3.11.4 Tool Tips | ||
3.11.5 Redrawing Objects | ||
3.11.6 Changing Many Attributes | ||
3.11.7 Symbols |
To change the color of a particular object use the routine
void fl_set_object_color(FL_OBJECT *obj, FL_COLOR col1, FL_COLOR col2); |
col1
and col2
are indices into a colormap. Which colors
are actually changed depends on the type of the object. For box and
text only col1
is important. It indicates the color of the box
or of the box in which the text is placed. For buttons, col1
is
the color of the button when released and col2
is the color of
the button when pushed. (Note that when changing the color of a button
the nice property that the color of a button changes when the mouse
moves over it disappears.) For light buttons the two colors indicate
the color of the light when off and when on. For bitmap buttons,
col1
is the color of the box and col2
is the color of
the bitmap. For sliders col1
is the color of the background of
the slider and col2
is the color of the slider itself. Finally,
for input objects col1
is the color of the input field when it
is not selected and col2
is the color when it has input focus,
i.e. the user can enter text. For all types of objects, the default
colors can be found in the file `forms.h'. For example, for input
fields the default colors are FL_INPUT_COL1
and
FL_INPUT_COL2
. Form Designer comes in very handy in
familiarizing you with various attributes since you can change all
attributes of an object and immediately see the difference by
"test"ing the object.
To find out the colors of an object use
void fl_get_object_color(FL_OBJECT *obj, FL_COLOR *col1, FL_COLOR *col2); |
The following pre-defined color symbols can be used in all color change requests. If the workstation does not support this many colors, substitution by the closest color will happen.
Of all the colors listed in the table above FL_FREE_COL1
has
the largest numerical value, and all color with indices smaller than
that are used (or can potentially be used) by the Forms Library
although, if you wish, they can also be changed using the following
routine prior to fl_initialize()
:
void fl_set_icm_color(FL_COLOR index, int r, int g, int b); |
Note that although the color of an object is indicated by a single index, it is not necessarily true that the Forms Library is operating in PseudoColor. Forms Library is capable of operating in all visuals and as a matter of fact the Forms Library will always select TrueColor or DirectColor if the hardware is capable of it.
The actual color is handled by an internal colormap of
FL_MAX_COLORS
entries (default is 1024). To change or query the
values of this internal colormap use the call
void fl_set_icm_color(FL_COLOR index, int r, int g, int b); void fl_get_icm_color(FL_COLOR index, int *r, int *g, int *b); |
Call fl_set_icm_color()
before
fl_initialize()
to change XForms's default colormap. Note
that these two routines do not communicate with the X server, they
only populate/return information about the internal colormap, which is
made known to the X server by the initialization routine
fl_initialize()
.
To change the colormap and make a color index active so that it can be
used in various drawing routines after fl_initialize()
initialization, use the following function
unsigned long fl_mapcolor(FL_COLOR i, int red, int green, int blue); |
This function frees the previous allocated pixel corresponding to color
index i
and re-allocates a pixel with the RGB value specified.
The pixel value is returned by the function. It is recommended that you
use an index larger than FL_FREE_COL1
for your remap request to
avoid accidentally freeing the colors you have not explicitly allocated.
Indices larger than 224 are reserved and should not be used.
Sometimes it may be more convenient to associate an index with a colorname, e.g., "red" etc., which may have been obtained via resources. To this end, the following routine exists
long fl_mapcolorname(FL_COLOR i, const char *name); |
where name
is the color name(2). The function returns -1 if the colorname name is
not resolved. You can obtain the RGB values of an index by using the
following routine
unsigned long fl_getmcolor(FL_COLOR i, int *red, int *green, int *blue); |
The function returns the pixel value as known by the Xserver. If the
requested index, i
, is never mapped or is freed, the RGB values
as well as the pixel value are random. Since this function communicates
with the Xserver to obtain the pixel information, it has a two-way
traffic overhead. If you're only interested in the internal colormap of
XForms, fl_get_icm_color()
is more efficient.
Note that the current version only uses the lower byte of the primary color. Thus all primary colors in the above functions should be specified in the range of 0-255 inclusive.
To free any colors that you no longer need, the following routine should be used
void fl_free_colors(FL_COLOR colors[], int ncolors); |
Prior to XForms version 0.76, there is a color "leakage" in the
implementation of the internal colormap that prevents the old index from
being freed in the call fl_mapcolor()
, resulting in accelerated
colormap overflow and some other undesirable behavior. Since there may
still be some applications based on older versions of the Forms Library,
a routine is provided to force the library to be compatible with the
(buggy) behavior:
void fl_set_color_leak(int flag); |
Due to the use of an internal colormap and the simplified user interface, changing the colormap value for the index may not result in a change of the color for the object. An actual redraw of the object (see below) whose color is changed may be required to have the change take effect. Therefore, a typical sequence of changing the color of a visible object is as follows:
fl_mapcolor(newcol, red, green, blue); /* obj uses newcol */ fl_redraw_object(obj); |
Each object has a bounding box. This bounding box can have different
shapes. For boxes it is determined by the type. For text it is normally
not visible. For input fields it normally is a FL_DOWN_BOX
, etc.
The shape of the box can be changed using the routine
void fl_set_object_boxtype(FL_OBJECT *obj, int boxtype); |
boxtype
should be one of the following: FL_UP_BOX
,
FL_DOWN_BOX
, FL_FLAT_BOX
, FL_BORDER_BOX
,
FL_SHADOW_BOX
, FL_ROUNDED_BOX
, FL_RFLAT_BOX
,
FL_RSHADOW_BOX
and FL_NO_BOX
, with the same meaning as the
type for boxes. Some care has to be taken when changing boxtypes. In
particular, for objects like sliders, input fields, etc. never use the
boxtype FL_NO_BOX
. Don't change the boxtype of objects that are
visible on the screen. It might have undesirable effects. If you must do
so, redraw the entire form after changing the boxtype of an object (see
below). See the program `boxtype.c' for the effect of the boxtype
on the different classes of objects.
It is possible to alter the appearance of an object by changing the border width attribute
void fl_set_object_bw(FL_OBJECT *obj, int bw); |
To find out about the current setting for the border width of an object call
int fl_get_object_bw(FL_OBJECT *obj); |
Border width controls the "height" of an object, e.g., a button having
a border width of 3 pixels appears more pronounced than one having a
border width of 2. The Forms Library's default is
FL_BOUND_WIDTH
(1) pixels (before version 1.0.91 the default
was 3). Note that the border width can be negative. Negative border
width does not make a down box, rather, it makes the object having an
upbox appear less pronounced and "softer". See program
`borderwidth.c' for the effect of border width on different
objects. All applications developed using XForms accept a command line
option `-bw', followed by an integer number, the user can use to
select the preferred border width. It is recommended that you document
this flag in your application documentation. If you prefer a certain
border width, use fl_set_defaults()
or
fl_set_border_width()
before fl_initialize()
to set the border width instead of hard-coding it on a per form or per
object basis so the user has the option to change it at run time via
the `-bw' flag.
There also exists a call that changes the object border width for the entire application
void fl_set_border_width(int border_width); |
There are also a number of routines to change the appearance of the label. The first one is
void fl_set_object_lcol(FL_OBJECT *obj, FL_COLOR lcol); |
It sets the color of the label. The default is black
(FL_BLACK
). The font size of the label
can be changed using the routine
void fl_set_object_lsize(FL_OBJECT *obj, int lsize); |
where lsize
gives the size in points. Depending on the server and
fonts installed, arbitrary sizes may or may not be possible. Fig 3.5
shows the font sizes that are standard with MIT/XConsortium
distribution. So use of these values is encouraged. In any case, if a
requested size can not be honored, substitution will be made. The
default size for XForms is 10pt.
FL_TINY_SIZE | 8pt |
FL_SMALL_SIZE | 10pt |
FL_NORMAL_SIZE | 12pt |
FL_MEDIUM_SIZE | 14pt |
FL_LARGE_SIZE | 18pt |
FL_HUGE_SIZE | 24pt |
Labels can be drawn in many different font styles. The style of the label can be controlled with the routine
void fl_set_object_lstyle(FL_OBJECT *obj, int lstyle); |
The default font for the Forms Library is Helvetica at 10pt.
Additional styles are available:
The last three styles are special in that they are modifiers, i.e.,
they do not cause font changes themselves, they only modify the
appearance of the font already active. E.g., to get a bold engraved
text, set lstyle
to FL_BOLD_STYLE|FL_ENGRAVED_STYLE
.
Other styles correspond to the first 12 fonts. The package, however, can handle up to 48 different fonts. The first 16 (numbers 0-15) have been pre-defined. The following table gives their names:
0 helvetica-medium-r 1 helvetica-bold-r 2 helvetica-medium-o 3 helvetica-bold-o 4 courier-medium-r 5 courier-bold-r 6 courier-medium-o 7 courier-bold-o 8 times-medium-r 9 times-bold-r 10 times-medium-o 11 times-bold-o 12 charter-medium-r 13 charter-bold-r 14 charter-medium-i 15 Symbol |
The other 32 fonts (numbers 16-47) can be filled in by the application program. Actually, the application program can also change the first 16 fonts if required (e.g., to force a particular resolution). To change a font for the the entire application, use the following routine:
int fl_set_font_name(int index, const char *name); |
where index
is the number of the font (between 0 and 47) and
name
should be a valid font name (with the exception of the size
field). If you are defining a completely different font family
starting at index k
, it's a good idea to define
k + FL_BOLD_STYLE
to be the corresponding bold font in the
family , and k + FL_ITALIC_STYLE
the corresponding italic font
in the family (so object like browser can obtain correct style when
switching font styles):
#define Pretty 30 #define PrettyBold (Pretty + FL_BOLD_STYLE) #define PrettyItalic (Pretty + FL_ITALIC_STYLE) fl_set_font_name(Pretty, fontname); fl_set_font_name(PrettyBold, boldfontname); fl_set_font_name(PrettyItalic, italicfontname); ... fl_set_object_lstyle(obj, PrettyBold); |
The function returns a negative value if the requested font is invalid
or otherwise can't be loaded. Note however, if this routine is called
before fl_initialize()
, it will return 0, but may fail later if
the font name is not valid. To change the default font
(helvetica-medium), a program should change font FL_NORMAL_STYLE
.
If a font name in XLFD is given, a question mark
(?
) in the point size position informs
the Forms Library that variable size will be requested later. It is
preferable that the complete XLFD name (i.e., with 14 dashes and
possibly wildcards) be given because a complete name has the advantage
that the font may be re-scalable if scalable fonts are available. This
means that although both
"-*-helvetica-medium-r-*-*-*-?-*-*-*-*-*-*" "-*-helvetica-medium-r-*-*-*-?-*-*" |
are valid font names, the first form may be re-scalable while the the second is not. To obtain the actual built-in font names, use the following function
int fl_enumerate_fonts(void (*cb)(const char *f), int shortform); |
where cb
is a callback function that gets called once for every
built-in font name. The font name is passed to the callback function as
the string pointer parameter while shortform
selects if a short
form of the name should be used.
XForms only specifies the absolutely needed parts of the font names, and
assumes the font path is set so that the server always chooses the most
optimal fonts for the system. If this is not true, you can use
fl_set_font_name()
to select the exact
font you want. In general, this is not recommended if your application
is to be run/displayed on different servers.
See `fonts.c' for a demonstration of all the built-in font styles available.
You can change the alignment of the label with respect to the bounding box of the object. For this you use the routine
void fl_set_object_lalign(FL_OBJECT *obj, int align); |
The following possibilities exist:
Normally, all the alignment requests place the text outside the box,
except for
FL_ALIGN_CENTER
. This can be changed by using a
special mask, FL_ALIGN_INSIDE
, to request alignments that place
the text inside the box. This works for most of the objects in the
library but not for all. For sliders, inputs and some others, placing
the label inside the box simply does not make sense. In these cases,
inside request is ignored. See the demo program `lalign.c' for an
example use of FL_ALIGN_INSIDE
.
void fl_set_object_label(FL_OBJECT *obj, const char *label); |
changes the label of a given object. An internal copy of the parameter
label
is made. As mentioned earlier, newline (\n
) can be
embedded in the label to generate multiple lines. By embedding
<Ctrl>H
(\010
) in the label, the entire label or one of
the characters in the label can be underlined. The function
const char * fl_get_object_label(FL_OBJECT *obj); |
returns the label string.
As will be seen later, an object can be decorated by icons instead of labels. For this kind of object, it is helpful to show a text string that explains the function the object controls under appropriate conditions. Forms Library elected to show the message after the mouse enters the object for about 600 milli-seconds. The text is removed when the mouse leaves the object or when the mouse is pressed.
To set the text, use the following routine
void fl_set_object_helper(FL_OBJECT *obj, const char *helpmsg); |
where helpmsg
is a text string (with possible embedded newlines
in it) that will be shown when the mouse enters the object, after about
a 600 milli-second delay. A internal copy of the string is made.
The boxtype, color and font for the message display can be customized further using the following routines
void fl_set_tooltip_boxtype(int boxtype); void fl_set_tooltip_color(FL_COLOR textcolor, FL_COLOR background); void fl_set_tooltip_font(int style, int size); void fl_set_tooltip_lalign(int align); |
where boxtype
is the backface of the form that displays the text.
The default is FL_BORDER_BOX
. textcolor
and
background
specify the colors of the text string and the
backface. The defaults for these are FL_BLACK
and
FL_YELLOW
respectively. The style
and size
parameters are the font style and size of the text. align
is the
alignment of the text string with respective to the box. The default is
FL_ALIGN_LEFT | FL_ALIGN_INSIDE
.
A word of caution is required. It is possible to change the attributes
of an object at any time. But when the form is already displayed on the
screen some care has to be taken. Whenever attributes change the
system redraws the object. This is fine when drawing the object erases
the old one but this is not always the case. For example, when placing
labels outside the box (not using FL_ALIGN_CENTER
) they are not
correctly erased. It is always possible to force the system to redraw an
object using
void fl_redraw_object(FL_OBJECT *obj); |
When the object is a group it redraws the complete group. To redraw an entire form, use
void fl_redraw_form(FL_FORM *form); |
Use of these routines is normally not necessary and should be kept to an absolute minimum.
Whenever you change an attribute of an object in a visible form the object is redrawn immediately to make the change visible. This can be undesirable when you change a number of attributes of the same object. You only want the changed object to be drawn after the last change. Drawing it after each change will give a flickering effect on the screen. This gets even worse when you e.g. want to hide a few objects. After each object you hide the entire form is redrawn. In addition to the flickering, it is also time consuming. Thus it is more efficient to tell the library to temporarily not redraw the form while changes are being made. This can be done by "freezing" the form. While a form is being frozen it is not redrawn, all changes made are instead buffered internally. Only when you unfreeze the form, all changes made in the meantime are drawn at once. For freezing and unfreezing two calls exist:
void fl_freeze_form(FL_FORM *form); |
and
void fl_unfreeze_form(FL_FORM *form); |
It is a good practice to place multiple changes to the contents of a form always between calls to these two procedures. Further, it is better to complete modifying the attributes of one object before starting work on the next.
Rather than textual labels, it is possible to place symbols like arrows etc. on objects. This is done in the following way:
When the label starts with the character @
no label is drawn but
a particular symbol is drawn instead(3). The rest of the label string indicates the
symbol. A number of pre-defined symbols are available:
-> | Normal arrow pointing to the right. |
<- | Normal arrow pointing to the left. |
> | Triangular arrow pointing to the right. |
< | Triangular arrow pointing to the left. |
>> | Double triangle pointing to the right. |
<< | Double triangle pointing to the left. |
<-> | Arrow pointing left and right. |
->| | A normal arrow with a bar at the end. |
>| | A triangular arrow with a bar at the end. |
--> | A thin arrow pointing to the right. |
= | Three embossed lines. |
arrow | Same as --> . |
returnarrow | <Return> key symbol. |
square | A square. |
circle | A circle. |
line | A horizontal line. |
plus | A plus sign (can be rotated to get a cross). |
UpLine | An embossed line. |
DnLine | An engraved line. |
UpArrow | An embossed arrow. |
DnArrow | An engraved arrow. |
See Fig. 3.6 for how some of them look. See also `symbols.c'.
It is possible to put the symbols in different orientations. When the
symbol name is preceded by a digit 1-9 (not 5) it is rotated like on the
numerical keypad, i.e., 6 indicates no rotation, 9 a rotation of 45
degrees counter-clockwise, 8 a rotation of 90 degrees, etc. Hence the
order is 6, 9, 8, 7, 4, 1, 2, 3. (Just think of the keypad as consisting
of arrow keys.) So to get an arrow that is pointing to the left top use
a label @7->
. To put the symbol in other orientations, put a 0
after the @
, followed by three digits that indicate the angle
(counter-clockwise). E.g. to draw an arrow at an angle of 30 degrees use
label @0030->
.
The symbol will be scaled to fit in the bounding box. When the bounding
box is not square, scaling in the x- and y-directions will be different.
If keeping the aspect ratio is desired, put a sharp (#
)
immediately after the . E.g., @#9->
.
Two additional prefixes, +
and -
, followed by a single
digit, can be used to make small symbol size adjustment. These prefixes
must be either immediately after @
or follow #
. The +
indicates increase the symbol size and - indicates decrease the symbol
size. The digit following the prefixes indicates the increment
(decrement) in pixels. For example, to draw a circle that is 3 pixels
smaller in radius than the default size, use @-3square
.
In addition to using symbols as object labels, symbols can also be drawn directly using
int fl_draw_symbol(const char *symbolname, FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, FL_Color col); |
(the function returns 1
on success and 0
on failure
when the symbol name isn't valid) or indirectly via
fl_drw_text()
. The application program can also add
symbols to the system which it can then use to display symbols on
objects that are not provided by the Forms Library. To add
a symbol, use the call
int fl_add_symbol(const char *name, void (*drawit)(),int sc); |
name
is the name under which the symbol should be known (at most
15 characters), without the leading @
. drawit()
is the
drawing routine that draws the symbol. sc
is reserved and
currently has no meaning. Simply setting it to zero would do.
The routine drawit()
should have the form
void drawit(FL_Coord x, FL_Coord y, FL_Coord w, FL_Coord h, int angle, FL_COLOR col); |
col
is the color in which to draw the symbol. This is the label
color that can be provided and changed by the application program. The
routine should draw the symbol centered inside the box by x
,
y
, w
, h
and rotated from its natural position by
angle
degrees. The draw function can call all types of drawing
routines, including fl_draw_symbol()
.
If the new symbol name is the same as one of the built-ins, the new definition overrides the built-in. Note the the new symbol does not have to be vector graphics, you can use pixmap or whatever in the drawing function.
The symbol handling routines really should be viewed as a means of associating an arbitrary piece of text (the label) with arbitrary graphics, application of which can be quite pleasant given the right tasks.
In some situations you might want to add an object to an existing form. This can be done using the call
FL_FORM *fl_addto_form(FL_FORM *form); |
After this call you can continue adding objects to the form with the
usual functions for adding objects like fl_add_button()
etc. To stop adding objects to the form again use
fl_end_form()
. It is possible to add objects to forms
that are being displayed, but this is not always a good idea because
not everything behaves well (e.g. strange things might happen when a
group is started but not yet finished).
To delete an object from a form simply use
void fl_delete_object(FL_OBJECT *obj); |
It deletes the object from the form it currently belongs to and also
from a group it may belong to. The argument can also be the
pseudo-object starting a group (i.e. the return value of
fl_bgn_group()
) in which case the whole group of objects
will be removed from the form.
The object remains available (except if it's an object that marks the start or end of a group) and can be added again to the same or another form later using
void fl_add_object(FL_FORM *form, FL_OBJECT *obj); |
Normally, this function is used in object classes to add a newly created object to a form. It may not be used for objects representing the start or end of a group.
If the application program does not need an object anymore, it can free the memory used by the object using the call
void fl_free_object(FL_OBJECT *obj); |
After this the object can no longer be used. If you hadn't deleted the object from the form it did belong to before this will happen automatically.
To free the memory used by an entire form use the call
void fl_free_form(FL_FORM *form); |
This will delete and free all the objects on the form and the form itself. A freed form may not be referenced anymore.
Standard color names are listed in a file named `rgb.txt' and usually resides in `/usr/lib/X11/'
If you want a literal
@
character as the first character, escape it with another
@
character.
[ << ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
This document was generated by Peter S Galbraith on May 26, 2010 using texi2html 1.82.