ref: 8741ed033996e3cfed28cfc97753ef0cd2c40b17
dir: /textscreen/txt_window.c/
// Emacs style mode select -*- C++ -*- //----------------------------------------------------------------------------- // // $Id$ // // Copyright(C) 1993-1996 Id Software, Inc. // Copyright(C) 2006 Simon Howard // // 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. // #include <stdlib.h> #include <string.h> #include "doomkeys.h" #include "txt_desktop.h" #include "txt_gui.h" #include "txt_main.h" #include "txt_separator.h" #include "txt_window.h" txt_window_t *TXT_NewWindow(char *title) { txt_window_t *win; win = malloc(sizeof(txt_window_t)); win->title = strdup(title); win->x = TXT_SCREEN_W / 2; win->y = TXT_SCREEN_H / 2; win->horiz_align = TXT_HORIZ_CENTER; win->vert_align = TXT_VERT_CENTER; win->widgets = NULL; win->num_widgets = 0; win->selected = 0; TXT_AddDesktopWindow(win); return win; } void TXT_CloseWindow(txt_window_t *window) { int i; // Free all widgets for (i=0; i<window->num_widgets; ++i) { TXT_DestroyWidget(window->widgets[i]); } // Free window resources free(window->widgets); free(window->title); free(window); TXT_RemoveDesktopWindow(window); } static void CalcWindowSize(txt_window_t *window, int *w, int *h) { txt_widget_t *widget; int max_width; int height; int i; int ww, wh; max_width = 10; height = 0; for (i=0; i<window->num_widgets; ++i) { TXT_CalcWidgetSize(window->widgets[i], &ww, &wh); if (ww > max_width) max_width = ww; if (window->widgets[i]->visible) { height += wh; } } *w = max_width; *h = height; } static void CalcWindowPosition(txt_window_t *window, int *x, int *y, int w, int h) { switch (window->horiz_align) { case TXT_HORIZ_LEFT: *x = window->x; break; case TXT_HORIZ_CENTER: *x = window->x - (w / 2); break; case TXT_HORIZ_RIGHT: *x = window->x - (w - 1); break; } switch (window->vert_align) { case TXT_VERT_TOP: *y = window->y; break; case TXT_VERT_CENTER: *y = window->y - (h / 2); break; case TXT_VERT_BOTTOM: *y = window->y - (h - 1); break; } } void TXT_DrawWindow(txt_window_t *window) { int widgets_w, widgets_h; int window_w, window_h; int x, y; int window_x, window_y; int i; int ww, wh; CalcWindowSize(window, &widgets_w, &widgets_h); // Actual window size after padding window_w = widgets_w + 2; window_h = widgets_h + 5; // Use the x,y position as the centerpoint and find the location to // draw the window. CalcWindowPosition(window, &window_x, &window_y, window_w, window_h); // Draw the window TXT_DrawWindowFrame(window->title, window_x, window_y, window_w, window_h); // Draw all widgets x = window_x + 1; y = window_y + 2; for (i=0; i<window->num_widgets; ++i) { if (window->widgets[i]->visible) { TXT_GotoXY(x, y); TXT_DrawWidget(window->widgets[i], widgets_w, i == window->selected); TXT_CalcWidgetSize(window->widgets[i], &ww, &wh); y += wh; } } // Separator for action area TXT_DrawSeparator(window_x, y, window_w); } void TXT_AddWidget(txt_window_t *window, void *uncast_widget) { txt_widget_t *widget; widget = (txt_widget_t *) uncast_widget; if (window->num_widgets == 0) { // This is the first widget added. // // The first widget in a window should always be a separator. // If we are not adding a separator now, add one in first. if (widget->widget_class != &txt_separator_class) { txt_separator_t *separator; separator = TXT_NewSeparator(NULL); window->widgets = malloc(sizeof(txt_widget_t *)); window->widgets[0] = &separator->widget; window->num_widgets = 1; } } window->widgets = realloc(window->widgets, sizeof(txt_widget_t *) * (window->num_widgets + 1)); window->widgets[window->num_widgets] = widget; ++window->num_widgets; } void TXT_SetWindowPosition(txt_window_t *window, txt_horiz_align_t horiz_align, txt_vert_align_t vert_align, int x, int y) { window->vert_align = vert_align; window->horiz_align = horiz_align; window->x = x; window->y = y; } void TXT_WindowKeyPress(txt_window_t *window, int c) { // Send to the currently selected widget first if (window->selected > 0 && window->selected <= window->num_widgets) { if (TXT_WidgetKeyPress(window->widgets[window->selected], c)) { return; } } if (c == KEY_DOWNARROW) { int newsel; // Move cursor down to the next selectable widget for (newsel = window->selected + 1; newsel < window->num_widgets; ++newsel) { if (window->widgets[newsel]->visible && window->widgets[newsel]->selectable) { window->selected = newsel; break; } } } if (c == KEY_UPARROW) { int newsel; // Move cursor down to the next selectable widget for (newsel = window->selected - 1; newsel >= 0; --newsel) { if (window->widgets[newsel]->visible && window->widgets[newsel]->selectable) { window->selected = newsel; break; } } } }