shithub: choc

ref: 8741ed033996e3cfed28cfc97753ef0cd2c40b17
dir: /textscreen/txt_window.c/

View raw version
// 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;
            }
        } 
    }
}