v03i072: xgraph -- graph points and functions, Part03/06
Dan Heller
argv at island.uu.net
Tue Apr 11 10:43:14 AEST 1989
Submitted-by: David Harrison <davidh at ic.berkeley.edu>
Posting-number: Volume 3, Issue 72
Archive-name: xgraph/part03
if `test ! -s ./xgraph-11/xtb/xtb.c`
then
echo "writing ./xgraph-11/xtb/xtb.c"
cat > ./xgraph-11/xtb/xtb.c << '\End\Of\Shar\'
/*
* Mini-Toolbox
*
* David Harrison
* University of California, Berkeley
* 1988, 1989
*
* This file contains routines which implement simple display widgets
* which can be used to construct simple dialog boxes.
* A mini-toolbox has been written here (overkill but I didn't
* want to use any of the standards yet -- they are too unstable).
*/
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include "xtb.h"
#define MAXKEYS 10
#ifdef __STDC__
#define FNPTR(fname, rtn, args) rtn (*fname)args
#include <stdarg.h>
#define VARARGS(func, rtn, args) rtn func args
#else
#define FNPTR(fname, rtn, args) rtn (*fname)()
#include <varargs.h>
#define VARARGS(func, rtn, args) /*VARARGS*/ rtn func(va_alist) va_dcl
#endif
typedef struct h_info {
FNPTR(func, xtb_hret, (XEvent *evt, xtb_data info)); /* Function to call */
xtb_data info; /* Additional info */
};
static Display *t_disp; /* Display */
static int t_scrn; /* Screen */
static unsigned long norm_pix; /* Foreground color */
static unsigned long back_pix; /* Background color */
static XFontStruct *norm_font; /* Normal font */
extern char *malloc();
#define STRDUP(str) (strcpy(malloc((unsigned) (strlen(str)+1)), (str)))
extern char *strcpy();
extern void free();
void xtb_init(disp, scrn, foreground, background, font)
Display *disp; /* Display */
int scrn; /* Screen number */
unsigned long foreground; /* Foreground color */
unsigned long background; /* Background color */
XFontStruct *font; /* Normal font */
/*
* Sets default parameters used by the mini-toolbox.
*/
{
t_disp = disp;
t_scrn = scrn;
norm_pix = foreground;
back_pix = background;
norm_font = font;
}
static GC set_gc(win, fg, bg, font)
Window win; /* For creating GC */
unsigned long fg; /* Foreground pixel */
unsigned long bg; /* Background pixel */
Font font; /* Font */
/*
* Sets and returns the fields listed above in a global graphics context.
* If graphics context does not exist, it is created.
*/
{
static GC t_gc = (GC) 0;
XGCValues gcvals;
unsigned long gcmask;
gcvals.foreground = fg;
gcvals.background = bg;
gcvals.font = font;
gcmask = GCForeground | GCBackground | GCFont;
if (t_gc == (GC) 0) {
t_gc = XCreateGC(t_disp, win, gcmask, &gcvals);
} else {
XChangeGC(t_disp, t_gc, gcmask, &gcvals);
}
return t_gc;
}
static XContext h_context = (XContext) 0;
void xtb_register(win, func, info)
Window win;
FNPTR( func, xtb_hret, (XEvent *evt, xtb_data info) );
xtb_data info;
/*
* Associates the event handling function `func' with the window
* `win'. Additional information `info' will be passed to `func'.
* The routine should return one of the return codes given above.
*/
{
struct h_info *new_info;
if (h_context == (XContext) 0) {
h_context = XUniqueContext();
}
new_info = (struct h_info *) malloc(sizeof(struct h_info));
new_info->func = func;
new_info->info = info;
XSaveContext(t_disp, win, h_context, (caddr_t) new_info);
}
xtb_data xtb_lookup(win)
Window win;
/*
* Returns the associated data with window `win'.
*/
{
xtb_data data;
if (!XFindContext(t_disp, win, h_context, &data)) {
return ((struct h_info *) data)->info;
} else {
return (xtb_data) 0;
}
}
xtb_hret xtb_dispatch(evt)
XEvent *evt;
/*
* Dispatches an event to a handler if its ours. Returns one of
* the return codes given above (XTB_NOTDEF, XTB_HANDLED, or XTB_STOP).
*/
{
struct h_info *info;
if (!XFindContext(t_disp, evt->xany.window, h_context, (caddr_t *) &info)) {
if (info->func) return (*info->func)(evt, info->info);
else return XTB_NOTDEF;
} else return XTB_NOTDEF;
}
int xtb_unregister(win, info)
Window win;
xtb_data *info;
/*
* Removes `win' from the dialog association table. `info' is
* returned to allow the user to delete it (if desired). Returns
* a non-zero status if the association was found and properly deleted.
*/
{
struct h_info *hi;
if (!XFindContext(t_disp, win, h_context, (caddr_t *) &hi)) {
(void) XDeleteContext(t_disp, win, h_context);
*info = hi->info;
free((char *) hi);
return 1;
} else return 0;
}
#define BT_HPAD 3
#define BT_VPAD 2
#define BT_LPAD 3
#define BT_BRDR 1
typedef struct b_info {
FNPTR( func, xtb_hret, (Window win, int state, xtb_data val) );
/* Function to call */
char *text; /* Text of button */
int flag; /* State of button */
int line_y, line_w; /* Entry/Exit line */
xtb_data val; /* User defined info */
};
static void bt_draw(win, ri)
Window win;
struct b_info *ri;
/*
* Draws a button window
*/
{
if (ri->flag) {
XDrawImageString(t_disp, win,
set_gc(win, back_pix, norm_pix, norm_font->fid),
BT_HPAD, BT_VPAD+norm_font->ascent,
ri->text, strlen(ri->text));
} else {
XDrawImageString(t_disp, win,
set_gc(win, norm_pix, back_pix, norm_font->fid),
BT_HPAD, BT_VPAD+norm_font->ascent,
ri->text, strlen(ri->text));
}
}
static void bt_line(win, ri, pix)
Window win;
struct b_info *ri;
unsigned long pix;
/*
* Draws a status line beneath the text to indicate the
* user has moved into the button.
*/
{
XDrawLine(t_disp, win,
set_gc(win, pix, back_pix, norm_font->fid),
BT_HPAD, ri->line_y, BT_HPAD+ri->line_w, ri->line_y);
}
static xtb_hret bt_h(evt, info)
XEvent *evt;
xtb_data info;
/*
* Handles button events.
*/
{
Window win = evt->xany.window;
struct b_info *ri = (struct b_info *) info;
xtb_hret rtn;
switch (evt->type) {
case Expose:
bt_draw(win, ri); rtn = XTB_HANDLED;
break;
case EnterNotify:
bt_line(win, ri, norm_pix); rtn = XTB_HANDLED;
break;
case LeaveNotify:
bt_line(win, ri, back_pix); rtn = XTB_HANDLED;
break;
case ButtonPress:
/* Nothing - just wait for button up */
break;
case ButtonRelease:
rtn = (*ri->func)(win, ri->flag, ri->val);
break;
default:
rtn = XTB_NOTDEF;
}
return rtn;
}
void xtb_bt_new(win, text, func, val, frame)
Window win; /* Parent window */
char *text; /* Text in button */
FNPTR( func, xtb_hret, (Window, int, xtb_data) ); /* Callback */
xtb_data val; /* User data */
xtb_frame *frame; /* Size (RETURN) */
/*
* Makes a new button under `win' with the text `text'. The
* window, size, and position of the button are returned in `frame'.
* The initial position is always (0, 0). When the
* button is pressed, `func' will be called with the button
* window, the current state of the button, and `val'.
* It is up to `func' to change the state of the button (if desired).
* The routine should return XTB_HANDLED normally and XTB_STOP if the
* dialog should stop. The window will be automatically mapped.
*/
{
XCharStruct bb;
struct b_info *info;
int dir, ascent, descent;
XTextExtents(norm_font, text, strlen(text), &dir, &ascent, &descent, &bb);
frame->width = bb.width + 2*BT_HPAD;
frame->height = norm_font->ascent + norm_font->descent + BT_VPAD + BT_LPAD;
frame->x_loc = frame->y_loc = 0;
frame->win = XCreateSimpleWindow(t_disp, win,
frame->x_loc, frame->y_loc,
frame->width, frame->height,
BT_BRDR, norm_pix, back_pix);
XSelectInput(t_disp, frame->win, ExposureMask|
ButtonPressMask|ButtonReleaseMask|
EnterWindowMask|LeaveWindowMask);
info = (struct b_info *) malloc(sizeof(struct b_info));
info->func = func;
info->text = STRDUP(text);
info->flag = 0;
info->val = val;
info->line_y = frame->height - 2;
info->line_w = frame->width - 2*BT_HPAD;
xtb_register(frame->win, bt_h, (xtb_data) info);
XMapWindow(t_disp, frame->win);
frame->width += (2*BT_BRDR);
frame->height += (2*BT_BRDR);
}
int xtb_bt_get(win, stuff)
Window win;
xtb_data *stuff;
/*
* Returns the state of button `win'. If provided, the button
* specific info is returned in `info'.
*/
{
struct b_info *info = (struct b_info *) xtb_lookup(win);
if (stuff) *stuff = info->val;
return info->flag;
}
int xtb_bt_set(win, val, stuff)
Window win;
int val;
xtb_data stuff;
/*
* Changes the value of a button and returns the new state.
* The button is drawn. If set, the button specific info
* will be set to `info'. This doesn't allow you to set
* the state to zero, but that may not be important. The
* change in button appearance will be immediate.
*/
{
struct b_info *info = (struct b_info *) xtb_lookup(win);
info->flag = (val != 0);
if (stuff) info->val = stuff;
bt_draw(win, info);
XFlush(t_disp);
return info->flag;
}
void xtb_bt_del(win, info)
Window win;
xtb_data *info;
/*
* Deletes the button `win' and returns the user defined information
* in `info' for destruction.
*/
{
struct b_info *bi;
if (xtb_unregister(win, (xtb_data *) &bi)) {
*info = bi->val;
free((char *) bi->text);
free((char *) bi);
XDestroyWindow(t_disp, win);
}
}
#define BR_XPAD 2
#define BR_YPAD 2
#define BR_INTER 2
typedef struct br_info {
Window main_win; /* Main button row */
int which_one; /* Which button is on */
int btn_cnt; /* How many buttons */
FNPTR( func, xtb_hret, (Window win, int prev, int this, xtb_data val) );
xtb_data val; /* User data */
Window *btns; /* Button windows */
};
/*ARGSUSED*/
static xtb_hret br_h(win, val, info)
Window win;
int val;
xtb_data info;
/*
* This handles events for button rows. When a button is pressed,
* it turns off the button selected in `which_one' and turns itself
* on.
*/
{
struct br_info *real_info = (struct br_info *) info;
int i, prev;
prev = real_info->which_one;
if ((prev >= 0) && (prev < real_info->btn_cnt)) {
(void) xtb_bt_set(real_info->btns[prev], 0, (xtb_data) 0);
}
for (i = 0; i < real_info->btn_cnt; i++) {
if (win == real_info->btns[i]) {
real_info->which_one = i;
break;
}
}
(void) xtb_bt_set(win, 1, (xtb_data) 0);
/* Callback */
if (real_info->func) {
return (*real_info->func)(real_info->main_win,
prev, real_info->which_one,
real_info->val);
} else {
return XTB_HANDLED;
}
}
void xtb_br_new(win, cnt, lbls, init, func, val, frame)
Window win; /* Parent window */
int cnt; /* Count of buttons */
char *lbls[]; /* Button labels */
int init; /* Initial button */
FNPTR( func, xtb_hret, (Window, int, int, xtb_data) ); /* Callback */
xtb_data val; /* User data */
xtb_frame *frame; /* Returned size */
/*
* This routine makes a new row of buttons in the window `win'
* and returns a frame containing all of these buttons. These
* buttons are designed so that only one of them will be active
* at a time. Initially, button `init' will be activated (if
* init is less than zero, none will be initially activated).
* Whenever a button is pushed, `func' will be called with the
* button row window, the index of the previous button (-1 if
* none), the index of the current button, and the user data, `val'.
* The function is optional (if zero, no function will be called).
* The size of the row is returned in `frame'. The window
* will be automatically mapped. Initially, the window containing
* the buttons will be placed at 0,0 in the parent window.
*/
{
struct br_info *info;
xtb_frame sub_frame;
int i, x, y;
frame->width = frame->height = 0;
frame->x_loc = frame->y_loc = 0;
frame->win = XCreateSimpleWindow(t_disp, win, 0, 0, 1, 1,
0, back_pix, back_pix);
info = (struct br_info *) malloc(sizeof(struct br_info));
info->main_win = frame->win;
info->btns = (Window *) malloc((unsigned) (sizeof(Window) * cnt));
info->btn_cnt = cnt;
info->which_one = init;
info->func = func;
info->val = val;
/* the handler is used simply to get information out */
xtb_register(frame->win, (xtb_hret (*)()) 0, (xtb_data) info);
x = BR_XPAD; y = BR_YPAD;
for (i = 0; i < cnt; i++) {
xtb_bt_new(frame->win, lbls[i], br_h, (xtb_data) info, &sub_frame);
info->btns[i] = sub_frame.win;
XMoveWindow(t_disp, info->btns[i], x, y);
x += (BR_INTER + sub_frame.width);
if (sub_frame.height > frame->height) frame->height = sub_frame.height;
if (i == init) (void) xtb_bt_set(info->btns[i], 1, (xtb_data) 0);
}
frame->width = x - BR_INTER + BR_XPAD;
frame->height += (2 * BR_YPAD);
XResizeWindow(t_disp, frame->win, frame->width, frame->height);
XMapWindow(t_disp, frame->win);
}
int xtb_br_get(win)
Window win;
/*
* This routine returns the index of the currently selected item of
* the button row given by the window `win'. Note: no checking
* is done to make sure `win' is a button row.
*/
{
struct br_info *info = (struct br_info *) xtb_lookup(win);
return info->which_one;
}
void xtb_br_del(win)
Window win;
/*
* Deletes a button row. All resources are reclaimed.
*/
{
struct br_info *info;
int i;
if (xtb_unregister(win, (xtb_data *) &info)) {
for (i = 0; i < info->btn_cnt; i++) {
xtb_bt_del(info->btns[i], (xtb_data *) &info);
}
free((char *) info->btns);
free((char *) info);
XDestroyWindow(t_disp, win);
}
}
/* Text widget */
#define TO_HPAD 1
#define TO_VPAD 1
typedef struct to_info {
char *text; /* Text to display */
XFontStruct *ft; /* Font to use */
};
static void to_draw(win, ri)
Window win;
struct to_info *ri;
/*
* Draws the text for a widget
*/
{
XDrawImageString(t_disp, win,
set_gc(win, norm_pix, back_pix, ri->ft->fid),
TO_HPAD, TO_VPAD+ri->ft->ascent,
ri->text, strlen(ri->text));
}
static xtb_hret to_h(evt, info)
XEvent *evt;
xtb_data info;
/*
* Handles text widget events
*/
{
Window win = evt->xany.window;
struct to_info *ri = (struct to_info *) info;
switch (evt->type) {
case Expose:
to_draw(win, ri);
return XTB_HANDLED;
default:
return XTB_NOTDEF;
}
}
void xtb_to_new(win, text, ft, frame)
Window win; /* Parent window */
char *text; /* Text */
XFontStruct *ft; /* Font to use */
xtb_frame *frame; /* Returned size */
/*
* Makes a new text widget under `win' with the text `text'.
* The size of the widget is returned in `w' and `h'. The
* window is created and mapped at 0,0 in `win'. The font
* used for the text is given in `ft'.
*/
{
struct to_info *info;
XCharStruct bb;
int dir, ascent, descent;
XTextExtents(ft, text, strlen(text), &dir, &ascent, &descent, &bb);
frame->width = bb.width + 2*TO_HPAD;
frame->height = bb.ascent + bb.descent + (2 * TO_VPAD);
frame->x_loc = frame->y_loc = 0;
frame->win = XCreateSimpleWindow(t_disp, win, 0, 0,
frame->width, frame->height, 0,
back_pix, back_pix);
XSelectInput(t_disp, frame->win, ExposureMask);
info = (struct to_info *) malloc(sizeof(struct to_info));
info->text = STRDUP(text);
info->ft = ft;
xtb_register(frame->win, to_h, (xtb_data) info);
XMapWindow(t_disp, frame->win);
}
void xtb_to_del(win)
Window win;
/*
* Deletes an output only text widget.
*/
{
struct to_info *info;
if (xtb_unregister(win, (xtb_data *) &info)) {
free((char *) info->text);
free((char *) info);
XDestroyWindow(t_disp, win);
}
}
/*
* Input text widget
*/
#define TI_HPAD 2
#define TI_VPAD 2
#define TI_LPAD 3
#define TI_BRDR 2
#define TI_CRSP 1
typedef struct ti_info {
FNPTR( func, xtb_hret, (Window win, int ch, char *textcopy, xtb_data *val) );
/* Function to call */
int maxlen; /* Maximum characters */
int curidx; /* Current index pos */
int curxval; /* Current draw loc */
char text[MAXCHBUF]; /* Current text array */
int line_y, line_w; /* Entry/Exit line */
int focus_flag; /* If on, we have focus */
xtb_data val; /* User info */
};
static int text_width(font, str, len)
XFontStruct *font; /* What font */
char *str; /* Character array */
int len; /* Length of array */
/*
* Returns the width of a string using XTextExtents.
*/
{
XCharStruct bb;
int dir, ascent, descent;
XTextExtents(font, str, len, &dir, &ascent, &descent, &bb);
return bb.width;
}
static void ti_cursor_on(win, ri)
Window win;
struct ti_info *ri;
/*
* Draws the cursor for the window. Uses pixel `pix'.
*/
{
XFillRectangle(t_disp, win,
set_gc(win, norm_pix, back_pix, norm_font->fid),
ri->curxval + TI_HPAD + TI_CRSP, TI_VPAD,
(ri->focus_flag ? 2 : 1),
norm_font->ascent + norm_font->descent - 1);
}
static void ti_cursor_off(win, ri)
Window win;
struct ti_info *ri;
/*
* Draws the cursor for the window. Uses pixel `pix'.
*/
{
XFillRectangle(t_disp, win,
set_gc(win, back_pix, back_pix, norm_font->fid),
ri->curxval + TI_HPAD + TI_CRSP, TI_VPAD,
(ri->focus_flag ? 2 : 1),
norm_font->ascent + norm_font->descent - 1);
}
static void ti_draw(win, ri, c_flag)
Window win;
struct ti_info *ri;
int c_flag;
/*
* Draws the indicated text widget. This includes drawing the
* text and cursor. If `c_flag' is set, the window will
* be cleared first.
*/
{
if (c_flag) XClearWindow(t_disp, win);
/* Text */
XDrawImageString(t_disp, win,
set_gc(win, norm_pix, back_pix, norm_font->fid),
TI_HPAD, TI_VPAD+norm_font->ascent,
ri->text, strlen(ri->text));
/* Cursor */
ti_cursor_on(win, ri);
}
static void ti_line(win, ri, pix)
Window win;
struct ti_info *ri;
unsigned long pix;
/*
* Draws a status line beneath the text in a text widget to indicate
* the user has moved into the text field.
*/
{
XDrawLine(t_disp, win,
set_gc(win, pix, back_pix, norm_font->fid),
TI_HPAD, ri->line_y, TI_HPAD+ri->line_w, ri->line_y);
}
/* For debugging */
focus_evt(evt)
XEvent *evt;
{
switch (evt->xfocus.mode) {
case NotifyNormal:
printf("NotifyNormal");
break;
case NotifyGrab:
printf("NotifyGrab");
break;
case NotifyUngrab:
printf("NotifyUngrab");
break;
}
printf(", detail = ");
switch (evt->xfocus.detail) {
case NotifyAncestor:
printf("NotifyAncestor");
break;
case NotifyVirtual:
printf("NotifyVirtual");
break;
case NotifyInferior:
printf("NotifyInferior");
break;
case NotifyNonlinear:
printf("NotifyNonLinear");
break;
case NotifyNonlinearVirtual:
printf("NotifyNonLinearVirtual");
break;
case NotifyPointer:
printf("NotifyPointer");
break;
case NotifyPointerRoot:
printf("NotifyPointerRoot");
break;
case NotifyDetailNone:
printf("NotifyDetailNone");
break;
}
printf("\n");
}
static xtb_hret ti_h(evt, info)
XEvent *evt;
xtb_data info;
/*
* Handles text input events.
*/
{
Window win = evt->xany.window;
struct ti_info *ri = (struct ti_info *) info;
char keys[MAXKEYS], textcopy[MAXCHBUF];
xtb_hret rtn;
int nbytes, i;
switch (evt->type) {
case Expose:
ti_draw(win, ri, 0); rtn = XTB_HANDLED;
break;
case KeyPress:
nbytes = XLookupString(&evt->xkey, keys, MAXKEYS,
(KeySym *) 0, (XComposeStatus *) 0);
for (i = 0; i < nbytes; i++) {
(void) strcpy(textcopy, ri->text);
if ((rtn = (*ri->func)(win, (int) keys[i],
textcopy, ri->val)) == XTB_STOP)
break;
}
break;
case FocusIn:
focus_evt(evt);
if (evt->xfocus.detail != NotifyPointer) {
ti_cursor_off(win, ri);
ri->focus_flag = 1;
ti_cursor_on(win, ri);
}
break;
case FocusOut:
focus_evt(evt);
if (evt->xfocus.detail != NotifyPointer) {
ti_cursor_off(win, ri);
ri->focus_flag = 0;
ti_cursor_on(win, ri);
}
break;
case EnterNotify:
ti_line(win, ri, norm_pix); rtn = XTB_HANDLED;
break;
case LeaveNotify:
ti_line(win, ri, back_pix); rtn = XTB_HANDLED;
break;
case ButtonPress:
/* Wait for release */
break;
case ButtonRelease:
/* Set input focus */
XSetInputFocus(t_disp, win, RevertToParent, CurrentTime);
break;
default:
rtn = XTB_NOTDEF;
break;
}
return rtn;
}
void xtb_ti_new(win, text, maxchar, func, val, frame)
Window win; /* Parent window */
char *text; /* Initial text */
int maxchar; /* Maximum characters */
FNPTR( func, xtb_hret, (Window, int, char *, xtb_data *) ); /* Callback */
xtb_data val; /* User data */
xtb_frame *frame; /* Returned size */
/*
* This routine creates a new editable text widget under `win'
* with the initial text `text'. The widget contains only
* one line of text which cannot exceed `maxchar' characters.
* The size of the widget is returned in `frame'. Each
* time a key is pressed in the window, `func' will be called
* with the window, the character, a copy of the text, and `val'.
* The state of the widget can be changed by the routines below.
* May set window to zero if the maximum overall character width
* (MAXCHBUF) is exceeded.
*/
{
struct ti_info *info;
if (maxchar >= MAXCHBUF) {
frame->win = (Window) 0;
return;
}
frame->width = XTextWidth(norm_font, "8", 1) * maxchar + 2*TI_HPAD;
frame->height = norm_font->ascent + norm_font->descent + TI_VPAD + TI_LPAD;
frame->x_loc = frame->y_loc = 0;
frame->win = XCreateSimpleWindow(t_disp, win, 0, 0,
frame->width, frame->height, TI_BRDR,
norm_pix, back_pix);
XSelectInput(t_disp, frame->win, ExposureMask|KeyPressMask|
EnterWindowMask|LeaveWindowMask|
FocusChangeMask|ButtonPressMask|
ButtonReleaseMask);
info = (struct ti_info *) malloc(sizeof(struct ti_info));
info->func = func;
info->val = val;
info->maxlen = maxchar;
if (text) (void) strcpy(info->text, text);
else info->text[0] = '\0';
info->curidx = strlen(info->text);
info->curxval = text_width(norm_font, info->text, info->curidx);
info->line_y = frame->height - 2;
info->line_w = frame->width - 2 * TI_HPAD;
info->focus_flag = 0;
xtb_register(frame->win, ti_h, (xtb_data) info);
XMapWindow(t_disp, frame->win);
frame->width += (2 * TI_BRDR);
frame->height += (2 * TI_BRDR);
}
void xtb_ti_get(win, text, val)
Window win; /* Widget widnow */
char text[MAXCHBUF]; /* Filled in text */
xtb_data *val; /* User info */
/*
* This routine returns the information associated with text
* widget `win'. The text is filled into the passed buffer
* `text' which should be MAXCHBUF characters in size. If
* `val' is non-zero, the user supplied info is returned there.
*/
{
struct ti_info *info = (struct ti_info *) xtb_lookup(win);
if (val) *val = info->val;
(void) strcpy(text, info->text);
}
int xtb_ti_set(win, text, val)
Window win; /* Widget window */
char *text; /* Replacement text */
xtb_data val; /* User info */
/*
* This routine sets the text of a text widget. The widget
* will be redrawn. Note: for incremental changes, ti_ins and
* ti_dch should be used. If `val' is non-zero, it will replace
* the user information for the widget. The widget is redrawn.
* Will return zero if `text' is too long.
*/
{
struct ti_info *info = (struct ti_info *) xtb_lookup(win);
int newlen;
if (text) {
if ((newlen = strlen(text)) >= info->maxlen) return 0;
} else {
newlen = 0;
}
info->curidx = newlen;
if (text) (void) strcpy(info->text, text);
else info->text[0] = '\0';
info->curxval = text_width(norm_font, info->text, info->curidx);
if (val) info->val = val;
ti_draw(win, info, 1);
return 1;
}
int xtb_ti_ins(win, ch)
Window win; /* Widget window */
int ch; /* Character */
/*
* Inserts the character `ch' onto the end of the text for `win'.
* Will return zero if there isn't any more room left. Does
* all appropriate display updates.
*/
{
struct ti_info *info = (struct ti_info *) xtb_lookup(win);
char lstr[1];
if (info->curidx >= info->maxlen-1) return 0;
info->text[info->curidx] = ch;
info->text[info->curidx+1] = '\0';
/* Turn off cursor */
ti_cursor_off(win, info);
/* Text */
lstr[0] = (char) ch;
XDrawImageString(t_disp, win,
set_gc(win, norm_pix, back_pix, norm_font->fid),
info->curxval+TI_HPAD, TI_VPAD+norm_font->ascent,
lstr, 1);
info->curidx += 1;
info->curxval += text_width(norm_font, lstr, 1);
ti_cursor_on(win, info);
return 1;
}
int xtb_ti_dch(win)
Window win; /* Widget window */
/*
* Deletes the character at the end of the text for `win'. Will
* return zero if there aren't any characters to delete. Does
* all appropriate display updates.
*/
{
struct ti_info *info = (struct ti_info *) xtb_lookup(win);
int chw;
if (info->curidx == 0) return 0;
/* Wipe out cursor */
ti_cursor_off(win, info);
info->curidx -= 1;
chw = text_width(norm_font, &(info->text[info->curidx]), 1);
info->curxval -= chw;
/* Wipe out character */
XClearArea(t_disp, win, info->curxval+TI_HPAD, TI_VPAD,
(unsigned int) chw + 1,
(unsigned int) norm_font->ascent + norm_font->descent,
False);
info->text[info->curidx] = '\0';
ti_cursor_on(win, info);
return 1;
}
void xtb_ti_del(win, info)
Window win;
xtb_data *info;
/*
* Deletes an input text widget. User defined data is returned in `info'.
*/
{
struct ti_info *ti;
if (xtb_unregister(win, (xtb_data *) &ti)) {
*info = ti->val;
free((char *) ti);
XDestroyWindow(t_disp, win);
}
}
/*
* Simple colored output frame - usually used for drawing lines
*/
void xtb_bk_new(win, width, height, frame)
Window win; /* Parent window */
unsigned width, height; /* Size */
xtb_frame *frame; /* Returned size */
/*
* This routine creates a new frame that displays a block
* of color whose size is given by `width' and `height'.
* It is usually used to draw lines. No user interaction
* is defined for the frame. The color used is the default
* foreground color set in xtb_init().
*/
{
frame->x_loc = frame->y_loc = 0;
frame->width = width;
frame->height = height;
frame->win = XCreateSimpleWindow(t_disp, win,
frame->x_loc, frame->y_loc,
frame->width, frame->height,
0, norm_pix, norm_pix);
XMapWindow(t_disp, frame->win);
}
void xtb_bk_del(win)
Window win;
/*
* Deletes a block frame.
*/
{
XDestroyWindow(t_disp, win);
}
/*
* Formatting support
*/
#define ERROR(msg) printf("%s\n", msg); abort();
xtb_fmt *xtb_w(w)
xtb_frame *w;
/*
* Returns formatting structure for a widget.
*/
{
xtb_fmt *ret;
ret = (xtb_fmt *) malloc((unsigned) sizeof(xtb_fmt));
ret->wid.type = W_TYPE;
ret->wid.w = w;
return ret;
}
VARARGS(xtb_hort, xtb_fmt *, (xtb_just just, int padding, int interspace, ...))
/*
* Builds a horizontal structure
*/
{
va_list ap;
xtb_fmt *ret, *val;
#ifdef __STDC__
va_start(ap, interspace);
#else
xtb_just just;
int padding, interspace;
va_start(ap);
just = va_arg(ap, xtb_just);
padding = va_arg(ap, int);
interspace = va_arg(ap, int);
#endif
ret = (xtb_fmt *) malloc((unsigned) sizeof(xtb_fmt));
ret->align.type = A_TYPE;
ret->align.dir = HORIZONTAL;
ret->align.just = just;
ret->align.padding = padding;
ret->align.interspace = interspace;
/* Build array of incoming xtb_fmt structures */
ret->align.ni = 0;
while ((val = va_arg(ap, xtb_fmt *)) != (xtb_fmt *) 0) {
if (ret->align.ni < MAX_BRANCH) {
ret->align.items[ret->align.ni] = val;
ret->align.ni++;
} else {
ERROR("too many branches\n");
}
}
return ret;
}
VARARGS(xtb_vert, xtb_fmt *, (xtb_just just, int padding, int interspace, ...))
/*
* Builds a vertical structure
*/
{
va_list ap;
xtb_fmt *ret, *val;
#ifdef __STDC__
va_start(ap, interspace);
#else
xtb_just just;
int padding, interspace;
va_start(ap);
just = va_arg(ap, xtb_just);
padding = va_arg(ap, int);
interspace = va_arg(ap, int);
#endif
ret = (xtb_fmt *) malloc((unsigned) sizeof(xtb_fmt));
ret->align.type = A_TYPE;
ret->align.dir = VERTICAL;
ret->align.just = just;
ret->align.padding = padding;
ret->align.interspace = interspace;
/* Build array of incoming xtb_fmt structures */
ret->align.ni = 0;
while ((val = va_arg(ap, xtb_fmt *)) != (xtb_fmt *) 0) {
if (ret->align.ni < MAX_BRANCH) {
ret->align.items[ret->align.ni] = val;
ret->align.ni++;
} else {
ERROR("too many branches\n");
}
}
return ret;
}
static void xtb_fmt_setpos(def, x, y)
xtb_fmt *def;
int x, y;
/*
* Sets all position fields of widgets in `def' to x,y.
*/
{
int i;
switch (def->type) {
case W_TYPE:
def->wid.w->x_loc = x;
def->wid.w->y_loc = y;
break;
case A_TYPE:
for (i = 0; i < def->align.ni; i++) {
xtb_fmt_setpos(def->align.items[i], x, y);
}
break;
default:
ERROR("bad type");
}
}
static void xtb_fmt_addpos(def, x, y)
xtb_fmt *def;
int x, y;
/*
* Adds the offset specified to all position fields of widgets in `def'.
*/
{
int i;
switch (def->type) {
case W_TYPE:
def->wid.w->x_loc += x;
def->wid.w->y_loc += y;
break;
case A_TYPE:
for (i = 0; i < def->align.ni; i++) {
xtb_fmt_addpos(def->align.items[i], x, y);
}
break;
default:
ERROR("bad type");
}
}
static void xtb_fmt_hort(nd, defs, widths, heights, just, pad, inter, rw, rh)
int nd; /* Number of children */
xtb_fmt *defs[]; /* Definitions themselves */
int widths[]; /* Widths of children */
int heights[]; /* Heights of children */
xtb_just just; /* Justification */
int pad, inter; /* Padding and interspace */
int *rw, *rh; /* Returned size */
/*
* Formats items horizontally subject to the widths and heights
* of the items passed.
*/
{
int i;
int max_height = 0;
int tot_width = 0;
int xspot;
/* Find parameters */
for (i = 0; i < nd; i++) {
if (heights[i] > max_height) max_height = heights[i];
tot_width += widths[i];
}
/* Place items -- assumes center justification */
xspot = pad;
for (i = 0; i < nd; i++) {
switch (just) {
case XTB_TOP:
xtb_fmt_addpos(defs[i], xspot, pad);
break;
case XTB_BOTTOM:
xtb_fmt_addpos(defs[i], xspot, max_height - heights[i] + pad);
break;
case XTB_CENTER:
default:
/* Everyone else center */
xtb_fmt_addpos(defs[i], xspot, (max_height - heights[i])/2 + pad);
break;
}
xspot += (widths[i] + inter);
}
/* Figure out resulting size */
*rw = tot_width + (nd-1)*inter + (2 * pad);
*rh = max_height + (2 * pad);
}
static void xtb_fmt_vert(nd, defs, widths, heights, just, pad, inter, rw, rh)
int nd; /* Number of children */
xtb_fmt *defs[]; /* Definitions themselves */
int widths[]; /* Widths of children */
int heights[]; /* Heights of children */
xtb_just just; /* Justification */
int pad, inter; /* Padding and interspace */
int *rw, *rh; /* Returned size */
/*
* Formats items vertically subject to the widths and heights
* of the items passed.
*/
{
int i;
int max_width = 0;
int tot_height = 0;
int yspot;
/* Find parameters */
for (i = 0; i < nd; i++) {
if (widths[i] > max_width) max_width = widths[i];
tot_height += heights[i];
}
/* Place items -- assumes center justification */
yspot = pad;
for (i = 0; i < nd; i++) {
switch (just) {
case XTB_LEFT:
xtb_fmt_addpos(defs[i], pad, yspot);
break;
case XTB_RIGHT:
xtb_fmt_addpos(defs[i], max_width - widths[i] + pad, yspot);
break;
case XTB_CENTER:
default:
/* Everyone else center */
xtb_fmt_addpos(defs[i], (max_width - widths[i])/2 + pad, yspot);
break;
}
yspot += (heights[i] + inter);
}
/* Figure out resulting size */
*rw = max_width + (2 * pad);
*rh = tot_height + (nd-1)*inter + (2 * pad);
}
static void xtb_fmt_top(def, w, h)
xtb_fmt *def;
unsigned *w, *h;
/*
* Recursive portion of formatter
*/
{
unsigned widths[MAX_BRANCH];
unsigned heights[MAX_BRANCH];
int i;
switch (def->type) {
case A_TYPE:
/* Formatting directive */
/* place children and determine sizes */
for (i = 0; i < def->align.ni; i++) {
xtb_fmt_top(def->align.items[i], &(widths[i]), &(heights[i]));
}
/* now format based on direction */
switch (def->align.dir) {
case HORIZONTAL:
xtb_fmt_hort(def->align.ni, def->align.items, widths, heights,
def->align.just, def->align.padding,
def->align.interspace, w, h);
break;
case VERTICAL:
xtb_fmt_vert(def->align.ni, def->align.items, widths, heights,
def->align.just, def->align.padding,
def->align.interspace, w, h);
break;
default:
ERROR("bad direction");
}
break;
case W_TYPE:
/* Simple widget - return size */
*w = def->wid.w->width;
*h = def->wid.w->height;
break;
default:
ERROR("bad type");
}
}
xtb_fmt *xtb_fmt_do(def, w, h)
xtb_fmt *def;
unsigned *w, *h;
/*
* Actually does formatting
*/
{
/* First zero out all positions */
xtb_fmt_setpos(def, 0, 0);
/* Now call recursive portion */
xtb_fmt_top(def, w, h);
return def;
}
void xtb_fmt_free(def)
xtb_fmt *def;
/*
* Frees resources associated with formatting routines
*/
{
int i;
if (def->type == A_TYPE) {
for (i = 0; i < def->align.ni; i++) {
xtb_fmt_free(def->align.items[i]);
}
}
free((char *) def);
}
void xtb_mv_frames(nf, frames)
int nf; /* Number of frames */
xtb_frame frames[]; /* Array of frames */
/*
* Moves frames to the location indicated in the frame
* structure for each item.
*/
{
int i;
for (i = 0; i < nf; i++) {
XMoveWindow(t_disp, frames[i].win, frames[i].x_loc, frames[i].y_loc);
}
}
\End\Of\Shar\
else
echo "will not over write ./xgraph-11/xtb/xtb.c"
fi
echo "Finished archive 3 of 6"
More information about the Comp.sources.x
mailing list