ngspice/xgraph/xtb.c

1619 lines
37 KiB
C
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*LINTLIBRARY*/
/*
* 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 <stdio.h> /* for printf() */
#include <X11/Xos.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <string.h> /* for strcpy() and strlen() */
#include "xtb.h"
extern void abort();
#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) /*VARARGS1*/ rtn func(va_alist) va_dcl
#endif
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 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;
}
#define gray_width 32
#define gray_height 32
static char gray_bits[] =
{
0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55,
0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa,
0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55,
0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa,
0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55,
0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa,
0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55,
0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa,
0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55,
0xaa, 0xaa, 0xaa, 0xaa, 0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa,
0x55, 0x55, 0x55, 0x55, 0xaa, 0xaa, 0xaa, 0xaa};
static Pixmap
make_stipple(disp, able, bits, width, height)
Display *disp;
Drawable able;
char *bits;
unsigned int width,
height;
{
unsigned int w,
h;
Pixmap stipple;
int bitmap_pad;
GC image_gc;
XGCValues image_vals;
XImage *stip_image;
if (!XQueryBestStipple(disp, able, width, height, &w, &h)) {
/* Can't query best stipple */
return (Pixmap) 0;
}
if ((w > width) || (h > height)) {
/* Given pattern is too small */
return (Pixmap) 0;
}
if (!(stipple = XCreatePixmap(disp, able, w, h, 1))) {
/* Can't create pixmap image */
return (Pixmap) 0;
}
bitmap_pad = 8;
stip_image = XCreateImage(disp, DefaultVisual(disp, DefaultScreen(disp)),
1, XYPixmap, 0, bits, width, height,
bitmap_pad, 0);
if (!stip_image)
return (Pixmap) 0;
image_gc = XCreateGC(disp, stipple, (unsigned long) 0, &image_vals);
XPutImage(disp, stipple, image_gc, stip_image, 0, 0, 0, 0, w, h);
return stipple;
}
static Pixmap
gray_map(win)
Window win;
/*
* Returns a gray pixmap suitable for stipple fill.
*/
{
static Pixmap map = (Pixmap) 0;
if (!map) {
map = make_stipple(t_disp, win, gray_bits, gray_width, gray_height);
}
return map;
}
static GC
set_gc(win, fg, bg, font, gray_p)
Window win; /* For creating GC */
unsigned long fg; /* Foreground pixel */
unsigned long bg; /* Background pixel */
Font font; /* Font */
int gray_p; /* Gray if set */
/*
* 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;
gcvals.stipple = gray_map(win);
gcvals.fill_style = (gray_p ? FillStippled : FillSolid);
gcmask = GCForeground | GCBackground | GCFont | GCFillStyle | GCStipple;
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, (XPointer*) &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
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 na; /* Non-zero if not active */
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, 0),
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, 0),
BT_HPAD, BT_VPAD + norm_font->ascent,
ri->text, strlen(ri->text));
}
if (ri->na) {
Window root;
int x,
y;
unsigned int w,
h,
b,
d;
XGetGeometry(t_disp, win, &root, &x, &y, &w, &h, &b, &d);
XFillRectangle(t_disp, win,
set_gc(win, (ri->flag ? norm_pix : back_pix),
back_pix, norm_font->fid, 1),
0, 0, w, h);
}
}
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, ri->na),
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 = 0;
switch (evt->type) {
case Expose:
bt_draw(win, ri);
rtn = XTB_HANDLED;
break;
case EnterNotify:
if (!ri->na)
bt_line(win, ri, norm_pix);
rtn = XTB_HANDLED;
break;
case LeaveNotify:
if (!ri->na)
bt_line(win, ri, back_pix);
rtn = XTB_HANDLED;
break;
case ButtonPress:
/* Nothing - just wait for button up */
rtn = XTB_HANDLED;
break;
case ButtonRelease:
if (!ri->na)
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->na = 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, na)
Window win;
xtb_data *stuff;
int *na;
/*
* Returns the state of button `win'. If provided, the button
* specific info is returned in `info' and the active state
* of the button is returned in `na'.
*/
{
struct b_info *info = (struct b_info *) xtb_lookup(win);
if (stuff)
*stuff = info->val;
if (na)
*na = info->na;
return info->flag;
}
int
xtb_bt_set(win, val, stuff, na)
Window win;
int val;
xtb_data stuff;
int na;
/*
* 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. If
* `na' is non-zero, the button will be deactivated (no
* activity will be allowed). The change in button appearance
* will be immediate.
*/
{
struct b_info *info = (struct b_info *) xtb_lookup(win);
info->flag = (val != 0);
info->na = (na != 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
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, 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, 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, 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
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, 0),
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;
frame->width = XTextWidth(ft, text, strlen(text)) + 2 * TO_HPAD;
frame->height = ft->ascent + ft->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)) {
if (info->text)
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
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, 0),
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, 0),
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, 0),
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, 0),
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 = 0;
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, 0),
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 */
unsigned int widths[]; /* Widths of children */
unsigned int heights[]; /* Heights of children */
xtb_just just; /* Justification */
int pad,
inter; /* Padding and interspace */
unsigned 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 */
unsigned int widths[]; /* Widths of children */
unsigned int heights[]; /* Heights of children */
xtb_just just; /* Justification */
int pad,
inter; /* Padding and interspace */
unsigned 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");
}
}
#ifdef DEBUG
static void
xtb_fmt_debug(def)
xtb_fmt *def;
/*
* Dumps formatting structure for debugging purposes.
*/
{
int i;
switch (def->type) {
case W_TYPE:
printf("%d %d %d %d\n",
def->wid.w->x_loc, def->wid.w->y_loc,
def->wid.w->width, def->wid.w->height);
break;
case A_TYPE:
for (i = 0; i < def->align.ni; i++) {
xtb_fmt_debug(def->align.items[i]);
}
break;
default:
ERROR("bad type");
}
}
#endif
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);
#ifdef DEBUG
xtb_fmt_debug(def);
#endif
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);
}
}