warzone2100/lib/widget/form.c

1664 lines
44 KiB
C
Raw Normal View History

/*
This file is part of Warzone 2100.
Copyright (C) 1999-2004 Eidos Interactive
Copyright (C) 2005-2010 Warzone 2100 Project
Warzone 2100 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.
Warzone 2100 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 Warzone 2100; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
/** @file
* Functionality for the form widget.
*/
#include <string.h>
#include "lib/framework/frame.h"
#include "widget.h"
#include "widgint.h"
#include "form.h"
#include "tip.h"
// FIXME Direct iVis implementation include!
#include "lib/ivis_common/pieblitfunc.h"
#include "lib/ivis_common/piepalette.h"
/* Control whether single tabs are displayed */
#define NO_DISPLAY_SINGLE_TABS 1
static inline void formFreeTips(W_TABFORM *psForm);
/* Store the position of a tab */
typedef struct _tab_pos
{
SDWORD index;
SDWORD x,y;
UDWORD width,height;
SDWORD TabMultiplier; //Added to keep track of tab scroll
} TAB_POS;
/* Set default colours for a form */
static void formSetDefaultColours(W_FORM *psForm)
{
psForm->aColours[WCOL_BKGRND] = WZCOL_FORM_BACKGROUND;
psForm->aColours[WCOL_TEXT] = WZCOL_FORM_TEXT;
psForm->aColours[WCOL_LIGHT] = WZCOL_FORM_LIGHT;
psForm->aColours[WCOL_DARK] = WZCOL_FORM_DARK;
psForm->aColours[WCOL_HILITE] = WZCOL_FORM_HILITE;
psForm->aColours[WCOL_CURSOR] = WZCOL_FORM_CURSOR;
psForm->aColours[WCOL_TIPBKGRND] = WZCOL_FORM_TIP_BACKGROUND;
psForm->aColours[WCOL_DISABLE] = WZCOL_FORM_DISABLE;
}
/* Create a plain form widget */
static W_FORM* formCreatePlain(const W_FORMINIT* psInit)
{
/* Allocate the required memory */
W_FORM* psWidget = (W_FORM *)malloc(sizeof(W_FORM));
if (psWidget == NULL)
{
debug(LOG_FATAL, "formCreatePlain: Out of memory");
abort();
return NULL;
}
/* Initialise the structure */
memset(psWidget, 0, sizeof(W_FORM));
psWidget->type = WIDG_FORM;
psWidget->id = psInit->id;
psWidget->formID = psInit->formID;
psWidget->style = psInit->style;
psWidget->disableChildren = psInit->disableChildren;
psWidget->animCount = 0;
psWidget->x = psInit->x;
psWidget->y = psInit->y;
psWidget->width = psInit->width;
psWidget->height = psInit->height;
if (psInit->pDisplay)
{
psWidget->display = psInit->pDisplay;
}
else
{
psWidget->display = formDisplay;
}
psWidget->callback = psInit->pCallback;
psWidget->pUserData = psInit->pUserData;
psWidget->UserData = psInit->UserData;
psWidget->psWidgets = NULL;
psWidget->psLastHiLite = NULL;
formSetDefaultColours(psWidget);
formInitialise(psWidget);
return psWidget;
}
/* Free a plain form widget */
static void formFreePlain(W_FORM *psWidget)
{
ASSERT( psWidget != NULL, "Invalid form pointer" );
widgReleaseWidgetList(psWidget->psWidgets);
CheckpsMouseOverWidget(psWidget); // clear global if needed
free(psWidget);
}
/* Create a plain form widget */
static W_CLICKFORM* formCreateClickable(const W_FORMINIT* psInit)
{
/* Allocate the required memory */
W_CLICKFORM* psWidget = (W_CLICKFORM *)malloc(sizeof(W_CLICKFORM));
if (psWidget == NULL)
{
debug(LOG_FATAL, "formCreateClickable: Out of memory");
abort();
return NULL;
}
/* Initialise the structure */
memset(psWidget, 0, sizeof(W_CLICKFORM));
psWidget->type = WIDG_FORM;
psWidget->id = psInit->id;
psWidget->formID = psInit->formID;
psWidget->style = psInit->style;
psWidget->disableChildren = psInit->disableChildren;
psWidget->animCount = 0;
psWidget->x = psInit->x;
psWidget->y = psInit->y;
psWidget->width = psInit->width;
psWidget->height = psInit->height;
psWidget->callback = psInit->pCallback;
psWidget->pUserData = psInit->pUserData;
psWidget->UserData = psInit->UserData;
psWidget->AudioCallback = WidgGetAudioCallback();
psWidget->HilightAudioID = WidgGetHilightAudioID();
psWidget->ClickedAudioID = WidgGetClickedAudioID();
if (psInit->pDisplay)
{
psWidget->display = psInit->pDisplay;
}
else
{
psWidget->display = formDisplayClickable;
}
psWidget->psWidgets = NULL;
psWidget->psLastHiLite = NULL;
psWidget->pTip = psInit->pTip;
formSetDefaultColours((W_FORM *)psWidget);
formInitialise((W_FORM *)psWidget);
return psWidget;
}
/* Free a plain form widget */
static void formFreeClickable(W_CLICKFORM *psWidget)
{
ASSERT( psWidget != NULL,
"formFreePlain: Invalid form pointer" );
widgReleaseWidgetList(psWidget->psWidgets);
free(psWidget);
}
/* Create a tabbed form widget */
static W_TABFORM* formCreateTabbed(const W_FORMINIT* psInit)
{
W_TABFORM* psWidget;
UDWORD major,minor;
W_MAJORTAB *psMajor;
if (psInit->numMajor == 0)
{
ASSERT(false, "formCreateTabbed: Must have at least one major tab on a tabbed form");
return NULL;
}
if (psInit->majorPos != 0
&& psInit->majorPos == psInit->minorPos)
{
ASSERT(false, "formCreateTabbed: Cannot have major and minor tabs on same side");
return NULL;
}
if (psInit->numMajor >= WFORM_MAXMAJOR)
{
ASSERT(false, "formCreateTabbed: Too many Major tabs" );
return NULL;
}
for(major=0; major<psInit->numMajor; major++)
{
if (psInit->aNumMinors[major] >= WFORM_MAXMINOR)
{
ASSERT(false, "formCreateTabbed: Too many Minor tabs for Major %u", major);
return NULL;
}
if (psInit->aNumMinors[major] == 0)
{
ASSERT(false, "formCreateTabbed: Must have at least one Minor tab for each major");
return NULL;
}
}
/* Allocate the required memory */
psWidget = (W_TABFORM *)malloc(sizeof(W_TABFORM));
if (psWidget == NULL)
{
debug(LOG_FATAL, "formCreateTabbed: Out of memory");
abort();
return NULL;
}
memset(psWidget, 0, sizeof(W_TABFORM));
/* Allocate the memory for tool tips and copy them in */
psMajor = psWidget->asMajor;
for (major = 0; major < psInit->numMajor; ++major)
{
/* Check for a tip for the major tab */
psMajor->pTip = psInit->apMajorTips[major];
/* Check for tips for the minor tab */
for(minor = 0; minor < psInit->aNumMinors[major]; ++minor)
{
psMajor->asMinor[minor].pTip = psInit->apMinorTips[major][minor];
}
psMajor++;
}
/* Initialise the structure */
psWidget->type = WIDG_FORM;
psWidget->id = psInit->id;
psWidget->formID = psInit->formID;
psWidget->style = psInit->style;
psWidget->disableChildren = psInit->disableChildren;
psWidget->animCount = 0;
psWidget->x = psInit->x;
psWidget->y = psInit->y;
psWidget->width = psInit->width;
psWidget->height = psInit->height;
if (psInit->pDisplay)
{
psWidget->display = psInit->pDisplay;
}
else
{
psWidget->display = formDisplayTabbed;
}
psWidget->callback = psInit->pCallback;
psWidget->pUserData = psInit->pUserData;
psWidget->UserData = psInit->UserData;
psWidget->psLastHiLite = NULL;
psWidget->majorSize = psInit->majorSize;
psWidget->minorSize = psInit->minorSize;
psWidget->tabMajorThickness = psInit->tabMajorThickness;
psWidget->tabMinorThickness = psInit->tabMinorThickness;
psWidget->tabMajorGap = psInit->tabMajorGap;
psWidget->tabMinorGap = psInit->tabMinorGap;
psWidget->tabVertOffset = psInit->tabVertOffset;
psWidget->tabHorzOffset = psInit->tabHorzOffset;
psWidget->majorOffset = psInit->majorOffset;
psWidget->minorOffset = psInit->minorOffset;
psWidget->majorPos = psInit->majorPos;
psWidget->minorPos = psInit->minorPos;
psWidget->pTabDisplay = psInit->pTabDisplay;
psWidget->TabMultiplier = psInit->TabMultiplier;
psWidget->numButtons = psInit->numButtons;
psWidget->numStats = psInit->numStats;
psWidget->majorT = 0;
psWidget->minorT = 0;
formSetDefaultColours((W_FORM *)psWidget);
/* Set up the tab data.
* All widget pointers have been zeroed by the memset above.
*/
psWidget->numMajor = psInit->numMajor;
for (major=0; major<psInit->numMajor; major++)
{
psWidget->asMajor[major].numMinor = psInit->aNumMinors[major];
}
formInitialise((W_FORM *)psWidget);
return psWidget;
}
/* Free the tips strings for a tabbed form */
static inline void formFreeTips(W_TABFORM *psForm)
{
psForm = psForm;
}
/* Free a tabbed form widget */
static void formFreeTabbed(W_TABFORM *psWidget)
{
WIDGET *psCurr;
W_FORMGETALL sGetAll;
ASSERT( psWidget != NULL,
"formFreeTabbed: Invalid form pointer" );
formFreeTips(psWidget);
formInitGetAllWidgets((W_FORM *)psWidget,&sGetAll);
psCurr = formGetAllWidgets(&sGetAll);
while (psCurr)
{
widgReleaseWidgetList(psCurr);
psCurr = formGetAllWidgets(&sGetAll);
}
free(psWidget);
}
/* Create a form widget data structure */
W_FORM* formCreate(const W_FORMINIT* psInit)
{
/* Check the style bits are OK */
if (psInit->style & ~(WFORM_TABBED | WFORM_INVISIBLE | WFORM_CLICKABLE
| WFORM_NOCLICKMOVE | WFORM_NOPRIMARY | WFORM_SECONDARY
| WIDG_HIDDEN))
{
ASSERT(false, "formCreate: Unknown style bit");
return NULL;
}
if ((psInit->style & WFORM_TABBED)
&& (psInit->style & (WFORM_INVISIBLE | WFORM_CLICKABLE)))
{
ASSERT(false, "formCreate: Tabbed form cannot be invisible or clickable");
return NULL;
}
if ((psInit->style & WFORM_INVISIBLE)
&& (psInit->style & WFORM_CLICKABLE))
{
ASSERT(false, "formCreate: Cannot have an invisible clickable form");
return NULL;
}
if (!(psInit->style & WFORM_CLICKABLE)
&& ((psInit->style & WFORM_NOPRIMARY)
|| (psInit->style & WFORM_SECONDARY)))
{
ASSERT(false, "formCreate: Cannot set keys if the form isn't clickable");
return NULL;
}
/* Create the correct type of form */
if (psInit->style & WFORM_TABBED)
{
return (W_FORM*)formCreateTabbed(psInit);
}
else if (psInit->style & WFORM_CLICKABLE)
{
return (W_FORM*)formCreateClickable(psInit);
}
else
{
return formCreatePlain(psInit);
}
return NULL;
}
/* Free the memory used by a form */
void formFree(W_FORM *psWidget)
{
if (psWidget->style & WFORM_TABBED)
{
formFreeTabbed((W_TABFORM *)psWidget);
}
else if (psWidget->style & WFORM_CLICKABLE)
{
formFreeClickable((W_CLICKFORM *)psWidget);
}
else
{
formFreePlain(psWidget);
}
}
/* Add a widget to a form */
BOOL formAddWidget(W_FORM *psForm, WIDGET *psWidget, W_INIT *psInit)
{
W_TABFORM *psTabForm;
WIDGET **ppsList;
W_MAJORTAB *psMajor;
ASSERT( psWidget != NULL,
"formAddWidget: Invalid widget pointer" );
if (psForm->style & WFORM_TABBED)
{
ASSERT( psForm != NULL,
"formAddWidget: Invalid tab form pointer" );
psTabForm = (W_TABFORM *)psForm;
if (psInit->majorID >= psTabForm->numMajor)
{
ASSERT( false, "formAddWidget: Major tab does not exist" );
return false;
}
psMajor = psTabForm->asMajor + psInit->majorID;
if (psInit->minorID >= psMajor->numMinor)
{
ASSERT( false, "formAddWidget: Minor tab does not exist" );
return false;
}
ppsList = &(psMajor->asMinor[psInit->minorID].psWidgets);
psWidget->psNext = *ppsList;
*ppsList = psWidget;
}
else
{
ASSERT( psForm != NULL,
"formAddWidget: Invalid form pointer" );
psWidget->psNext = psForm->psWidgets;
psForm->psWidgets = psWidget;
}
return true;
}
/* Get the button state of a click form */
UDWORD formGetClickState(W_CLICKFORM *psForm)
{
UDWORD State = 0;
if (psForm->state & WCLICK_GREY)
{
State |= WBUT_DISABLE;
}
if (psForm->state & WCLICK_LOCKED)
{
State |= WBUT_LOCK;
}
if (psForm->state & WCLICK_CLICKLOCK)
{
State |= WBUT_CLICKLOCK;
}
return State;
}
/* Set the button state of a click form */
void formSetClickState(W_CLICKFORM *psForm, UDWORD state)
{
ASSERT( !((state & WBUT_LOCK) && (state & WBUT_CLICKLOCK)),
"widgSetButtonState: Cannot have WBUT_LOCK and WBUT_CLICKLOCK" );
if (state & WBUT_DISABLE)
{
psForm->state |= WCLICK_GREY;
}
else
{
psForm->state &= ~WCLICK_GREY;
}
if (state & WBUT_LOCK)
{
psForm->state |= WCLICK_LOCKED;
}
else
{
psForm->state &= ~WCLICK_LOCKED;
}
if (state & WBUT_CLICKLOCK)
{
psForm->state |= WCLICK_CLICKLOCK;
}
else
{
psForm->state &= ~WCLICK_CLICKLOCK;
}
}
/* Return the widgets currently displayed by a form */
WIDGET *formGetWidgets(W_FORM *psWidget)
{
W_TABFORM *psTabForm;
W_MAJORTAB *psMajor;
if (psWidget->style & WFORM_TABBED)
{
psTabForm = (W_TABFORM *)psWidget;
psMajor = psTabForm->asMajor + psTabForm->majorT;
return psMajor->asMinor[psTabForm->minorT].psWidgets;
}
else
{
return psWidget->psWidgets;
}
}
/* Initialise the formGetAllWidgets function */
void formInitGetAllWidgets(W_FORM *psWidget, W_FORMGETALL *psCtrl)
{
if (psWidget->style & WFORM_TABBED)
{
psCtrl->psGAWList = NULL;
psCtrl->psGAWForm = (W_TABFORM *)psWidget;
psCtrl->psGAWMajor = psCtrl->psGAWForm->asMajor;
psCtrl->GAWMajor = 0;
psCtrl->GAWMinor = 0;
}
else
{
psCtrl->psGAWList = psWidget->psWidgets;
psCtrl->psGAWForm = NULL;
}
}
/* Repeated calls to this function will return widget lists
* until all widgets in a form have been returned.
* When a NULL list is returned, all widgets have been seen.
*/
WIDGET *formGetAllWidgets(W_FORMGETALL *psCtrl)
{
WIDGET *psRetList;
if (psCtrl->psGAWForm == NULL)
{
/* Not a tabbed form, return the list */
psRetList = psCtrl->psGAWList;
psCtrl->psGAWList = NULL;
}
else
{
/* Working with a tabbed form - search for the first widget list */
psRetList = NULL;
while (psRetList == NULL && psCtrl->GAWMajor < psCtrl->psGAWForm->numMajor)
{
psRetList = psCtrl->psGAWMajor->asMinor[psCtrl->GAWMinor++].psWidgets;
if (psCtrl->GAWMinor >= psCtrl->psGAWMajor->numMinor)
{
psCtrl->GAWMinor = 0;
psCtrl->GAWMajor ++;
psCtrl->psGAWMajor ++;
}
}
}
return psRetList;
}
static W_TABFORM* widgGetTabbedFormById(W_SCREEN * const psScreen, const UDWORD id)
{
W_TABFORM * const psForm = (W_TABFORM *)widgGetFromID(psScreen, id);
if (psForm == NULL
|| psForm->type != WIDG_FORM
|| !(psForm->style & WFORM_TABBED))
{
return NULL;
}
return psForm;
}
/* Set the current tabs for a tab form */
void widgSetTabs(W_SCREEN *psScreen, UDWORD id, UWORD major, UWORD minor)
{
W_TABFORM * const psForm = widgGetTabbedFormById(psScreen, id);
ASSERT(psForm != NULL, "widgSetTabs: Invalid tab form pointer" );
if (psForm == NULL)
{
return; /* make us work fine in no assert compiles */
}
ASSERT(major < widgGetNumTabMajor(psScreen, id), "widgSetTabs id=%u: invalid major id %u >= max %u", id,
major, widgGetNumTabMajor(psScreen, id));
ASSERT(minor < widgGetNumTabMinor(psScreen, id, major), "widgSetTabs id=%u: invalid minor id %u >= max %u", id,
minor, widgGetNumTabMinor(psScreen, id, major));
// Make sure to bail out when we've been passed out-of-bounds major or minor numbers
if (major >= widgGetNumTabMajor(psScreen, id)
|| minor >= widgGetNumTabMinor(psScreen, id, major))
{
return;
}
psForm->majorT = major;
psForm->minorT = minor;
psForm->asMajor[major].lastMinor = minor;
}
int widgGetNumTabMajor(W_SCREEN *psScreen, UDWORD id)
{
W_TABFORM * const psForm = widgGetTabbedFormById(psScreen, id);
ASSERT(psForm != NULL, "Couldn't find a tabbed form with ID %u", id);
if (psForm == NULL)
{
return 0;
}
return psForm->numMajor;
}
int widgGetNumTabMinor(W_SCREEN *psScreen, UDWORD id, UWORD pMajor)
{
W_TABFORM * const psForm = widgGetTabbedFormById(psScreen, id);
ASSERT(psForm != NULL, "Couldn't find a tabbed form with ID %u", id);
if (psForm == NULL)
{
return 0;
}
return psForm->asMajor[pMajor].numMinor;
}
/* Get the current tabs for a tab form */
void widgGetTabs(W_SCREEN *psScreen, UDWORD id, UWORD *pMajor, UWORD *pMinor)
{
W_TABFORM *psForm;
psForm = (W_TABFORM *)widgGetFromID(psScreen, id);
if (psForm == NULL || psForm->type != WIDG_FORM ||
!(psForm->style & WFORM_TABBED))
{
ASSERT( false,"widgGetTabs: couldn't find tabbed form from id" );
return;
}
ASSERT(psForm != NULL, "widgGetTabs: Invalid tab form pointer");
ASSERT(psForm->majorT < psForm->numMajor, "widgGetTabs: invalid major id %u >= max %u", psForm->majorT, psForm->numMajor);
ASSERT(psForm->minorT < psForm->asMajor[psForm->majorT].numMinor, "widgGetTabs: invalid minor id %u >= max %u",
psForm->minorT, psForm->asMajor[psForm->majorT].numMinor);
*pMajor = psForm->majorT;
*pMinor = psForm->minorT;
}
/* Set a colour on a form */
void widgSetColour(W_SCREEN *psScreen, UDWORD id, UDWORD index, PIELIGHT colour)
{
W_TABFORM *psForm = (W_TABFORM *)widgGetFromID(psScreen, id);
ASSERT_OR_RETURN(, psForm && psForm->type == WIDG_FORM, "Could not find form from id %u", id);
ASSERT_OR_RETURN(, index < WCOL_MAX, "Colour id %u out of range", index);
psForm->aColours[index] = colour;
}
/* Return the origin on the form from which button locations are calculated */
void formGetOrigin(W_FORM *psWidget, SDWORD *pXOrigin, SDWORD *pYOrigin)
{
W_TABFORM *psTabForm;
ASSERT( psWidget != NULL,
"formGetOrigin: Invalid form pointer" );
if (psWidget->style & WFORM_TABBED)
{
psTabForm = (W_TABFORM *)psWidget;
if(psTabForm->majorPos == WFORM_TABTOP) {
*pYOrigin = psTabForm->tabMajorThickness;
} else if(psTabForm->minorPos == WFORM_TABTOP) {
*pYOrigin = psTabForm->tabMinorThickness;
} else {
*pYOrigin = 0;
}
if(psTabForm->majorPos == WFORM_TABLEFT) {
*pXOrigin = psTabForm->tabMajorThickness;
} else if(psTabForm->minorPos == WFORM_TABLEFT) {
*pXOrigin = psTabForm->tabMinorThickness;
} else {
*pXOrigin = 0;
}
// if ((psTabForm->majorPos == WFORM_TABTOP) ||
// (psTabForm->minorPos == WFORM_TABTOP))
// {
// *pYOrigin = psTabForm->tabThickness;
// }
// else
// {
// *pYOrigin = 0;
// }
// if ((psTabForm->majorPos == WFORM_TABLEFT) ||
// (psTabForm->minorPos == WFORM_TABLEFT))
// {
// *pXOrigin = psTabForm->tabThickness;
// }
// else
// {
// *pXOrigin = 0;
// }
}
else
{
*pXOrigin = 0;
*pYOrigin = 0;
}
}
/* Initialise a form widget before running it */
void formInitialise(W_FORM *psWidget)
{
W_TABFORM *psTabForm;
W_CLICKFORM *psClickForm;
UDWORD i;
if (psWidget->style & WFORM_TABBED)
{
ASSERT( psWidget != NULL,
"formInitialise: invalid tab form pointer" );
psTabForm = (W_TABFORM *)psWidget;
psTabForm->majorT = 0;
psTabForm->minorT = 0;
psTabForm->tabHiLite = (UWORD)(-1);
for (i=0; i<psTabForm->numMajor; i++)
{
psTabForm->asMajor[i].lastMinor = 0;
}
}
else if (psWidget->style & WFORM_CLICKABLE)
{
ASSERT( psWidget != NULL,
"formInitialise: invalid clickable form pointer" );
psClickForm = (W_CLICKFORM *)psWidget;
psClickForm->state = WCLICK_NORMAL;
}
else
{
ASSERT( psWidget != NULL,
"formInitialise: invalid form pointer" );
}
psWidget->psLastHiLite = NULL;
}
// Currently in game, I can only find that warzone uses horizontal tabs.
// So ONLY this routine was modified. Will have to modify the vert. tab
// routine if we ever use it.
// Choose a horizontal tab from a coordinate
static BOOL formPickHTab(TAB_POS *psTabPos,
SDWORD x0, SDWORD y0,
UDWORD width, UDWORD height, UDWORD gap,
UDWORD number, SDWORD fx, SDWORD fy)
{
SDWORD x, y1;
UDWORD i;
#if NO_DISPLAY_SINGLE_TABS
if (number == 1)
{
// Don't have single tabs
return false;
}
#endif
x = x0;
y1 = y0 + height;
// We need to filter out some tabs, since we can only display 7 at a time, with
// the scroll tabs in place, and 8 without. MAX_TAB_SMALL_SHOWN (currently 8) is the case
// when we do NOT want scrolltabs, and are using smallTab icons.
// Also need to check if the TabMultiplier is set or not, if not then it means
// we have not yet added the code to display/handle the tab scroll buttons.
// At this time, I think only the design screen has this limitation of only 8 tabs.
if (number > (MAX_TAB_SMALL_SHOWN - 1) && psTabPos->TabMultiplier) // of course only do this if we actually need >8 tabs.
{
number -= (psTabPos->TabMultiplier - 1) * TAB_SEVEN;
if (number > TAB_SEVEN) // is it still > than TAB_SEVEN?
{
number = TAB_SEVEN;
}
}
else if (number > MAX_TAB_SMALL_SHOWN)
{
// we need to clip the tab count to max amount *without* the scrolltabs visible.
// The reason for this, is that in design screen & 'feature' debug & others(?),
// we can get over max # of tabs that the game originally supported.
// This made it look bad.
number = MAX_TAB_SMALL_SHOWN;
}
for (i=0; i < number; i++)
{
// if (fx >= x && fx <= x + (SDWORD)(width - gap) &&
if (fx >= x && fx <= x + (SDWORD)(width) &&
fy >= y0 && fy <= y1)
{
// found a tab under the coordinate
if (psTabPos->TabMultiplier) //Checks to see we need the extra tab scroll buttons
{ // holds the VIRTUAL tab #, since obviously, we can't display more than 7
psTabPos->index = (i % TAB_SEVEN)+ ((psTabPos->TabMultiplier -1)*TAB_SEVEN);
}
else
{ // This is a normal request.
psTabPos->index = i;
}
psTabPos->x = x ;
psTabPos->y = y0;
psTabPos->width = width;
psTabPos->height = height;
return true;
}
x += width + gap;
}
/* Didn't find any */
return false;
}
// NOTE: This routine is NOT modified to use the tab scroll buttons.
// Choose a vertical tab from a coordinate
static BOOL formPickVTab(TAB_POS *psTabPos,
SDWORD x0, SDWORD y0,
UDWORD width, UDWORD height, UDWORD gap,
UDWORD number, SDWORD fx, SDWORD fy)
{
SDWORD x1, y;
UDWORD i;
#if NO_DISPLAY_SINGLE_TABS
if (number == 1)
{
/* Don't have single tabs */
return false;
}
#endif
x1 = x0 + width;
y = y0;
for (i=0; i < number; i++)
{
if (fx >= x0 && fx <= x1 &&
fy >= y && fy <= y + (SDWORD)(height))
// fy >= y && fy <= y + (SDWORD)(height - gap))
{
/* found a tab under the coordinate */
psTabPos->index = i;
psTabPos->x = x0;
psTabPos->y = y;
psTabPos->width = width;
psTabPos->height = height;
return true;
}
y += height + gap;
}
/* Didn't find any */
return false;
}
/* Find which tab is under a form coordinate */
static BOOL formPickTab(W_TABFORM *psForm, UDWORD fx, UDWORD fy,
TAB_POS *psTabPos)
{
SDWORD x0,y0, x1,y1;
W_MAJORTAB *psMajor;
SDWORD xOffset,yOffset;
SDWORD xOffset2,yOffset2;
/* Get the basic position of the form */
x0 = 0;
y0 = 0;
x1 = psForm->width;
y1 = psForm->height;
/* Adjust for where the tabs are */
if(psForm->majorPos == WFORM_TABLEFT) {
x0 += psForm->tabMajorThickness;
} else if(psForm->minorPos == WFORM_TABLEFT) {
x0 += psForm->tabMinorThickness;
}
if(psForm->majorPos == WFORM_TABRIGHT) {
x1 -= psForm->tabMajorThickness;
} else if (psForm->minorPos == WFORM_TABRIGHT) {
x1 -= psForm->tabMinorThickness;
}
if(psForm->majorPos == WFORM_TABTOP) {
y0 += psForm->tabMajorThickness;
} else if(psForm->minorPos == WFORM_TABTOP) {
y0 += psForm->tabMinorThickness;
}
if(psForm->majorPos == WFORM_TABBOTTOM) {
y1 -= psForm->tabMajorThickness;
} else if(psForm->minorPos == WFORM_TABBOTTOM) {
y1 -= psForm->tabMinorThickness;
}
// /* Adjust for where the tabs are */
// if (psForm->majorPos == WFORM_TABLEFT || psForm->minorPos == WFORM_TABLEFT)
// {
// x0 += psForm->tabThickness;
// }
// if (psForm->majorPos == WFORM_TABRIGHT || psForm->minorPos == WFORM_TABRIGHT)
// {
// x1 -= psForm->tabThickness;
// }
// if (psForm->majorPos == WFORM_TABTOP || psForm->minorPos == WFORM_TABTOP)
// {
// y0 += psForm->tabThickness;
// }
// if (psForm->majorPos == WFORM_TABBOTTOM || psForm->minorPos == WFORM_TABBOTTOM)
// {
// y1 -= psForm->tabThickness;
// }
xOffset = yOffset = 0;
switch (psForm->minorPos)
{
case WFORM_TABTOP:
yOffset = psForm->tabVertOffset;
break;
case WFORM_TABLEFT:
xOffset = psForm->tabHorzOffset;
break;
case WFORM_TABBOTTOM:
yOffset = psForm->tabVertOffset;
break;
case WFORM_TABRIGHT:
xOffset = psForm->tabHorzOffset;
break;
}
xOffset2 = yOffset2 = 0;
psTabPos->TabMultiplier = psForm->TabMultiplier;
/* Check the major tabs */
switch (psForm->majorPos)
{
case WFORM_TABTOP:
if (formPickHTab(psTabPos, x0+psForm->majorOffset - xOffset, y0 - psForm->tabMajorThickness,
psForm->majorSize, psForm->tabMajorThickness, psForm->tabMajorGap,
psForm->numMajor, fx, fy))
{
return true;
}
yOffset2 = -psForm->tabVertOffset;
break;
case WFORM_TABBOTTOM:
if (formPickHTab(psTabPos, x0+psForm->majorOffset - xOffset, y1,
psForm->majorSize, psForm->tabMajorThickness, psForm->tabMajorGap,
psForm->numMajor, fx, fy))
{
return true;
}
break;
case WFORM_TABLEFT:
if (formPickVTab(psTabPos, x0 - psForm->tabMajorThickness, y0+psForm->majorOffset - yOffset,
psForm->tabMajorThickness, psForm->majorSize, psForm->tabMajorGap,
psForm->numMajor, fx, fy))
{
return true;
}
xOffset2 = psForm->tabHorzOffset;
break;
case WFORM_TABRIGHT:
if (formPickVTab(psTabPos, x1, y0+psForm->majorOffset - yOffset,
psForm->tabMajorThickness, psForm->majorSize, psForm->tabMajorGap,
psForm->numMajor, fx, fy))
{
return true;
}
break;
case WFORM_TABNONE:
ASSERT( false, "formDisplayTabbed: Cannot have a tabbed form with no major tabs" );
break;
}
/* Draw the minor tabs */
psMajor = psForm->asMajor + psForm->majorT;
switch (psForm->minorPos)
{
case WFORM_TABTOP:
if (formPickHTab(psTabPos, x0+psForm->minorOffset - xOffset2, y0 - psForm->tabMinorThickness,
psForm->minorSize, psForm->tabMinorThickness, psForm->tabMinorGap,
psMajor->numMinor, fx, fy))
{
psTabPos->index += psForm->numMajor;
return true;
}
break;
case WFORM_TABBOTTOM:
if (formPickHTab(psTabPos, x0+psForm->minorOffset - xOffset2, y1,
psForm->minorSize, psForm->tabMinorThickness, psForm->tabMinorGap,
psMajor->numMinor, fx, fy))
{
psTabPos->index += psForm->numMajor;
return true;
}
break;
case WFORM_TABLEFT:
if (formPickVTab(psTabPos, x0+xOffset - psForm->tabMinorThickness, y0+psForm->minorOffset - yOffset2,
psForm->tabMinorThickness, psForm->minorSize, psForm->tabMinorGap,
psMajor->numMinor, fx, fy))
{
psTabPos->index += psForm->numMajor;
return true;
}
break;
case WFORM_TABRIGHT:
if (formPickVTab(psTabPos, x1+xOffset, y0+psForm->minorOffset - yOffset2,
psForm->tabMinorThickness, psForm->minorSize, psForm->tabMinorGap,
psMajor->numMinor, fx, fy))
{
psTabPos->index += psForm->numMajor;
return true;
}
break;
/* case WFORM_TABNONE - no minor tabs so nothing to display */
}
return false;
}
extern UDWORD realTime; // FIXME Include a header...
/* Run a form widget */
void formRun(W_FORM *psWidget, W_CONTEXT *psContext)
{
SDWORD mx,my;
TAB_POS sTabPos;
char *pTip;
W_TABFORM *psTabForm;
memset(&sTabPos, 0x0, sizeof(TAB_POS));
if(psWidget->style & WFORM_CLICKABLE) {
if(((W_CLICKFORM *)psWidget)->state & WCLICK_FLASH) {
if (((realTime/250) % 2) == 0) {
((W_CLICKFORM *)psWidget)->state &= ~WCLICK_FLASHON;
} else {
((W_CLICKFORM *)psWidget)->state |= WCLICK_FLASHON;
}
}
}
if (psWidget->style & WFORM_TABBED)
{
mx = psContext->mx;
my = psContext->my;
psTabForm = (W_TABFORM *)psWidget;
/* If the mouse is over the form, see if any tabs need to be hilited */
if (mx >= 0 && mx <= psTabForm->width &&
my >= 0 && my <= psTabForm->height)
{
if (formPickTab(psTabForm, mx,my, &sTabPos))
{
if (psTabForm->tabHiLite != (UWORD)sTabPos.index)
{
/* Got a new tab - start the tool tip if there is one */
psTabForm->tabHiLite = (UWORD)sTabPos.index;
if (sTabPos.index >= psTabForm->numMajor)
{
pTip = psTabForm->asMajor[psTabForm->majorT].
asMinor[sTabPos.index - psTabForm->numMajor].pTip;
}
else
{
pTip = psTabForm->asMajor[sTabPos.index].pTip;
}
if (pTip)
{
/* Got a tip - start it off */
tipStart((WIDGET *)psTabForm, pTip,
psContext->psScreen->TipFontID, psTabForm->aColours,
sTabPos.x + psContext->xOffset,
sTabPos.y + psContext->yOffset,
sTabPos.width,sTabPos.height);
}
else
{
/* No tip - clear any old tip */
tipStop((WIDGET *)psWidget);
}
}
}
else
{
/* No tab - clear the tool tip */
tipStop((WIDGET *)psWidget);
/* And clear the hilite */
psTabForm->tabHiLite = (UWORD)(-1);
}
}
}
}
void formSetFlash(W_FORM *psWidget)
{
if(psWidget->style & WFORM_CLICKABLE) {
((W_CLICKFORM *)psWidget)->state |= WCLICK_FLASH;
}
}
void formClearFlash(W_FORM *psWidget)
{
if(psWidget->style & WFORM_CLICKABLE) {
((W_CLICKFORM *)psWidget)->state &= ~WCLICK_FLASH;
((W_CLICKFORM *)psWidget)->state &= ~WCLICK_FLASHON;
}
}
/* Respond to a mouse click */
void formClicked(W_FORM *psWidget, UDWORD key)
{
W_CLICKFORM *psClickForm;
/* Stop the tip if there is one */
tipStop((WIDGET *)psWidget);
if (psWidget->style & WFORM_CLICKABLE)
{
/* Can't click a button if it is disabled or locked down */
if (!(((W_CLICKFORM *)psWidget)->state & (WCLICK_GREY | WCLICK_LOCKED)))
{
// Check this is the correct key
if ((!(psWidget->style & WFORM_NOPRIMARY) && key == WKEY_PRIMARY) ||
((psWidget->style & WFORM_SECONDARY) && key == WKEY_SECONDARY))
{
((W_CLICKFORM *)psWidget)->state &= ~WCLICK_FLASH; // Stop it flashing
((W_CLICKFORM *)psWidget)->state &= ~WCLICK_FLASHON;
((W_CLICKFORM *)psWidget)->state |= WCLICK_DOWN;
psClickForm = (W_CLICKFORM *)psWidget;
if(psClickForm->AudioCallback) {
psClickForm->AudioCallback(psClickForm->ClickedAudioID);
}
}
}
}
}
/* Respond to a mouse form up */
void formReleased(W_FORM *psWidget, UDWORD key, W_CONTEXT *psContext)
{
W_TABFORM *psTabForm;
W_CLICKFORM *psClickForm;
TAB_POS sTabPos;
if (psWidget->style & WFORM_TABBED)
{
psTabForm = (W_TABFORM *)psWidget;
/* See if a tab has been clicked on */
if (formPickTab(psTabForm, psContext->mx,psContext->my, &sTabPos))
{
if (sTabPos.index >= psTabForm->numMajor)
{
/* Clicked on a minor tab */
psTabForm->minorT = (UWORD)(sTabPos.index - psTabForm->numMajor);
psTabForm->asMajor[psTabForm->majorT].lastMinor = psTabForm->minorT;
widgSetReturn(psContext->psScreen, (WIDGET *)psWidget);
}
else
{
/* Clicked on a major tab */
ASSERT(psTabForm->majorT < psTabForm->numMajor,
"formReleased: invalid major id %u >= max %u", sTabPos.index, psTabForm->numMajor);
psTabForm->majorT = (UWORD)sTabPos.index;
psTabForm->minorT = psTabForm->asMajor[sTabPos.index].lastMinor;
widgSetReturn(psContext->psScreen, (WIDGET *)psWidget);
}
}
}
else if (psWidget->style & WFORM_CLICKABLE)
{
psClickForm = (W_CLICKFORM *)psWidget;
if (psClickForm->state & WCLICK_DOWN)
{
// Check this is the correct key
if ((!(psWidget->style & WFORM_NOPRIMARY) && key == WKEY_PRIMARY) ||
((psWidget->style & WFORM_SECONDARY) && key == WKEY_SECONDARY))
{
widgSetReturn(psContext->psScreen, (WIDGET *)psClickForm);
psClickForm->state &= ~WCLICK_DOWN;
}
}
}
}
/* Respond to a mouse moving over a form */
void formHiLite(W_FORM *psWidget, W_CONTEXT *psContext)
{
W_CLICKFORM *psClickForm;
if (psWidget->style & WFORM_CLICKABLE)
{
psClickForm = (W_CLICKFORM *)psWidget;
psClickForm->state |= WCLICK_HILITE;
/* If there is a tip string start the tool tip */
if (psClickForm->pTip)
{
tipStart((WIDGET *)psClickForm, psClickForm->pTip,
psContext->psScreen->TipFontID, psContext->psForm->aColours,
psWidget->x + psContext->xOffset, psWidget->y + psContext->yOffset,
psWidget->width, psWidget->height);
}
if(psClickForm->AudioCallback) {
psClickForm->AudioCallback(psClickForm->HilightAudioID);
}
}
}
/* Respond to the mouse moving off a form */
void formHiLiteLost(W_FORM *psWidget, W_CONTEXT *psContext)
{
/* If one of the widgets were hilited that has to loose it as well */
if (psWidget->psLastHiLite != NULL)
{
widgHiLiteLost(psWidget->psLastHiLite, psContext);
}
if (psWidget->style & WFORM_TABBED)
{
((W_TABFORM *)psWidget)->tabHiLite = (UWORD)(-1);
}
if (psWidget->style & WFORM_CLICKABLE)
{
((W_CLICKFORM *)psWidget)->state &= ~(WCLICK_DOWN | WCLICK_HILITE);
}
/* Clear the tool tip if there is one */
tipStop((WIDGET *)psWidget);
}
/* Display a form */
void formDisplay(WIDGET *psWidget, UDWORD xOffset, UDWORD yOffset, PIELIGHT *pColours)
{
UDWORD x0,y0,x1,y1;
if (!(psWidget->style & WFORM_INVISIBLE))
{
x0 = psWidget->x + xOffset;
y0 = psWidget->y + yOffset;
x1 = x0 + psWidget->width;
y1 = y0 + psWidget->height;
pie_BoxFill(x0 + 1, y0 + 1, x1 - 1, y1 - 1, pColours[WCOL_BKGRND]);
iV_Line(x0,y1,x0,y0, pColours[WCOL_LIGHT]);
iV_Line(x0,y0,x1,y0, pColours[WCOL_LIGHT]);
iV_Line(x1,y0,x1,y1, pColours[WCOL_DARK]);
iV_Line(x1,y1,x0,y1, pColours[WCOL_DARK]);
}
}
/* Display a clickable form */
void formDisplayClickable(WIDGET *psWidget, UDWORD xOffset, UDWORD yOffset, PIELIGHT *pColours)
{
UDWORD x0,y0,x1,y1;
W_CLICKFORM *psForm;
psForm = (W_CLICKFORM *)psWidget;
x0 = psWidget->x + xOffset;
y0 = psWidget->y + yOffset;
x1 = x0 + psWidget->width;
y1 = y0 + psWidget->height;
/* Fill the background */
pie_BoxFill(x0 + 1, y0 + 1, x1 - 1, y1 - 1, pColours[WCOL_BKGRND]);
/* Display the border */
if (psForm->state & (WCLICK_DOWN | WCLICK_LOCKED | WCLICK_CLICKLOCK))
{
/* Form down */
iV_Line(x0,y1,x0,y0, pColours[WCOL_DARK]);
iV_Line(x0,y0,x1,y0, pColours[WCOL_DARK]);
iV_Line(x1,y0,x1,y1, pColours[WCOL_LIGHT]);
iV_Line(x1,y1,x0,y1, pColours[WCOL_LIGHT]);
}
else
{
/* Form up */
iV_Line(x0,y1,x0,y0, pColours[WCOL_LIGHT]);
iV_Line(x0,y0,x1,y0, pColours[WCOL_LIGHT]);
iV_Line(x1,y0,x1,y1, pColours[WCOL_DARK]);
iV_Line(x1,y1,x0,y1, pColours[WCOL_DARK]);
}
}
/* Draw top tabs */
static void formDisplayTTabs(W_TABFORM *psForm,SDWORD x0, SDWORD y0,
UDWORD width, UDWORD height,
UDWORD number, UDWORD selected, UDWORD hilite,
PIELIGHT *pColours,UDWORD TabType,UDWORD TabGap)
{
SDWORD x,x1, y1;
UDWORD i, drawnumber;
#if NO_DISPLAY_SINGLE_TABS
if (number == 1)
{
/* Don't display single tabs */
return;
}
#endif
x = x0 + 2;
x1 = x + width - 2;
y1 = y0 + height;
if (number > (MAX_TAB_SMALL_SHOWN - 1)) //we can display 8 tabs fine with no extra voodoo.
{ // We do NOT want to draw all the tabs once we have drawn 7 tabs
// Both selected & hilite are converted from virtual tab range, to a range
// that is seen on the form itself. This would be 0-6 (7 tabs)
// We also fix drawnumber, so we don't display too many tabs since the pages
// will be empty.
drawnumber = (number - (( psForm->TabMultiplier -1) * TAB_SEVEN));
if (drawnumber > TAB_SEVEN) drawnumber = TAB_SEVEN ;
selected = (selected % TAB_SEVEN); //Go from Virtual range, to our range
if(hilite != 65535) //sigh. Don't blame me for this!It is THEIR 'hack'.
hilite = hilite % TAB_SEVEN; //we want to hilite tab 0 - 6.
}
else
{ // normal draw
drawnumber = number;
}
for (i=0; i < drawnumber; i++)
{
if(psForm->pTabDisplay) {
psForm->pTabDisplay((WIDGET*)psForm,TabType,WFORM_TABTOP,i,i==selected,i==hilite,x,y0,width,height);
} else {
if (i == selected)
{
/* Fill in the tab */
pie_BoxFill(x + 1, y0 + 1, x1 - 1, y1, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x,y0+2, x,y1-1, pColours[WCOL_LIGHT]);
iV_Line(x,y0+2, x+2,y0, pColours[WCOL_LIGHT]);
iV_Line(x+2,y0, x1-1,y0, pColours[WCOL_LIGHT]);
iV_Line(x1,y0+1, x1,y1, pColours[WCOL_DARK]);
}
else
{
/* Fill in the tab */
pie_BoxFill(x + 1, y0 + 2, x1 - 1, y1 - 1, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x,y0+3, x,y1-1, pColours[WCOL_LIGHT]);
iV_Line(x,y0+3, x+2,y0+1, pColours[WCOL_LIGHT]);
iV_Line(x+2,y0+1, x1-1,y0+1, pColours[WCOL_LIGHT]);
iV_Line(x1,y0+2, x1,y1-1, pColours[WCOL_DARK]);
}
if (i == hilite)
{
/* Draw the hilite box */
iV_Box(x+2,y0+4, x1-3, y1-3, pColours[WCOL_HILITE]);
}
}
x += width + TabGap;
x1 += width + TabGap;
}
}
/* Draw bottom tabs */
static void formDisplayBTabs(W_TABFORM *psForm,SDWORD x0, SDWORD y0,
UDWORD width, UDWORD height,
UDWORD number, UDWORD selected, UDWORD hilite,
PIELIGHT *pColours,UDWORD TabType,UDWORD TabGap)
{
SDWORD x,x1, y1;
UDWORD i;
#if NO_DISPLAY_SINGLE_TABS
if (number == 1)
{
/* Don't display single tabs */
return;
}
#endif
x = x0 + 2;
x1 = x + width - 2;
y1 = y0 + height;
for (i=0; i < number; i++)
{
if(psForm->pTabDisplay) {
psForm->pTabDisplay((WIDGET*)psForm,TabType,WFORM_TABBOTTOM,i,i==selected,i==hilite,x,y0,width,height);
} else {
if (i == selected)
{
/* Fill in the tab */
pie_BoxFill(x + 1, y0, x1 - 1, y1 - 1, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x,y0, x,y1-1, pColours[WCOL_LIGHT]);
iV_Line(x,y1, x1-3,y1, pColours[WCOL_DARK]);
iV_Line(x1-2,y1, x1,y1-2, pColours[WCOL_DARK]);
iV_Line(x1,y1-3, x1,y0+1, pColours[WCOL_DARK]);
}
else
{
/* Fill in the tab */
pie_BoxFill(x + 1, y0 + 1, x1 - 1, y1 - 2, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x,y0+1, x,y1-1, pColours[WCOL_LIGHT]);
iV_Line(x+1,y1-1, x1-3,y1-1, pColours[WCOL_DARK]);
iV_Line(x1-2,y1-1, x1,y1-3, pColours[WCOL_DARK]);
iV_Line(x1,y1-4, x1,y0+1, pColours[WCOL_DARK]);
}
if (i == hilite)
{
/* Draw the hilite box */
iV_Box(x+2,y0+3, x1-3, y1-4, pColours[WCOL_HILITE]);
}
}
x += width + TabGap;
x1 += width + TabGap;
}
}
/* Draw left tabs */
static void formDisplayLTabs(W_TABFORM *psForm,SDWORD x0, SDWORD y0,
UDWORD width, UDWORD height,
UDWORD number, UDWORD selected, UDWORD hilite,
PIELIGHT *pColours,UDWORD TabType,UDWORD TabGap)
{
SDWORD x1, y,y1;
UDWORD i;
#if NO_DISPLAY_SINGLE_TABS
if (number == 1)
{
/* Don't display single tabs */
return;
}
#endif
x1 = x0 + width;
y = y0+2;
y1 = y + height - 2;
for (i=0; i < number; i++)
{
if(psForm->pTabDisplay) {
psForm->pTabDisplay((WIDGET*)psForm,TabType,WFORM_TABLEFT,i,i==selected,i==hilite,x0,y,width,height);
} else {
if (i == selected)
{
/* Fill in the tab */
pie_BoxFill(x0 + 1, y + 1, x1, y1 - 1, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x0,y, x1-1,y, pColours[WCOL_LIGHT]);
iV_Line(x0,y+1, x0,y1-2, pColours[WCOL_LIGHT]);
iV_Line(x0+1,y1-1, x0+2,y1, pColours[WCOL_DARK]);
iV_Line(x0+3,y1, x1,y1, pColours[WCOL_DARK]);
}
else
{
/* Fill in the tab */
pie_BoxFill(x0 + 2, y + 1, x1 - 1, y1 - 1, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x0+1,y, x1-1,y, pColours[WCOL_LIGHT]);
iV_Line(x0+1,y+1, x0+1,y1-2, pColours[WCOL_LIGHT]);
iV_Line(x0+2,y1-1, x0+3,y1, pColours[WCOL_DARK]);
iV_Line(x0+4,y1, x1-1,y1, pColours[WCOL_DARK]);
}
if (i == hilite)
{
iV_Box(x0+4,y+2, x1-2, y1-3, pColours[WCOL_HILITE]);
}
}
y += height + TabGap;
y1 += height + TabGap;
}
}
/* Draw right tabs */
static void formDisplayRTabs(W_TABFORM *psForm,SDWORD x0, SDWORD y0,
UDWORD width, UDWORD height,
UDWORD number, UDWORD selected, UDWORD hilite,
PIELIGHT *pColours,UDWORD TabType,UDWORD TabGap)
{
SDWORD x1, y,y1;
UDWORD i;
#if NO_DISPLAY_SINGLE_TABS
if (number == 1)
{
/* Don't display single tabs */
return;
}
#endif
x1 = x0 + width;
y = y0+2;
y1 = y + height - 2;
for (i=0; i < number; i++)
{
if(psForm->pTabDisplay) {
psForm->pTabDisplay((WIDGET*)psForm,TabType,WFORM_TABRIGHT,i,i==selected,i==hilite,x0,y,width,height);
} else {
if (i == selected)
{
/* Fill in the tab */
pie_BoxFill(x0,y+1, x1-1,y1-1, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x0,y, x1-1,y, pColours[WCOL_LIGHT]);
iV_Line(x1,y, x1,y1-2, pColours[WCOL_DARK]);
iV_Line(x1-1,y1-1, x1-2,y1, pColours[WCOL_DARK]);
iV_Line(x1-3,y1, x0,y1, pColours[WCOL_DARK]);
}
else
{
/* Fill in the tab */
pie_BoxFill(x0+1,y+1, x1-2,y1-1, pColours[WCOL_BKGRND]);
/* Draw the outline */
iV_Line(x0+1,y, x1-1,y, pColours[WCOL_LIGHT]);
iV_Line(x1-1,y, x1-1,y1-2, pColours[WCOL_DARK]);
iV_Line(x1-2,y1-1, x1-3,y1, pColours[WCOL_DARK]);
iV_Line(x1-4,y1, x0+1,y1, pColours[WCOL_DARK]);
}
if (i == hilite)
{
iV_Box(x0+2,y+2, x1-4, y1-3, pColours[WCOL_HILITE]);
}
}
y += height + TabGap;
y1 += height + TabGap;
}
}
/* Display a tabbed form */
void formDisplayTabbed(WIDGET *psWidget, UDWORD xOffset, UDWORD yOffset, PIELIGHT *pColours)
{
UDWORD x0,y0,x1,y1;
W_TABFORM *psForm;
W_MAJORTAB *psMajor;
psForm = (W_TABFORM *)psWidget;
/* Get the basic position of the form */
x0 = psForm->x + xOffset;
y0 = psForm->y + yOffset;
x1 = x0 + psForm->width;
y1 = y0 + psForm->height;
/* Adjust for where the tabs are */
if(psForm->majorPos == WFORM_TABLEFT) {
x0 += psForm->tabMajorThickness - psForm->tabHorzOffset;
} else if(psForm->minorPos == WFORM_TABLEFT) {
x0 += psForm->tabMinorThickness - psForm->tabHorzOffset;
}
if(psForm->majorPos == WFORM_TABRIGHT) {
x1 -= psForm->tabMajorThickness - psForm->tabHorzOffset;
} else if(psForm->minorPos == WFORM_TABRIGHT) {
x1 -= psForm->tabMinorThickness - psForm->tabHorzOffset;
}
if(psForm->majorPos == WFORM_TABTOP) {
y0 += psForm->tabMajorThickness - psForm->tabVertOffset;
} else if(psForm->minorPos == WFORM_TABTOP) {
y0 += psForm->tabMinorThickness - psForm->tabVertOffset;
}
if(psForm->majorPos == WFORM_TABBOTTOM) {
y1 -= psForm->tabMajorThickness - psForm->tabVertOffset;
} else if(psForm->minorPos == WFORM_TABBOTTOM) {
y1 -= psForm->tabMinorThickness - psForm->tabVertOffset;
}
/* Draw the major tabs */
switch (psForm->majorPos)
{
case WFORM_TABTOP:
formDisplayTTabs(psForm,x0+psForm->majorOffset, y0 - psForm->tabMajorThickness + psForm->tabVertOffset,
psForm->majorSize, psForm->tabMajorThickness,
psForm->numMajor, psForm->majorT, psForm->tabHiLite,
pColours,TAB_MAJOR,psForm->tabMajorGap);
break;
case WFORM_TABBOTTOM:
formDisplayBTabs(psForm,x0+psForm->majorOffset, y1 + psForm->tabVertOffset,
psForm->majorSize, psForm->tabMajorThickness,
psForm->numMajor, psForm->majorT, psForm->tabHiLite,
pColours,TAB_MAJOR,psForm->tabMajorGap);
break;
case WFORM_TABLEFT:
formDisplayLTabs(psForm,x0 - psForm->tabMajorThickness + psForm->tabHorzOffset, y0+psForm->majorOffset,
psForm->tabMajorThickness, psForm->majorSize,
psForm->numMajor, psForm->majorT, psForm->tabHiLite,
pColours,TAB_MAJOR,psForm->tabMajorGap);
break;
case WFORM_TABRIGHT:
formDisplayRTabs(psForm,x1 - psForm->tabHorzOffset, y0+psForm->majorOffset,
psForm->tabMajorThickness, psForm->majorSize,
psForm->numMajor, psForm->majorT, psForm->tabHiLite,
pColours,TAB_MAJOR,psForm->tabMajorGap);
break;
case WFORM_TABNONE:
ASSERT( false, "formDisplayTabbed: Cannot have a tabbed form with no major tabs" );
break;
}
/* Draw the minor tabs */
psMajor = psForm->asMajor + psForm->majorT;
switch (psForm->minorPos)
{
case WFORM_TABTOP:
formDisplayTTabs(psForm,x0 + psForm->minorOffset, y0 - psForm->tabMinorThickness + psForm->tabVertOffset,
psForm->minorSize, psForm->tabMinorThickness,
psMajor->numMinor, psForm->minorT,
psForm->tabHiLite - psForm->numMajor, pColours,TAB_MINOR,psForm->tabMinorGap);
break;
case WFORM_TABBOTTOM:
formDisplayBTabs(psForm,x0 + psForm->minorOffset, y1 + psForm->tabVertOffset,
psForm->minorSize, psForm->tabMinorThickness,
psMajor->numMinor, psForm->minorT,
psForm->tabHiLite - psForm->numMajor, pColours,TAB_MINOR,psForm->tabMinorGap);
break;
case WFORM_TABLEFT:
formDisplayLTabs(psForm,x0 - psForm->tabMinorThickness + psForm->tabHorzOffset + psForm->minorOffset,
y0+psForm->minorOffset,
psForm->tabMinorThickness, psForm->minorSize,
psMajor->numMinor, psForm->minorT,
psForm->tabHiLite - psForm->numMajor, pColours,TAB_MINOR,psForm->tabMinorGap);
break;
case WFORM_TABRIGHT:
formDisplayRTabs(psForm,x1 + psForm->tabHorzOffset, y0+psForm->minorOffset,
psForm->tabMinorThickness, psForm->minorSize,
psMajor->numMinor, psForm->minorT,
psForm->tabHiLite - psForm->numMajor, pColours,TAB_MINOR,psForm->tabMinorGap);
break;
/* case WFORM_TABNONE - no minor tabs so nothing to display */
}
}