Next: Programming Model, Previous: Naming Conventions, Up: Part I Getting Started [Contents][Index]
Before using forms for interaction with the user you first have to define them. Next you can display them and perform interaction with them. Both stages are simple. Before explaining all the details let us first look at some examples. A very simple form definition would look as
FL_FORM *simpleform; simpleform = fl_bgn_form(FL_UP_BOX, 230, 160); fl_add_button(FL_NORMAL_BUTTON, 40, 50, 150, 60, "Push Me"); fl_end_form();
The first line indicates the start of the form definition.
simpleform will later be used to identify the form. The type of
the form is FL_UP_BOX. This means that the background of the
form is a raised box that looks like it is coming out of the screen.
The form has a size of 230 by 160 pixels. Next we add a button to the
form. The type of the button is FL_NORMAL_BUTTON which will be
explained below in detail. It is positioned in the form by virtue of
the button geometry supplied and has "Push Me" as its label. After
having defined the form we can display it using the call
fl_show_form(simpleform, FL_PLACE_MOUSE, FL_NOBORDER,
             "SimpleForm");
 
This will show the form on the screen at the mouse position. (The third argument indicates whether the form gets window manager’s decoration and the fourth is the window title.)
Next we give the control over the interaction to the Forms Library’s main event loop by calling
fl_do_forms();
This will handle interaction with the form until you press and release the button with the mouse, at which moment control is returned to the program. Now the form can be removed from the screen (and have its associated window destroyed) using
fl_hide_form(simpleform);
The complete program is given in the file pushme.c in the subdirectory demos. All demonstration programs can be found in this directory. Studying them is a good way of learning how the library works.
Compile and run it to see the effect. To compile a program using the Forms Library use the following command or something similar
cc -o pushme pushme.c -lforms
Please note that linking against the Forms library requires some
other libraries to be istalled, at least the X11 and the
Xpm library. Some applications may also require the JPEG
and/or the GL library. These libraries don’t need to be
specified explicitely in the linker command but must be available
since the Forms library depends on them. If not installed contact
your systems administrator.
This simple example is, of course, of little use. Let us look at a slightly more complicated one (the program can be found in yesno.c.)
#include <forms.h>
int main(int argc, char *argv[]) {
    FL_FORM *form;
    FL_OBJECT *yes,
              *no,
              *but;
    fl_initialize(&argc, argv, "FormDemo", 0, 0);
    form = fl_bgn_form(FL_UP_BOX, 320, 120);
    fl_add_box(FL_NO_BOX, 160, 40, 0, 0, "Do you want to Quit?");
    yes = fl_add_button(FL_NORMAL_BUTTON, 40, 70, 80, 30, "Yes");
    no  = fl_add_button(FL_NORMAL_BUTTON, 200, 70, 80, 30, "No");
    fl_end_form();
    fl_show_form(form, FL_PLACE_MOUSE, FL_TRANSIENT, "Question");
    while (1) {
        if (fl_do_forms() == yes)
        {
            printf("Yes is pushed\n");
            break;
        }
        else
            printf("No is pushed\n");
    }
    fl_finish();
    return 0;
}
It creates a form with a simple text and two buttons. After displaying
the form fl_do_forms() is called. This routine returns
the object being pushed. Simply checking whether this is object
yes or no determines whether we should quit.
 
As you see, the program starts by calling the routine
fl_initialize(). This routine should be called before any
other calls to the library are made (except for
fl_set_defaults()). One of the things this routine does
is to establish a connection to the X server and initialize a resource
database used by the X resource manager. It also does many other
things, such as parsing command line options and initializing internal
Forms Library structures. For now, it suffices to know that by calling
this routine, a program automatically recognizes the following command
line options
| Option | Value type | Meaning | 
|---|---|---|
| -displayhost:dpy | string | Remote host | 
| -nameappname | string | change application name | 
| -visualclass | string | TrueColor, PseudoColor etc. | 
| -depthdepth | integer | Preferred visual depth | 
| -private | none | Force a private colormap | 
| -shared | none | Always share colormap | 
| -stdcmap | none | Use standard colormap | 
| -fldebuglevel | integer | Print some debug information | 
| -flhelp | none | Print out these options | 
| -sync | none | Force synchronous mode | 
Note that the executable name argv[0] should not contain period
or *. See Overview of
Main Functions, for further details. The above program can in fact be
made a lot simpler, using the goodies described in Goodies. You can simply write:
while (!fl_show_question("Do you want to Quit?", 0))
    /* empty */ ;
Except printing out a message telling which button was pressed it will have exactly the same effect.
The above program only shows one of the event handling methods
provided by the library. The direct method of event handling shown is
appropriate for simple programs. But, obviously, already for a program
with just a few nmore objects it would become rather tedious to have
to check each time fl_do_forms() returns each of those
objects to find out which of them was responsible and react
accordingly. Utilizing object callback functions is then typically
much easier and thus os strongly recommended.
We demonstrate the use of object callbacks using the previous example with some modifications so that event processing via callbacks is utilized. It is recommended and also typical of a good XForms application to separate the UI components and the application program itself. Typically the UI components are generated by the bundled GUI builder and the application program consists mostly of callbacks and some glue code that combines the UI and the program.
To use callbacks, a typical procedure would be to define all the
callback functions first, then register them with the system using
fl_set_object_callback(). After the form is realized
(shown), control is handed to Forms Library’s main loop
fl_do_forms(), which responds to user events indefinitely
and never returns.
After modifications are made to utilize object callbacks, the simple question example looks as follows:
#include <stdio.h>
#include <stdlib.h>
#include <forms.h>
void yes_callback(FL_OBJECT *obj, long user_data) {
    printf("Yes is pushed\n");
    fl_finish();
    exit(0);
}
void no_callback(FL_OBJECT *obj, long user_data) {
    printf("No is pushed\n");
}
int main(int argc, char *argv[]) {
    FL_FORM *form;
    FL_OBJECT *obj;
    fl_initialize(&argc, argv, "FormDemo", 0, 0);
    form = fl_bgn_form(FL_UP_BOX, 320, 120);
    fl_add_box(FL_NO_BOX, 160, 40, 0, 0, "Do you want to Quit?");
    obj = fl_add_button(FL_NORMAL_BUTTON, 40, 70, 80, 30,"Yes");
    fl_set_object_callback(obj, yes_callback, 0);
    obj = fl_add_button(FL_NORMAL_BUTTON, 200, 70, 80, 30,"No");
    fl_set_object_callback(obj, no_callback, 0);
    fl_end_form();
    fl_show_form(form, FL_PLACE_MOUSE, FL_TRANSIENT, "Question");
    fl_do_forms();
    return 0;
}
In this example, callback routines for both the yes and no buttons are
first defined. Then they are registered with the system using
fl_set_object_callback(). After the form is shown, the
event handling is again handed to the main loop in Forms Library via
fl_do_forms(). In this case, whenever the buttons are
pushed, the callback routine is invoked with the object being pushed
as the first argument to the callback function, and
fl_do_forms() never returns.
You might also have noticed that in this example both buttons are made anonymous, that is, it is not possible to reference the buttons outside of the creating routine. This is often desirable when callback functions are bound to objects as the objects themselves will not be referenced except as callback arguments. By creating anonymous objects a program avoids littering itself with useless identifiers.
The callback model presented above is the preferred way of interaction for typical programs and it is strongly recommended that programs using XForms be coded using object callbacks.
Next: Programming Model, Previous: Naming Conventions, Up: Part I Getting Started [Contents][Index]