/* viDOOM - level editor for DOOM Copyright (C) 2000 Ian Cowburn (ianc@noddybox.demon.co.uk) This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ------------------------------------------------------------------------- Platform specific GUI type routines */ static const char rcs_id[]="$Id$"; #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include "platgui.h" #include "gfx.h" #include "gui.h" #include "mem.h" #include "debug.h" #define COLDEP 16 #define ACOL(c) (makecol_depth(COLDEP,((c)&0xff0000)>>16, \ ((c)&0xff00)>>8, \ ((c)&0xff))) #define DIALOG_STRLEN 16 #define SMALL_BEVEL 1 /* Bevels on radio and flag boxes */ #define BEVEL 2 /* All other bevels */ #define TICK 2 /* ---------------------------------------- VARS */ static int SCRW,SCRH; /* Global picklist data and the picklist dialog */ #define PL_BORDER 0 #define PL_TITLE 1 #define PL_PICKLIST 2 #define PL_END 3 static DIALOG picklist[PL_END+1]; static int pick_no; static char **pick_data; /* Global image picklist data and the image picklist dialog */ #define IPL_BORDER 0 #define IPL_TITLE 1 #define IPL_PICKLIST 2 #define IPL_IMG_BORDER 3 #define IPL_IMAGE 4 #define IPL_END 5 static DIALOG img_picklist[IPL_END+1]; static int img_pick_no; static PLAT_IMG_PICKLIST *img_pick_data; typedef struct { int x; int y; int w; int h; } BoundBox; /* ---------------------------------------- GUI OBJECT PROCS SUPPORT FUNCS */ static void Rect3D(int x,int y,int w,int h, int invert,int bevel) { int f; int mid,lo,hi; mid=ACOL(GUI_MID); if (invert) { lo=ACOL(GUI_HI); hi=ACOL(GUI_LO); } else { hi=ACOL(GUI_HI); lo=ACOL(GUI_LO); } rectfill(screen,x,y,x+w-1,y+h-1,mid); for(f=0;fox) { x--; y--; } for(f=0;fx, d->y, d->x+d->w, d->y+d->h, fg_color); /* possibly draw scrollbar */ if (listsize > height) { vline(screen, d->x+d->w-12, d->y+1, d->y+d->h-1, fg_color); /* create and draw the scrollbar */ pattern=create_bitmap(2,2); /* putpixel(pattern,0,0,d->fg); putpixel(pattern,1,1,d->fg); putpixel(pattern,1,0,d->bg); putpixel(pattern,0,1,d->bg); */ putpixel(pattern,0,0,ACOL(GUI_MID)); putpixel(pattern,1,1,ACOL(GUI_MID)); putpixel(pattern,1,0,ACOL(GUI_LO)); putpixel(pattern,0,1,ACOL(GUI_LO)); i = ((d->h-4) * height + listsize/2) / listsize; xx = d->x+d->w-11; yy = d->y+2; if (offset > 0) { len = (((d->h-4) * offset) + listsize/2) / listsize; drawing_mode(DRAW_MODE_COPY_PATTERN, pattern, 0, 0); rectfill(screen, xx, yy, xx+10, yy+len-1, ACOL(GUI_MID)); solid_mode(); yy += len; } if (yy+i < d->y+d->h-2) { Rect3D(xx, yy, 11, i, FALSE, BEVEL); yy += i; drawing_mode(DRAW_MODE_COPY_PATTERN, pattern, 0, 0); rectfill(screen, xx, yy, xx+10, d->y+d->h-2, ACOL(GUI_MID)); solid_mode(); } else Rect3D(xx, yy, 11, i, FALSE, BEVEL); destroy_bitmap(pattern); } } /* This code is copied from _draw_listbox() from guiproc.c in Allegro, so we can force it to call DrawScrollableFrame() */ static void DrawListbox(DIALOG *d) { typedef char *(*getfuncptr)(int, int *); int height, listsize, i, len, bar, x, y, w; int fg_color, fg, bg; char *sel = d->dp2; char *s; char store; (*(getfuncptr)d->dp)(-1, &listsize); height = (d->h-3) / text_height(font); bar = (listsize > height); w = (bar ? d->w-14 : d->w-2); fg_color = (d->flags & D_DISABLED) ? gui_mg_color : d->fg; /* draw box contents */ for (i=0; id2+i < listsize) { if (d->d2+i == d->d1) { fg = d->bg; bg = fg_color; } else if ((sel) && (sel[d->d2+i])) { fg = d->bg; bg = gui_mg_color; } else { fg = fg_color; bg = d->bg; } s = (*(getfuncptr)d->dp)(i+d->d2, NULL); x = d->x + 2; y = d->y + 2 + i*text_height(font); text_mode(bg); rectfill(screen, x, y, x+7, y+text_height(font)-1, bg); x += 8; len = strlen(s); store = 0; while (text_length(font, s) >= d->w - (bar ? 22 : 10)) { s[len] = store; len--; store = s[len]; s[len] = 0; } textout(screen, font, s, x, y, fg); x += text_length(font, s); s[len] = store; if (x <= d->x+w) rectfill(screen, x, y, d->x+w, y+text_height(font)-1, bg); } else rectfill(screen, d->x+2, d->y+2+i*text_height(font), d->x+w, d->y+1+(i+1)*text_height(font), d->bg); } if (d->y+2+i*text_height(font) <= d->y+d->h-2) rectfill(screen, d->x+2, d->y+2+i*text_height(font), d->x+w, d->y+d->h-2, d->bg); /* draw frame, maybe with scrollbar */ DrawScrollableFrame(d, listsize, d->d2, height, fg_color, d->bg); } /* ---------------------------------------- GUI OBJECT PROCS Note that many of the functions are based on the source available in guiproc.c in Allegro */ static int d_3d_box(int msg, DIALOG *d, int c) { if (msg==MSG_DRAW) Rect3D(d->x,d->y,d->w,d->h,FALSE,BEVEL); return(D_O_K); } static int d_inv_3d_box(int msg, DIALOG *d, int c) { if (msg==MSG_DRAW) Rect3D(d->x,d->y,d->w,d->h,TRUE,BEVEL); return(D_O_K); } static int d_img_bmap_proc(int msg, DIALOG *d, int c) { RLE_SPRITE *bm; if (msg==MSG_DRAW) { d_inv_3d_box(MSG_DRAW,&img_picklist[IPL_IMG_BORDER],0); if ((bm=d->dp)) draw_rle_sprite(screen,bm,d->x,d->y); else gui_textout(screen,"No Image", d->x+d->w/2,d->y+d->h/2,ACOL(BLACK),TRUE); } return(D_O_K); } static int d_img_list_proc(int msg, DIALOG *d, int c) { static int my_d_list_proc(int msg, DIALOG *d, int c); int d1; int ret; d1=img_picklist[IPL_PICKLIST].d1; ret=my_d_list_proc(msg,d,c); if (d1!=img_picklist[IPL_PICKLIST].d1) { img_picklist[IPL_IMAGE].dp= img_pick_data[img_picklist[IPL_PICKLIST].d1].img; d_inv_3d_box(MSG_DRAW,&img_picklist[IPL_IMG_BORDER],0); d_img_bmap_proc(MSG_DRAW,&img_picklist[IPL_IMAGE],0); } return(ret); } int my_d_check_proc(int msg, DIALOG *d, int c) { int x; int fg; if (msg==MSG_DRAW) { fg = (d->flags & D_DISABLED) ? gui_mg_color : d->fg; text_mode(d->bg); gui_textout(screen, d->dp, d->x, d->y+(d->h-(text_height(font)-gui_font_baseline))/2, fg, FALSE); x=d->x+d->w-d->h; Rect3D(x,d->y,d->h,d->h,TRUE,SMALL_BEVEL); rectfill(screen,x+SMALL_BEVEL,d->y+SMALL_BEVEL, x+d->h-SMALL_BEVEL*2,d->y+d->h-SMALL_BEVEL*2, ACOL(WHITE)); if (d->flags & D_SELECTED) DrawTick(x+SMALL_BEVEL,d->y+SMALL_BEVEL, d->h-SMALL_BEVEL*2,d->h-SMALL_BEVEL*2); return D_O_K; } return d_button_proc(msg, d, 0); } int my_d_radio_proc(int msg, DIALOG *d, int c) { int x,fg; switch(msg) { case MSG_DRAW: fg=(d->flags & D_DISABLED) ? gui_mg_color : d->fg; text_mode(d->bg); gui_textout (screen, d->dp,d->x+d->h+text_height(font), d->y+(d->h-(text_height(font)-gui_font_baseline))/2,fg,FALSE); x = d->x; Rect3D(x,d->y,d->h,d->h,TRUE,SMALL_BEVEL); rectfill(screen,x+SMALL_BEVEL,d->y+SMALL_BEVEL, x+d->h-SMALL_BEVEL*2,d->y+d->h-SMALL_BEVEL*2, ACOL(WHITE)); if (d->flags & D_SELECTED) { rectfill(screen,x+SMALL_BEVEL+2,d->y+SMALL_BEVEL+2, x+d->h-SMALL_BEVEL*2-2, d->y+d->h-SMALL_BEVEL*2-2, ACOL(BLACK)); } break; default: return(d_radio_proc(msg,d,c)); break; } return(D_O_K); } static int my_d_list_proc(int msg, DIALOG *d, int c) { switch(msg) { case MSG_DRAW: DrawListbox(d); break; default: return(d_list_proc(msg,d,c)); break; } return (D_O_K); } static int my_d_button_proc(int msg, DIALOG *d, int c) { int inv; switch(msg) { case MSG_DRAW: if (d->flags&D_SELECTED) inv=TRUE; else inv=FALSE; Rect3D(d->x,d->y,d->w,d->h,inv,BEVEL); text_mode(-1); gui_textout(screen,d->dp,d->x+(d->w/2), d->y+d->h/2-GFX_fh()/2,d->fg,TRUE); break; default: return(d_button_proc(msg,d,c)); break; } return (D_O_K); } /* ---------------------------------------- FILE SELECTION CODE This code is a copy of the file selector code provided by Shawn Hargreaves in his Allegro library (original filename fsel.c). The only changes are to the file_selector[] DIALOG structure and some of the fs_*_proc() callbacks to use my 3D buttons, etc. Shawn Hargreaves can be contacted at: http://www.talula.demon.co.uk/allegro/ shawn@talula.demon.co.uk If this code causes problems, blame me, not Shawn! */ #ifndef _USE_LFN #define _USE_LFN 0 #endif static int fs_edit_proc(int, DIALOG *, int ); static int fs_flist_proc(int, DIALOG *, int ); static int fs_dlist_proc(int, DIALOG *, int ); static char *fs_flist_getter(int, int *); static char *fs_dlist_getter(int, int *); #define FLIST_SIZE 2048 typedef struct FLIST { char dir[80]; int size; char *name[FLIST_SIZE]; } FLIST; static FLIST *flist = NULL; static char *fext = NULL; static DIALOG file_selector[] = { /* (dialog proc) (x) (y) (w) (h) (fg) (bg) (key) (flags) (d1) (d2) (dp) */ { d_3d_box, 0, 0, 304, 160, 0, 0, 0, 0, 0, 0, NULL }, { d_inv_3d_box, 14, 26, 276, 12, 0, 0, 0, 0, 0, 0, NULL }, { d_ctext_proc, 152, 8, 1, 1, 0, 0, 0, 0, 0, 0, NULL }, { my_d_button_proc, 208, 107, 80, 16, 0, 0, 0, D_EXIT, 0, 0, "OK" }, { my_d_button_proc, 208, 129, 80, 16, 0, 0, 27, D_EXIT, 0, 0, "Cancel" }, { fs_edit_proc, 16, 28, 272, 8, 0, 0, 0, 0, 79, 0, NULL }, { fs_flist_proc, 16, 46, 176, 99, 0, 0, 0, D_EXIT, 0, 0, fs_flist_getter }, { fs_dlist_proc, 208, 46, 80, 51, 0, 0, 0, D_EXIT, 0, 0, fs_dlist_getter }, { NULL } }; #define FS_MESSAGE 2 #define FS_OK 3 #define FS_CANCEL 4 #define FS_EDIT 5 #define FS_FILES 6 #define FS_DISKS 7 /* fs_dlist_getter: * Listbox data getter routine for the file selector disk list. */ static char *fs_dlist_getter(int index, int *list_size) { static char d[] = "A:\\"; if (index < 0) { if (list_size) *list_size = 26; return NULL; } d[0] = 'A' + index; return d; } /* fs_dlist_proc: * Dialog procedure for the file selector disk list. */ static int fs_dlist_proc(int msg, DIALOG *d, int c) { int ret; char *s = file_selector[FS_EDIT].dp; if (msg == MSG_START) d->d1 = d->d2 = 0; ret = my_d_list_proc(msg, d, c); if (ret == D_CLOSE) { *(s++) = 'A' + d->d1; *(s++) = ':'; *(s++) = '\\'; *s = 0; show_mouse(NULL); SEND_MESSAGE(file_selector+FS_FILES, MSG_START, 0); SEND_MESSAGE(file_selector+FS_FILES, MSG_DRAW, 0); SEND_MESSAGE(file_selector+FS_EDIT, MSG_START, 0); SEND_MESSAGE(file_selector+FS_EDIT, MSG_DRAW, 0); show_mouse(screen); return D_O_K; } return ret; } /* fs_edit_proc: * Dialog procedure for the file selector editable string. */ static int fs_edit_proc(int msg, DIALOG *d, int c) { char *s = d->dp; int ch; int attr; int x; char b[80]; if (msg == MSG_START) { if (s[0]) { _fixpath(s, b); errno = 0; x = s[strlen(s)-1]; if ((x=='/') || (x=='\\')) put_backslash(b); for (x=0; b[x]; x++) { if (b[x] == '/') s[x] = '\\'; else s[x] = toupper(b[x]); } s[x] = 0; } } if (msg == MSG_KEY) { if (*s) ch = s[strlen(s)-1]; else ch = 0; if (ch == ':') put_backslash(s); else { if ((ch != '/') && (ch != '\\')) { if (file_exists(s, FA_RDONLY | FA_HIDDEN | FA_DIREC, &attr)) { if (attr & FA_DIREC) put_backslash(s); else return D_CLOSE; } else return D_CLOSE; } } show_mouse(NULL); SEND_MESSAGE(file_selector+FS_FILES, MSG_START, 0); SEND_MESSAGE(file_selector+FS_FILES, MSG_DRAW, 0); SEND_MESSAGE(d, MSG_START, 0); SEND_MESSAGE(d, MSG_DRAW, 0); show_mouse(screen); return D_O_K; } if (msg==MSG_CHAR) { ch = c & 0xff; if ((ch >= 'a') && (ch <= 'z')) c = (c & 0xffffff00L) | (ch - 'a' + 'A'); else if (ch == '/') c = (c & 0xffffff00L) | '\\'; else if (_USE_LFN) { if ((ch > 127) || ((ch < 32) && (ch != 8) && (ch != 0))) return D_O_K; } else { if ((ch != '\\') && (ch != '_') && (ch != ':') && (ch != '.') && ((ch < 'A') || (ch > 'Z')) && ((ch < '0') || (ch > '9')) && (ch != 8) && (ch != 127) && (ch != 0)) return D_O_K; } } return d_edit_proc(msg, d, c); } /* fs_flist_putter: * Callback routine for for_each_file() to fill the file selector listbox. */ static void fs_flist_putter(char *str, int attrib) { int c, c2; char *s, *ext, *tok; char tmp[80]; static char ext_tokens[] = " ,;"; s = get_filename(str); strupr(s); if ((fext) && (!(attrib & FA_DIREC))) { strcpy(tmp, fext); ext = get_extension(s); tok = strtok(tmp, ext_tokens); while (tok) { if (stricmp(ext, tok) == 0) break; tok = strtok(NULL, ext_tokens); } if (!tok) return; } if ((flist->size < FLIST_SIZE) && (strcmp(s,".") != 0)) { for (c=0; csize; c++) { if (flist->name[c][strlen(flist->name[c])-1]=='\\') { if (attrib & FA_DIREC) if (strcmp(s, flist->name[c]) < 0) break; } else { if (attrib & FA_DIREC) break; if (strcmp(s, flist->name[c]) < 0) break; } } for (c2=flist->size; c2>c; c2--) flist->name[c2] = flist->name[c2-1]; flist->name[c] = malloc(strlen(s) + ((attrib & FA_DIREC) ? 2 : 1)); strcpy(flist->name[c], s); if (attrib & FA_DIREC) put_backslash(flist->name[c]); flist->size++; } } /* fs_flist_getter: * Listbox data getter routine for the file selector list. */ static char *fs_flist_getter(int index, int *list_size) { if (index < 0) { if (list_size) *list_size = flist->size; return NULL; } return flist->name[index]; } /* fs_flist_proc: * Dialog procedure for the file selector list. */ static int fs_flist_proc(int msg, DIALOG *d, int c) { int i, ret; int sel = d->d1; char *s = file_selector[FS_EDIT].dp; static int recurse_flag = 0; if (msg == MSG_START) { if (!flist) flist = malloc(sizeof(FLIST)); else { for (i=0; isize; i++) if (flist->name[i]) free(flist->name[i]); } if (!flist) { errno = ENOMEM; return D_CLOSE; } flist->size = 0; for (i=0; isize; i++) flist->name[i] = NULL; strcpy(flist->dir, s); *get_filename(flist->dir) = 0; put_backslash(flist->dir); strcat(flist->dir,"*.*"); for_each_file(flist->dir, FA_RDONLY | FA_DIREC | FA_ARCH, fs_flist_putter, 0); if (errno) alert(NULL, "Disk error", NULL, "OK", NULL, 13, 0); *get_filename(flist->dir) = 0; d->d1 = d->d2 = 0; sel = 0; } if (msg == MSG_END) { if (flist) { for (i=0; isize; i++) if (flist->name[i]) free(flist->name[i]); free(flist); } flist = NULL; } recurse_flag++; ret = my_d_list_proc(msg,d,c); /* call the parent procedure */ recurse_flag--; if (((sel != d->d1) || (ret == D_CLOSE)) && (recurse_flag == 0)) { strcpy(s, flist->dir); *get_filename(s) = 0; put_backslash(s); strcat(s, flist->name[d->d1]); show_mouse(NULL); SEND_MESSAGE(file_selector+FS_EDIT, MSG_START, 0); SEND_MESSAGE(file_selector+FS_EDIT, MSG_DRAW, 0); show_mouse(screen); if (ret == D_CLOSE) return SEND_MESSAGE(file_selector+FS_EDIT, MSG_KEY, 0); } return ret; } /* my_file_select: * Displays the Allegro file selector, with the message as caption. * Allows the user to select a file, and stores the selection in path * (which should have room for at least 80 characters). The files are * filtered according to the file extensions in ext. Passing NULL * includes all files, "PCX;BMP" includes only files with .PCX or .BMP * extensions. Returns zero if it was closed with the Cancel button, * non-zero if it was OK'd. */ int my_file_select(char *message, char *path, char *ext) { int ret; char *p; int f; f=0; while(file_selector[f].proc) { switch(f) { case FS_EDIT: case FS_FILES: case FS_DISKS: file_selector[f].fg=ACOL(BLACK); file_selector[f].bg=ACOL(WHITE); break; default: file_selector[f].fg=gui_fg_color; file_selector[f].bg=gui_bg_color; break; } f++; } file_selector[FS_MESSAGE].dp = message; file_selector[FS_EDIT].dp = path; fext = ext; if (!path[0]) { getcwd(path, 80); for (p=path; *p; p++) if (*p=='/') *p = '\\'; else *p = toupper(*p); put_backslash(path); } clear_keybuf(); do { } while (mouse_b); centre_dialog(file_selector); ret = do_dialog(file_selector, FS_EDIT); if ((ret == FS_CANCEL) || (*get_filename(path) == 0)) return FALSE; p = get_extension(path); if ((*p == 0) && (ext) && (!strpbrk(ext, " ,;"))) { *p = '.'; strcpy(p+1, ext); } return TRUE; } /* ---------------------------------------- GUI MENU SUPPORT FUNCS */ static void DrawMenuItem(BoundBox *b,char *txt,int sel) { int fg; fg=ACOL(GUI_TEXT); if (sel) Rect3D(b->x,b->y,b->w,b->h,FALSE,SMALL_BEVEL); else rectfill(screen,b->x,b->y,b->x+b->w-1,b->y+b->h-1,ACOL(GUI_MID)); text_mode(-1); textout(screen,font,txt,b->x+SMALL_BEVEL+2,b->y+b->h/2-GFX_fh()/2,fg); } static int MouseInBox(BoundBox *box,int no) { int f; for(f=0;f=box[f].x)&&(mouse_x=box[f].y)&&(mouse_ySCRW) x=SCRW-w; if ((y+h)>SCRH) y=SCRH-h; if ((x<0)||(y<0)) { alert("Cannot display menu:",title,"Menu is bigger than display!", "OK",NULL,0,0); return(defval); } cx=x+w/2; cur=0; /* Save screen contents */ show_mouse(NULL); if (!(under=create_bitmap(w+1,h+1))) GFX_exit(EXIT_FAILURE,"No more memory to create MENU save under\n"); blit(screen,under,x,y,0,0,w+1,h+1); /* Draw menu border and title */ Rect3D(x,y,w,h,FALSE,BEVEL); text_mode(-1); textout_centre(screen,font,title,cx,y+BEVEL*2,ACOL(GUI_TEXTBOLD)); /* Draw menu items */ by=y+BEVEL*2+GFX_fh()+4; for(f=0;f>8) { case KEY_ESC: quit=TRUE; break; case KEY_ENTER: done=TRUE; break; case KEY_DOWN: show_mouse(NULL); DrawMenuItem(&box[cur],menu[cur].text,FALSE); cur=(cur+1)%no; DrawMenuItem(&box[cur],menu[cur].text,TRUE); show_mouse(screen); break; case KEY_UP: show_mouse(NULL); DrawMenuItem(&box[cur],menu[cur].text,FALSE); if (cur) cur--; else cur=no-1; DrawMenuItem(&box[cur],menu[cur].text,TRUE); show_mouse(screen); break; } else if ((lx!=mouse_x)||(ly!=mouse_y)) { int new; lx=mouse_x; ly=mouse_y; new=MouseInBox(box,no); if ((new!=-1)&&(new!=cur)) { show_mouse(NULL); DrawMenuItem(&box[cur],menu[cur].text,FALSE); cur=new; DrawMenuItem(&box[cur],menu[cur].text,TRUE); show_mouse(screen); } } } show_mouse(NULL); blit(under,screen,0,0,x,y,w+1,h+1); show_mouse(screen); destroy_bitmap(under); Release(box); if (!quit) no=menu[cur].client_index; else no=defval; return(no); } char *GUI_fsel(char *title, char *default_path, char *filter) { char path[PATH_MAX+1]; if ((filter)&&(*filter=='.')) filter++; strcpy(path,default_path); GFX_bounce(); if (my_file_select(title,path,filter)) return(Strdup(path)); return(NULL); } int GUI_picklist(char *title,char *opt[]) { extern void GFX_FORCE_REDRAW(void); int ml; int ret; pick_no=0; ml=gui_strlen(title); while(opt[pick_no]) { if (gui_strlen(opt[pick_no])>ml) ml=gui_strlen(opt[pick_no]); pick_no++; } pick_data=opt; init_picklist_dialog(title,ml); GFX_bounce(); ret=do_dialog(picklist,-1); GFX_FORCE_REDRAW(); if (ret==-1) return(-1); return(picklist[IPL_PICKLIST].d1); } int GUI_client_picklist(char *title,PLAT_PICKLIST opt[],int defval) { extern void GFX_FORCE_REDRAW(void); int ml; int ret; char **c_opt; int f; pick_no=0; ml=gui_strlen(title); while(opt[pick_no].text) { if (gui_strlen(opt[pick_no].text)>ml) ml=gui_strlen(opt[pick_no].text); pick_no++; } c_opt=Grab(sizeof(char *)*(pick_no+1)); for(f=0;fml) ml=l; h+=GFX_fh()*4; ml+=GFX_fw()*4; x=SCRW/2-ml/2; y=SCRH/2-h/2; /* Surrounding box */ d[0].proc=d_3d_box; d[0].x=x; d[0].y=y; d[0].w=ml; d[0].h=h; d[0].fg=ACOL(GUI_TEXT); d[0].bg=ACOL(GUI_MID); d[0].key=0; d[0].flags=0; d[0].d1=d[0].d2=0; d[0].dp=d[0].dp2=d[0].dp3=NULL; /* Title */ d[1].proc=d_ctext_proc; d[1].x=SCRW/2; d[1].w=ml-20; d[1].y=y+10; d[1].h=10; d[1].fg=ACOL(GUI_TEXTBOLD); d[1].bg=ACOL(GUI_MID); d[1].key=0; d[1].flags=0; d[1].d1=d[1].d2=0; d[1].dp=title; d[1].dp2=d[1].dp3=NULL; /* OK */ d[2].proc=my_d_button_proc; d[2].x=x+10; d[2].y=y+h-15; d[2].w=(ml/2)-20; d[2].h=10+BEVEL; d[2].fg=ACOL(GUI_TEXT); d[2].bg=ACOL(GUI_MID); d[2].key=0; d[2].flags=D_EXIT; d[2].d1=d[2].d2=0; d[2].dp="OK"; d[2].dp2=d[2].dp3=NULL; /* CANCEL */ d[3].proc=my_d_button_proc; d[3].x=SCRW/2+10; d[3].y=y+h-15; d[3].w=(ml/2)-20; d[3].h=10+BEVEL; d[3].fg=ACOL(GUI_TEXT); d[3].bg=ACOL(GUI_MID); d[3].key=0; d[3].flags=D_EXIT; d[3].d1=d[3].d2=0; d[3].dp="Cancel"; d[3].dp2=d[3].dp3=NULL; /* Buttons */ for(f=0;fml) ml=l; h+=GFX_fh()*4; ml+=GFX_fw()*4; x=SCRW/2-ml/2; y=SCRH/2-h/2; /* Surrounding box */ d[0].proc=d_3d_box; d[0].x=x; d[0].y=y; d[0].w=ml; d[0].h=h; d[0].fg=ACOL(GUI_TEXT); d[0].bg=ACOL(GUI_MID); d[0].key=0; d[0].flags=0; d[0].d1=d[0].d2=0; d[0].dp=d[0].dp2=d[0].dp3=NULL; /* Title */ d[1].proc=d_ctext_proc; d[1].x=SCRW/2; d[1].w=ml-20; d[1].y=y+10; d[1].h=10; d[1].fg=ACOL(GUI_TEXTBOLD); d[1].bg=ACOL(GUI_MID); d[1].key=0; d[1].flags=0; d[1].d1=d[1].d2=0; d[1].dp=title; d[1].dp2=d[1].dp3=NULL; /* OK */ d[2].proc=my_d_button_proc; d[2].x=x+10; d[2].y=y+h-15; d[2].w=(ml/2)-20; d[2].h=10+BEVEL; d[2].fg=ACOL(GUI_TEXT); d[2].bg=ACOL(GUI_MID); d[2].key=0; d[2].flags=D_EXIT; d[2].d1=d[2].d2=0; d[2].dp="OK"; d[2].dp2=d[2].dp3=NULL; /* CANCEL */ d[3].proc=my_d_button_proc; d[3].x=SCRW/2+10; d[3].y=y+h-15; d[3].w=(ml/2)-20; d[3].h=10+BEVEL; d[3].fg=ACOL(GUI_TEXT); d[3].bg=ACOL(GUI_MID); d[3].key=0; d[3].flags=D_EXIT; d[3].d1=d[3].d2=0; d[3].dp="Cancel"; d[3].dp2=d[3].dp3=NULL; /* Buttons */ for(f=0;fml)) ml=l; ml+=text_length(font," ")*(DIALOG_STRLEN+3); h+=GFX_fh()*5; x=SCRW/2-ml/2; y=SCRH/2-h/2; /* Surrounding box */ d[0].proc=d_3d_box; d[0].x=x; d[0].y=y; d[0].w=ml; d[0].h=h; d[0].fg=ACOL(GUI_TEXT); d[0].bg=ACOL(GUI_MID); d[0].key=0; d[0].flags=0; d[0].d1=d[0].d2=0; d[0].dp=d[0].dp2=d[0].dp3=NULL; /* Title */ d[1].proc=d_ctext_proc; d[1].x=SCRW/2; d[1].w=ml-20; d[1].y=y+3; d[1].h=10; d[1].fg=ACOL(GUI_TEXTBOLD); d[1].bg=ACOL(GUI_MID); d[1].key=0; d[1].flags=0; d[1].d1=d[1].d2=0; d[1].dp=title; d[1].dp2=d[1].dp3=NULL; /* OK */ d[2].proc=my_d_button_proc; d[2].x=x+10; d[2].w=(ml/2)-20; d[2].h=GFX_fh()+3+BEVEL; d[2].y=y+h-d[2].h-5-BEVEL; d[2].fg=ACOL(GUI_TEXT); d[2].bg=ACOL(GUI_MID); d[2].key=0; d[2].flags=D_EXIT; d[2].d1=d[2].d2=0; d[2].dp="OK"; d[2].dp2=d[2].dp3=NULL; /* CANCEL */ d[3].proc=my_d_button_proc; d[3].x=SCRW/2+10; d[3].y=d[2].y; d[3].w=(ml/2)-20; d[3].h=d[2].h; d[3].fg=ACOL(GUI_TEXT); d[3].bg=ACOL(GUI_MID); d[3].key=0; d[3].flags=D_EXIT; d[3].d1=d[3].d2=0; d[3].dp="Cancel"; d[3].dp2=d[3].dp3=NULL; /* Labels and text entries */ for(f=0;f