shithub: choc

Download patch

ref: 978ddf539803405ab8fed17e21014ee1ae69fac8
parent: ff6493e0efe1c7ea628d8a6b596f915d9c9764e1
author: Simon Howard <[email protected]>
date: Thu May 18 14:48:24 EDT 2006

Initial working text-mode GUI framework.

Subversion-branch: /trunk/chocolate-doom
Subversion-revision: 480

--- a/textscreen/Makefile.am
+++ b/textscreen/Makefile.am
@@ -2,11 +2,18 @@
 AM_CFLAGS = @SDL_CFLAGS@
 
 noinst_LIBRARIES=libtextscreen.a
+bin_PROGRAMS=guitest
 
 libtextscreen_a_SOURCES =              \
 	txt_gui.c     txt_gui.h        \
 	txt_io.c      txt_io.h         \
 	txt_main.c    txt_main.h       \
+	txt_button.c  txt_button.h     \
+	txt_separator.c txt_separator.h\
+	txt_widget.c  txt_widget.h     \
+	txt_window.c  txt_window.h     \
 	txt_font.h
 
+guitest_LDADD = @SDL_LIBS@ libtextscreen.a
+guitest_SOURCES = guitest.c
 
--- /dev/null
+++ b/textscreen/guitest.c
@@ -1,0 +1,88 @@
+#include <stdlib.h>
+
+
+#include "txt_main.h"
+
+#include "txt_button.h"
+#include "txt_separator.h"
+#include "txt_window.h"
+
+txt_window_t *firstwin;
+
+void SetupWindow(void)
+{
+    txt_window_t *window;
+    char buf[100];
+    int i;
+    
+    window = TXT_NewWindow("Window test", 40, 12);
+
+    strcpy(buf, "This is a button label: ");
+
+    {
+        txt_separator_t *sep;
+        sep = TXT_NewSeparator("Main Section");
+        TXT_AddWidget(window, &sep->widget);
+    }
+
+    for (i=0; i<8; ++i)
+    {
+        txt_button_t *button;
+
+        button = TXT_NewButton(buf);
+        strcat(buf, "a");
+        TXT_AddWidget(window, &button->widget);
+
+        if (i == 4)
+        {
+            txt_separator_t *sep;
+
+            sep = TXT_NewSeparator("Section");
+            TXT_AddWidget(window, &sep->widget);
+        }
+
+        if (i == 6)
+        {
+            txt_separator_t *sep;
+
+            sep = TXT_NewSeparator(NULL);
+            TXT_AddWidget(window, &sep->widget);
+        }
+    }
+
+    firstwin = window;
+}
+
+void Window2(void)
+{
+    txt_window_t *window;
+    int i;
+    
+    window = TXT_NewWindow("Another test", 30, 7);
+
+    for (i=0; i<5; ++i)
+    {
+        txt_button_t *button;
+
+        button = TXT_NewButton("hello there blah blah blah blah");
+        TXT_AddWidget(window, &button->widget);
+    }
+}
+
+int main()
+{
+    TXT_Init();
+
+    Window2();
+    SetupWindow();
+
+    for (;;)
+    {
+        firstwin->selected = (firstwin->selected + 1) % firstwin->num_widgets;
+
+    TXT_DrawAllWindows();
+
+    }
+}
+
+
--- /dev/null
+++ b/textscreen/txt_button.c
@@ -1,0 +1,69 @@
+
+#include <string.h>
+
+#include "txt_button.h"
+#include "txt_io.h"
+#include "txt_main.h"
+#include "txt_widget.h"
+#include "txt_window.h"
+
+static int TXT_ButtonSizeCalc(txt_widget_t *widget)
+{
+    txt_button_t *button = (txt_button_t *) widget;
+
+    // Minimum width is the string length + two spaces for padding
+
+    return strlen(button->label) + 2;
+}
+
+static void TXT_ButtonDrawer(txt_widget_t *widget, int w, int selected)
+{
+    txt_button_t *button = (txt_button_t *) widget;
+    int i;
+
+    TXT_BGColor(TXT_COLOR_BLUE, 0);
+    TXT_FGColor(TXT_COLOR_BRIGHT_WHITE);
+    TXT_PutChar(' ');
+
+    if (selected)
+    {
+        TXT_BGColor(TXT_COLOR_GREY, 0);
+    }
+
+    for (i=0; i<strlen(button->label); ++i)
+    {
+        TXT_PutChar(button->label[i]);
+    }
+    
+    for (i=strlen(button->label); i < w-2; ++i)
+    {
+        TXT_PutChar(' ');
+    }
+}
+
+static void TXT_ButtonDestructor(txt_widget_t *widget)
+{
+    txt_button_t *button = (txt_button_t *) widget;
+
+    free(button->label);
+}
+
+txt_widget_class_t txt_button_class =
+{
+    TXT_ButtonSizeCalc,
+    TXT_ButtonDrawer,
+    TXT_ButtonDestructor,
+};
+
+txt_button_t *TXT_NewButton(char *label)
+{
+    txt_button_t *button;
+
+    button = malloc(sizeof(txt_button_t));
+
+    button->widget.widget_class = &txt_button_class;
+    button->label = strdup(label);
+
+    return button;
+}
+
--- /dev/null
+++ b/textscreen/txt_button.h
@@ -1,0 +1,42 @@
+// 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.
+//
+
+#ifndef TXT_BUTTON_H
+#define TXT_BUTTON_H
+
+typedef struct txt_button_s txt_button_t;
+
+#include "txt_widget.h"
+
+struct txt_button_s
+{
+    txt_widget_t widget;
+    char *label;
+};
+
+txt_button_t *TXT_NewButton(char *label);
+
+#endif /* #ifndef TXT_BUTTON_H */
+
+
--- a/textscreen/txt_io.c
+++ b/textscreen/txt_io.c
@@ -1,7 +1,7 @@
 // Emacs style mode select   -*- C++ -*- 
 //-----------------------------------------------------------------------------
 //
-// $Id: txt_io.c 292 2006-01-14 00:10:54Z fraggle $
+// $Id: txt_io.c 480 2006-05-18 18:48:24Z fraggle $
 //
 // Copyright(C) 1993-1996 Id Software, Inc.
 // Copyright(C) 2005 Simon Howard
@@ -230,6 +230,12 @@
 {
     cur_x = x;
     cur_y = y;
+}
+
+void TXT_GetXY(int *x, int *y)
+{
+    *x = cur_x;
+    *y = cur_y;
 }
 
 void TXT_FGColor(txt_color_t color)
--- a/textscreen/txt_io.h
+++ b/textscreen/txt_io.h
@@ -1,7 +1,7 @@
 // Emacs style mode select   -*- C++ -*- 
 //-----------------------------------------------------------------------------
 //
-// $Id: txt_io.h 291 2006-01-13 23:56:00Z fraggle $
+// $Id: txt_io.h 480 2006-05-18 18:48:24Z fraggle $
 //
 // Copyright(C) 1993-1996 Id Software, Inc.
 // Copyright(C) 2005 Simon Howard
@@ -47,6 +47,7 @@
 void TXT_PutChar(int c);
 void TXT_Puts(char *s);
 void TXT_GotoXY(int x, int y);
+void TXT_GetXY(int *x, int *y);
 void TXT_FGColor(txt_color_t color);
 void TXT_BGColor(int color, int blinking);
 void TXT_ClearScreen(void);
--- /dev/null
+++ b/textscreen/txt_separator.c
@@ -1,0 +1,89 @@
+
+#include <string.h>
+
+#include "txt_separator.h"
+#include "txt_io.h"
+#include "txt_main.h"
+#include "txt_widget.h"
+#include "txt_window.h"
+
+static int TXT_SeparatorSizeCalc(txt_widget_t *widget)
+{
+    txt_separator_t *separator = (txt_separator_t *) widget;
+
+    if (separator->label != NULL)
+    {
+        // Minimum width is the string length + two spaces for padding
+
+        return strlen(separator->label) + 2;
+    }
+    else
+    {
+        return 0;
+    }
+}
+
+static void TXT_SeparatorDrawer(txt_widget_t *widget, int w, int selected)
+{
+    txt_separator_t *separator = (txt_separator_t *) widget;
+    int i;
+    int x, y;
+
+    TXT_GetXY(&x, &y);
+
+    // Draw separator.  Go back one character and draw two extra
+    // to overlap the window borders.
+
+    TXT_DrawSeparator(x-1, y, w + 2);
+    
+    if (separator->label != NULL)
+    {
+        TXT_GotoXY(x, y);
+
+        TXT_BGColor(TXT_COLOR_BLUE, 0);
+        TXT_FGColor(TXT_COLOR_BRIGHT_GREEN);
+        TXT_PutChar(' ');
+
+        for (i=0; i<strlen(separator->label); ++i)
+        {
+            TXT_PutChar(separator->label[i]);
+        }
+    
+        TXT_PutChar(' ');
+    }
+}
+
+static void TXT_SeparatorDestructor(txt_widget_t *widget)
+{
+    txt_separator_t *separator = (txt_separator_t *) widget;
+
+    free(separator->label);
+}
+
+txt_widget_class_t txt_separator_class =
+{
+    TXT_SeparatorSizeCalc,
+    TXT_SeparatorDrawer,
+    TXT_SeparatorDestructor,
+};
+
+txt_separator_t *TXT_NewSeparator(char *label)
+{
+    txt_separator_t *separator;
+
+    separator = malloc(sizeof(txt_separator_t));
+
+    separator->widget.widget_class = &txt_separator_class;
+
+    if (label != NULL)
+    {
+        separator->label = strdup(label);
+    }
+    else
+    {
+        separator->label = NULL;
+    }
+
+    return separator;
+}
+
--- /dev/null
+++ b/textscreen/txt_separator.h
@@ -1,0 +1,44 @@
+// 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.
+//
+
+#ifndef TXT_SEPARATOR_H
+#define TXT_SEPARATOR_H
+
+typedef struct txt_separator_s txt_separator_t;
+
+#include "txt_widget.h"
+
+struct txt_separator_s
+{
+    txt_widget_t widget;
+    char *label;
+};
+
+extern txt_widget_class_t txt_separator_class;
+
+txt_separator_t *TXT_NewSeparator(char *label);
+
+#endif /* #ifndef TXT_SEPARATOR_H */
+
+
--- /dev/null
+++ b/textscreen/txt_widget.c
@@ -1,0 +1,19 @@
+
+#include "txt_widget.h"
+
+int TXT_WidgetWidth(txt_widget_t *widget)
+{
+    return widget->widget_class->size_calc(widget);
+}
+
+void TXT_DrawWidget(txt_widget_t *widget, int w, int selected)
+{
+    widget->widget_class->drawer(widget, w, selected);
+}
+
+void TXT_DestroyWidget(txt_widget_t *widget)
+{
+    widget->widget_class->destructor(widget);
+}
+
+
--- /dev/null
+++ b/textscreen/txt_widget.h
@@ -1,0 +1,55 @@
+// 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.
+//
+
+// Base GUI "widget" class that all widgets inherit from.
+
+#ifndef TXT_WIDGET_H
+#define TXT_WIDGET_H
+
+typedef struct txt_widget_class_s txt_widget_class_t;
+typedef struct txt_widget_s txt_widget_t;
+
+typedef int (*TxtWidgetSizeCalc)(txt_widget_t *widget);
+typedef void (*TxtWidgetDrawer)(txt_widget_t *widget, int w, int selected);
+typedef void (*TxtWidgetDestroy)(txt_widget_t *widget);
+
+struct txt_widget_class_s
+{
+    TxtWidgetSizeCalc size_calc;
+    TxtWidgetDrawer drawer;
+    TxtWidgetDestroy destructor;
+};
+
+struct txt_widget_s
+{
+    txt_widget_class_t *widget_class;
+};
+
+int TXT_WidgetWidth(txt_widget_t *widget);
+void TXT_DrawWidget(txt_widget_t *widget, int w, int selected);
+void TXT_DestroyWidget(txt_widget_t *widget);
+
+#endif /* #ifndef TXT_WIDGET_H */
+
+
--- /dev/null
+++ b/textscreen/txt_window.c
@@ -1,0 +1,195 @@
+// 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 "txt_gui.h"
+#include "txt_separator.h"
+#include "txt_window.h"
+
+#define MAXWINDOWS 128
+
+static txt_window_t *all_windows[MAXWINDOWS];
+static int num_windows = 0;
+
+static void AddWindow(txt_window_t *win)
+{
+    all_windows[num_windows] = win;
+    ++num_windows;
+}
+
+static void RemoveWindow(txt_window_t *win)
+{
+    int from, to;
+
+    for (from=0, to=0; from<num_windows; ++from)
+    {
+        if (all_windows[from] != win)
+        {
+            all_windows[to] = all_windows[from];
+            ++to;
+        }
+    }
+    
+    num_windows = to;
+}
+
+txt_window_t *TXT_NewWindow(char *title, int x, int y)
+{
+    txt_window_t *win;
+
+    win = malloc(sizeof(txt_window_t));
+
+    win->title = strdup(title);
+    win->x = x;
+    win->y = y;
+    win->widgets = NULL;
+    win->num_widgets = 0;
+    win->selected = 0;
+
+    AddWindow(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);
+
+    RemoveWindow(window);
+}
+
+static void TXT_WindowSize(txt_window_t *window, int *w, int *h)
+{
+    int max_width;
+    int i;
+    int ww;
+
+    max_width = 10;
+
+    for (i=0; i<window->num_widgets; ++i)
+    {
+        ww = TXT_WidgetWidth(window->widgets[i]);
+
+        if (ww > max_width)
+            max_width = ww;
+    }
+
+    *w = max_width;
+    *h = window->num_widgets;
+}
+
+static void 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;
+    
+    TXT_WindowSize(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.
+
+    window_x = window->x - (window_w / 2);
+    window_y = window->y - (window_h / 2);
+
+    // Draw the window
+
+    TXT_DrawWindow(window->title, window_x, window_y, window_w, window_h);
+
+    // Draw all widgets
+
+    for (i=0; i<window->num_widgets; ++i)
+    {
+        TXT_GotoXY(window_x + 1, window_y + 2 + i);
+        TXT_DrawWidget(window->widgets[i], widgets_w, i == window->selected);
+    }
+
+    // Separator for action area
+
+    TXT_DrawSeparator(window_x, window_y + 2 + window->num_widgets, window_w);
+}
+
+void TXT_DrawAllWindows(void)
+{
+    int i;
+
+    TXT_DrawDesktop("Not Chocolate Doom setup");
+
+    for (i=0; i<num_windows; ++i)
+    {
+        DrawWindow(all_windows[i]);
+    }
+
+    TXT_UpdateScreen();
+}
+
+void TXT_AddWidget(txt_window_t *window, txt_widget_t *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;
+}
+
--- /dev/null
+++ b/textscreen/txt_window.h
@@ -1,0 +1,60 @@
+// 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.
+//
+
+#ifndef TXT_WINDOW_H
+#define TXT_WINDOW_H
+
+typedef struct txt_window_s txt_window_t;
+
+#include "txt_widget.h" 
+
+struct txt_window_s
+{
+    // Window title
+
+    char *title;
+
+    // Screen coordinates of the centerpoint of the window
+
+    int x, y;
+
+    // Widgets in this window
+
+    txt_widget_t **widgets;
+    int num_widgets;
+
+    // Index of the current selected widget.
+
+    int selected;
+};
+
+txt_window_t *TXT_NewWindow(char *title, int x, int y);
+void TXT_CloseWindow(txt_window_t *window);
+void TXT_AddWidget(txt_window_t *window, txt_widget_t *widget);
+
+void TXT_DrawAllWindows(void);
+
+#endif /* #ifndef TXT_WINDOW_T */
+
+