jedioutcast/code/ui/ui_shared.cpp
2013-04-04 09:52:42 -05:00

8499 lines
173 KiB
C++

//
// string allocation/managment
// leave this at the top of all UI_xxxx files for PCH reasons...
//
#include "../server/exe_headers.h"
#include "ui_local.h"
#include "ui_shared.h"
#include "menudef.h"
void *UI_Alloc( int size );
void Controls_GetConfig( void );
void Fade(int *flags, float *f, float clamp, int *nextTime, int offsetTime, qboolean bFlags, float fadeAmount);
void Item_Init(itemDef_t *item);
void Item_InitControls(itemDef_t *item);
qboolean Item_Parse(itemDef_t *item);
void Item_RunScript(itemDef_t *item, const char *s);
void Item_SetupKeywordHash(void);
void Item_Text_AutoWrapped_Paint(itemDef_t *item);
void Item_UpdatePosition(itemDef_t *item);
void Item_ValidateTypeData(itemDef_t *item);
void LerpColor(vec4_t a, vec4_t b, vec4_t c, float t);
itemDef_t *Menu_FindItemByName(menuDef_t *menu, const char *p);
itemDef_t *Menu_GetMatchingItemByNumber(menuDef_t *menu, int index, const char *name);
void Menu_Paint(menuDef_t *menu, qboolean forcePaint);
void Menu_SetupKeywordHash(void);
void Menus_ShowItems(const char *menuName);
qboolean ParseRect(const char **p, rectDef_t *r);
const char *String_Alloc(const char *p);
void ToWindowCoords(float *x, float *y, windowDef_t *window);
void Window_Paint(Window *w, float fadeAmount, float fadeClamp, float fadeCycle);
int Item_ListBox_ThumbDrawPosition(itemDef_t *item);
int Item_ListBox_ThumbPosition(itemDef_t *item);
//static qboolean debugMode = qfalse;
static qboolean g_waitingForKey = qfalse;
static qboolean g_editingField = qfalse;
static itemDef_t *g_bindItem = NULL;
static itemDef_t *g_editItem = NULL;
static itemDef_t *itemCapture = NULL; // item that has the mouse captured ( if any )
#define DOUBLE_CLICK_DELAY 300
static int lastListBoxClickTime = 0;
static void (*captureFunc) (void *p) = NULL;
static void *captureData = NULL;
char defaultString[10] =
{"default"};
#ifdef CGAME
#define MEM_POOL_SIZE 128 * 1024
#else
#define MEM_POOL_SIZE 1024 * 1024
#endif
#define SCROLL_TIME_START 500
#define SCROLL_TIME_ADJUST 150
#define SCROLL_TIME_ADJUSTOFFSET 40
#define SCROLL_TIME_FLOOR 20
typedef struct scrollInfo_s {
int nextScrollTime;
int nextAdjustTime;
int adjustValue;
int scrollKey;
float xStart;
float yStart;
itemDef_t *item;
qboolean scrollDir;
} scrollInfo_t;
static scrollInfo_t scrollInfo;
static char memoryPool[MEM_POOL_SIZE];
static int allocPoint, outOfMemory;
displayContextDef_t *DC = NULL;
menuDef_t Menus[MAX_MENUS]; // defined menus
int menuCount = 0; // how many
menuDef_t *menuStack[MAX_OPEN_MENUS];
int openMenuCount = 0;
static int strPoolIndex = 0;
static char strPool[STRING_POOL_SIZE];
typedef struct stringDef_s {
struct stringDef_s *next;
const char *str;
} stringDef_t;
#define HASH_TABLE_SIZE 2048
static int strHandleCount = 0;
static stringDef_t *strHandle[HASH_TABLE_SIZE];
typedef struct itemFlagsDef_s {
char *string;
int value;
} itemFlagsDef_t;
itemFlagsDef_t itemFlags [] = {
"WINDOW_INACTIVE", WINDOW_INACTIVE,
NULL, NULL
};
char *styles [] = {
"WINDOW_STYLE_EMPTY",
"WINDOW_STYLE_FILLED",
"WINDOW_STYLE_GRADIENT",
"WINDOW_STYLE_SHADER",
"WINDOW_STYLE_TEAMCOLOR",
"WINDOW_STYLE_CINEMATIC",
NULL
};
char *types [] = {
"ITEM_TYPE_TEXT",
"ITEM_TYPE_BUTTON",
"ITEM_TYPE_RADIOBUTTON",
"ITEM_TYPE_CHECKBOX",
"ITEM_TYPE_EDITFIELD",
"ITEM_TYPE_COMBO",
"ITEM_TYPE_LISTBOX",
"ITEM_TYPE_OWNERDRAW",
"ITEM_TYPE_MODEL",
"ITEM_TYPE_NUMERICFIELD",
"ITEM_TYPE_SLIDER",
"ITEM_TYPE_YESNO",
"ITEM_TYPE_MULTI",
"ITEM_TYPE_BIND",
NULL
};
char *alignment [] = {
"ITEM_ALIGN_LEFT",
"ITEM_ALIGN_CENTER",
"ITEM_ALIGN_RIGHT",
NULL
};
/*
==================
Init_Display
Initializes the display with a structure to all the drawing routines
==================
*/
void Init_Display(displayContextDef_t *dc)
{
DC = dc;
}
/*
==================
Window_Init
Initializes a window structure ( windowDef_t ) with defaults
==================
*/
void Window_Init(Window *w)
{
memset(w, 0, sizeof(windowDef_t));
w->borderSize = 1;
w->foreColor[0] = w->foreColor[1] = w->foreColor[2] = w->foreColor[3] = 1.0;
w->cinematic = -1;
}
/*
=================
PC_SourceError
=================
*/
void PC_SourceError(int handle, char *format, ...)
{
int line;
char filename[128];
va_list argptr;
static char string[4096];
va_start (argptr, format);
vsprintf (string, format, argptr);
va_end (argptr);
filename[0] = '\0';
line = 0;
Com_Printf(S_COLOR_RED "ERROR: %s, line %d: %s\n", filename, line, string);
}
/*
=================
PC_ParseStringMem
=================
*/
qboolean PC_ParseStringMem(const char **out)
{
const char *temp;
if (PC_ParseString(&temp))
{
return qfalse;
}
if (*temp == '@') // Is it a localized text?
{
const int ID = SP_GetStringID(va("%s_%s",uiInfo.uiDC.Assets.stripedFile,(temp+1))); // The +1 is to offset the @ at the beginning of the text
if (ID != -1)
{
*(out) = (char*)-ID;
return ID;
}
PC_ParseWarning(va("Can't find StriP '%s' in %s", temp, uiInfo.uiDC.Assets.stripedFile));
}
*(out) = String_Alloc(temp);
return qtrue;
}
/*
=================
PC_ParseRect
=================
*/
qboolean PC_ParseRect(rectDef_t *r)
{
if (!PC_ParseFloat(&r->x))
{
if (!PC_ParseFloat(&r->y))
{
if (!PC_ParseFloat(&r->w))
{
if (!PC_ParseFloat(&r->h))
{
return qtrue;
}
}
}
}
return qfalse;
}
/*
=================
PC_Script_Parse
=================
*/
qboolean PC_Script_Parse(const char **out)
{
char script[1024];
// pc_token_t token;
char *token2;
memset(script, 0, sizeof(script));
// scripts start with { and have ; separated command lists.. commands are command, arg..
// basically we want everything between the { } as it will be interpreted at run time
token2 = PC_ParseExt();
if (!token2)
{
return qfalse;
}
if (*token2 !='{')
{
return qfalse;
}
while ( 1 )
{
token2 = PC_ParseExt();
if (!token2)
{
return qfalse;
}
if (*token2 =='}') // End of the script?
{
*out = String_Alloc(script);
return qtrue;
}
if (*(token2 +1) != '\0')
{
Q_strcat(script, 1024, va("\"%s\"", token2));
}
else
{
Q_strcat(script, 1024, token2);
}
Q_strcat(script, 1024, " ");
}
}
//--------------------------------------------------------------------------------------------
// Menu Keyword Parse functions
//--------------------------------------------------------------------------------------------
/*
=================
MenuParse_font
=================
*/
qboolean MenuParse_font( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_ParseStringMem(&menu->font))
{
return qfalse;
}
if (!DC->Assets.fontRegistered)
{
DC->Assets.qhMediumFont = DC->registerFont(menu->font);
DC->Assets.fontRegistered = qtrue;
}
return qtrue;
}
/*
=================
MenuParse_name
=================
*/
qboolean MenuParse_name(itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_ParseStringMem((const char **) &menu->window.name))
{
return qfalse;
}
// if (Q_stricmp(menu->window.name, "main") == 0)
// {
// default main as having focus
// menu->window.flags |= WINDOW_HASFOCUS;
// }
return qtrue;
}
/*
=================
MenuParse_fullscreen
=================
*/
qboolean MenuParse_fullscreen( itemDef_t *item )
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt((int *) &menu->fullScreen))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_rect
=================
*/
qboolean MenuParse_rect( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_ParseRect(&menu->window.rect))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_style
=================
*/
qboolean MenuParse_style( itemDef_t *item)
{
int i;
char *tempStr;
menuDef_t *menu = (menuDef_t*)item;
// if (PC_ParseInt(&menu->window.style))
if (!PC_ParseStringMem((const char **) &tempStr))
{
return qfalse;
}
i=0;
while (styles[i])
{
if (Q_stricmp(tempStr,styles[i])==0)
{
menu->window.style = i;
break;
}
i++;
}
if (styles[i] == NULL)
{
PC_ParseWarning(va("Unknown menu style value '%s'",tempStr));
}
return qtrue;
}
/*
=================
MenuParse_visible
=================
*/
qboolean MenuParse_visible( itemDef_t *item )
{
int i;
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt(&i))
{
return qfalse;
}
if (i)
{
menu->window.flags |= WINDOW_VISIBLE;
}
return qtrue;
}
/*
=================
MenuParse_onOpen
=================
*/
qboolean MenuParse_onOpen( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_Script_Parse(&menu->onOpen))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_onClose
=================
*/
qboolean MenuParse_onClose( itemDef_t *item )
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_Script_Parse(&menu->onClose))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_onESC
=================
*/
qboolean MenuParse_onESC( itemDef_t *item )
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_Script_Parse(&menu->onESC))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_border
=================
*/
qboolean MenuParse_border( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt(&menu->window.border))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_borderSize
=================
*/
qboolean MenuParse_borderSize( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseFloat(&menu->window.borderSize))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_backcolor
=================
*/
qboolean MenuParse_backcolor( itemDef_t *item )
{
int i;
float f;
menuDef_t *menu = (menuDef_t*)item;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
menu->window.backColor[i] = f;
}
return qtrue;
}
/*
=================
MenuParse_forecolor
=================
*/
qboolean MenuParse_forecolor( itemDef_t *item)
{
int i;
float f;
menuDef_t *menu = (menuDef_t*)item;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
menu->window.foreColor[i] = f;
menu->window.flags |= WINDOW_FORECOLORSET;
}
return qtrue;
}
/*
=================
MenuParse_bordercolor
=================
*/
qboolean MenuParse_bordercolor( itemDef_t *item )
{
int i;
float f;
menuDef_t *menu = (menuDef_t*)item;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
menu->window.borderColor[i] = f;
}
return qtrue;
}
/*
=================
MenuParse_focuscolor
=================
*/
qboolean MenuParse_focuscolor( itemDef_t *item)
{
int i;
float f;
menuDef_t *menu = (menuDef_t*)item;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
menu->focusColor[i] = f;
}
return qtrue;
}
/*
=================
MenuParse_focuscolor
=================
*/
qboolean MenuParse_appearanceIncrement( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseFloat(&menu->appearanceIncrement))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_descAlignment
=================
*/
qboolean MenuParse_descAlignment( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
char *tempStr;
int i;
if (!PC_ParseStringMem((const char **) &tempStr))
{
return qfalse;
}
i=0;
while (alignment[i])
{
if (Q_stricmp(tempStr,alignment[i])==0)
{
menu->descAlignment = i;
break;
}
i++;
}
if (alignment[i] == NULL)
{
PC_ParseWarning(va("Unknown desc alignment value '%s'",tempStr));
}
return qtrue;
}
/*
=================
MenuParse_descX
=================
*/
qboolean MenuParse_descX( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt(&menu->descX))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_descY
=================
*/
qboolean MenuParse_descY( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt(&menu->descY))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_descScale
=================
*/
qboolean MenuParse_descScale( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseFloat(&menu->descScale))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_descColor
=================
*/
qboolean MenuParse_descColor( itemDef_t *item)
{
int i;
float f;
menuDef_t *menu = (menuDef_t*)item;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
menu->descColor[i] = f;
}
return qtrue;
}
/*
=================
MenuParse_disablecolor
=================
*/
qboolean MenuParse_disablecolor( itemDef_t *item)
{
int i;
float f;
menuDef_t *menu = (menuDef_t*)item;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
menu->disableColor[i] = f;
}
return qtrue;
}
/*
=================
MenuParse_outlinecolor
=================
*/
qboolean MenuParse_outlinecolor( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseColor(&menu->window.outlineColor))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_background
=================
*/
qboolean MenuParse_background( itemDef_t *item)
{
const char *buff;
menuDef_t *menu = (menuDef_t*)item;
if (!PC_ParseStringMem((const char **) &buff))
{
return qfalse;
}
menu->window.background = ui.R_RegisterShaderNoMip(buff);
return qtrue;
}
/*
=================
MenuParse_cinematic
=================
*/
qboolean MenuParse_cinematic( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_ParseStringMem((const char **) &menu->window.cinematicName))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_ownerdrawFlag
=================
*/
qboolean MenuParse_ownerdrawFlag( itemDef_t *item)
{
int i;
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt(&i))
{
return qfalse;
}
menu->window.ownerDrawFlags |= i;
return qtrue;
}
/*
=================
MenuParse_ownerdraw
=================
*/
qboolean MenuParse_ownerdraw( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt(&menu->window.ownerDraw))
{
return qfalse;
}
return qtrue;
}
/*
=================
MenuParse_popup
=================
*/
qboolean MenuParse_popup( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
menu->window.flags |= WINDOW_POPUP;
return qtrue;
}
/*
=================
MenuParse_outOfBounds
=================
*/
qboolean MenuParse_outOfBounds( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
menu->window.flags |= WINDOW_OOB_CLICK;
return qtrue;
}
/*
=================
MenuParse_soundLoop
=================
*/
qboolean MenuParse_soundLoop( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (!PC_ParseStringMem((const char **) &menu->soundName))
{
return qfalse;
}
return qtrue;
}
/*
================
MenuParse_fadeClamp
================
*/
qboolean MenuParse_fadeClamp( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseFloat(&menu->fadeClamp))
{
return qfalse;
}
return qtrue;
}
/*
================
MenuParse_fadeAmount
================
*/
qboolean MenuParse_fadeAmount( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseFloat(&menu->fadeAmount))
{
return qfalse;
}
return qtrue;
}
/*
================
MenuParse_fadeCycle
================
*/
qboolean MenuParse_fadeCycle( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (PC_ParseInt(&menu->fadeCycle))
{
return qfalse;
}
return qtrue;
}
/*
================
MenuParse_itemDef
================
*/
qboolean MenuParse_itemDef( itemDef_t *item)
{
menuDef_t *menu = (menuDef_t*)item;
if (menu->itemCount < MAX_MENUITEMS)
{
menu->items[menu->itemCount] = (struct itemDef_s *) UI_Alloc(sizeof(itemDef_t));
Item_Init(menu->items[menu->itemCount]);
if (!Item_Parse(menu->items[menu->itemCount]))
{
return qfalse;
}
Item_InitControls(menu->items[menu->itemCount]);
menu->items[menu->itemCount++]->parent = menu;
}
else
{
PC_ParseWarning(va("Exceeded item/menu max of %d", MAX_MENUITEMS));
}
return qtrue;
}
#define KEYWORDHASH_SIZE 512
typedef struct keywordHash_s
{
char *keyword;
qboolean (*func)(itemDef_t *item);
struct keywordHash_s *next;
} keywordHash_t;
keywordHash_t menuParseKeywords[] = {
{"appearanceIncrement", MenuParse_appearanceIncrement},
{"backcolor", MenuParse_backcolor, },
{"background", MenuParse_background, },
{"border", MenuParse_border, },
{"bordercolor", MenuParse_bordercolor, },
{"borderSize", MenuParse_borderSize, },
{"cinematic", MenuParse_cinematic, },
{"descAlignment", MenuParse_descAlignment },
{"desccolor", MenuParse_descColor },
{"descX", MenuParse_descX },
{"descY", MenuParse_descY },
{"descScale", MenuParse_descScale },
{"disablecolor", MenuParse_disablecolor, },
{"fadeClamp", MenuParse_fadeClamp, },
{"fadeCycle", MenuParse_fadeCycle, },
{"fadeAmount", MenuParse_fadeAmount, },
{"focuscolor", MenuParse_focuscolor, },
{"font", MenuParse_font, },
{"forecolor", MenuParse_forecolor, },
{"fullscreen", MenuParse_fullscreen, },
{"itemDef", MenuParse_itemDef, },
{"name", MenuParse_name, },
{"onClose", MenuParse_onClose, },
{"onESC", MenuParse_onESC, },
{"onOpen", MenuParse_onOpen, },
{"outlinecolor", MenuParse_outlinecolor, },
{"outOfBoundsClick", MenuParse_outOfBounds, },
{"ownerdraw", MenuParse_ownerdraw, },
{"ownerdrawFlag", MenuParse_ownerdrawFlag,},
{"popup", MenuParse_popup, },
{"rect", MenuParse_rect, },
{"soundLoop", MenuParse_soundLoop, },
{"style", MenuParse_style, },
{"visible", MenuParse_visible, },
{NULL, NULL, }
};
keywordHash_t *menuParseKeywordHash[KEYWORDHASH_SIZE];
/*
================
KeywordHash_Key
================
*/
int KeywordHash_Key(char *keyword)
{
int register hash, i;
hash = 0;
for (i = 0; keyword[i] != '\0'; i++) {
if (keyword[i] >= 'A' && keyword[i] <= 'Z')
hash += (keyword[i] + ('a' - 'A')) * (119 + i);
else
hash += keyword[i] * (119 + i);
}
hash = (hash ^ (hash >> 10) ^ (hash >> 20)) & (KEYWORDHASH_SIZE-1);
return hash;
}
/*
================
KeywordHash_Add
================
*/
void KeywordHash_Add(keywordHash_t *table[], keywordHash_t *key)
{
int hash;
hash = KeywordHash_Key(key->keyword);
key->next = table[hash];
table[hash] = key;
}
/*
===============
KeywordHash_Find
===============
*/
keywordHash_t *KeywordHash_Find(keywordHash_t *table[], char *keyword)
{
keywordHash_t *key;
int hash;
hash = KeywordHash_Key(keyword);
for (key = table[hash]; key; key = key->next)
{
if (!Q_stricmp(key->keyword, keyword))
return key;
}
return NULL;
}
/*
================
hashForString
return a hash value for the string
================
*/
static long hashForString(const char *str)
{
int i;
long hash;
char letter;
hash = 0;
i = 0;
while (str[i] != '\0')
{
letter = tolower(str[i]);
hash+=(long)(letter)*(i+119);
i++;
}
hash &= (HASH_TABLE_SIZE-1);
return hash;
}
/*
=================
String_Alloc
=================
*/
const char *String_Alloc(const char *p)
{
int len;
long hash;
stringDef_t *str, *last;
static const char *staticNULL = "";
if (p == NULL)
{
return NULL;
}
if (*p == 0)
{
return staticNULL;
}
hash = hashForString(p);
str = strHandle[hash];
while (str)
{
if (strcmp(p, str->str) == 0)
{
return str->str;
}
str = str->next;
}
len = strlen(p);
if (len + strPoolIndex + 1 < STRING_POOL_SIZE)
{
int ph = strPoolIndex;
strcpy(&strPool[strPoolIndex], p);
strPoolIndex += len + 1;
str = strHandle[hash];
last = str;
while (str && str->next)
{
last = str;
str = str->next;
}
str = (stringDef_s *) UI_Alloc( sizeof(stringDef_t));
str->next = NULL;
str->str = &strPool[ph];
if (last)
{
last->next = str;
}
else
{
strHandle[hash] = str;
}
return &strPool[ph];
}
return NULL;
}
/*
=================
String_Report
=================
*/
void String_Report(void)
{
float f;
Com_Printf("Memory/String Pool Info\n");
Com_Printf("----------------\n");
f = strPoolIndex;
f /= STRING_POOL_SIZE;
f *= 100;
Com_Printf("String Pool is %.1f%% full, %i bytes out of %i used.\n", f, strPoolIndex, STRING_POOL_SIZE);
f = allocPoint;
f /= MEM_POOL_SIZE;
f *= 100;
Com_Printf("Memory Pool is %.1f%% full, %i bytes out of %i used.\n", f, allocPoint, MEM_POOL_SIZE);
}
/*
=================
String_Init
=================
*/
void String_Init(void)
{
int i;
for (i = 0; i < HASH_TABLE_SIZE; i++)
{
strHandle[i] = 0;
}
strHandleCount = 0;
strPoolIndex = 0;
UI_InitMemory();
Item_SetupKeywordHash();
Menu_SetupKeywordHash();
if (DC && DC->getBindingBuf)
{
Controls_GetConfig();
}
}
//---------------------------------------------------------------------------------------------------------
// Memory
//---------------------------------------------------------------------------------------------------------
/*
===============
UI_Alloc
===============
*/
void *UI_Alloc( int size )
{
char *p;
if ( allocPoint + size > MEM_POOL_SIZE )
{
outOfMemory = qtrue;
if (DC->Print)
{
DC->Print("UI_Alloc: Failure. Out of memory!\n");
}
return NULL;
}
p = &memoryPool[allocPoint];
allocPoint += ( size + 15 ) & ~15;
return p;
}
/*
===============
UI_InitMemory
===============
*/
void UI_InitMemory( void )
{
allocPoint = 0;
outOfMemory = qfalse;
}
/*
===============
Menu_ItemsMatchingGroup
===============
*/
int Menu_ItemsMatchingGroup(menuDef_t *menu, const char *name)
{
int i;
int count = 0;
for (i = 0; i < menu->itemCount; i++)
{
if ((!menu->items[i]->window.name) && (!menu->items[i]->window.group))
{
Com_Printf(S_COLOR_YELLOW"WARNING: item has neither name or group\n");
continue;
}
if (Q_stricmp(menu->items[i]->window.name, name) == 0 || (menu->items[i]->window.group && Q_stricmp(menu->items[i]->window.group, name) == 0))
{
count++;
}
}
return count;
}
/*
===============
Menu_GetMatchingItemByNumber
===============
*/
itemDef_t *Menu_GetMatchingItemByNumber(menuDef_t *menu, int index, const char *name)
{
int i;
int count = 0;
for (i = 0; i < menu->itemCount; i++)
{
if (Q_stricmp(menu->items[i]->window.name, name) == 0 || (menu->items[i]->window.group && Q_stricmp(menu->items[i]->window.group, name) == 0))
{
if (count == index)
{
return menu->items[i];
}
count++;
}
}
return NULL;
}
/*
===============
Menu_FadeItemByName
===============
*/
void Menu_FadeItemByName(menuDef_t *menu, const char *p, qboolean fadeOut)
{
itemDef_t *item;
int i;
int count = Menu_ItemsMatchingGroup(menu, p);
for (i = 0; i < count; i++)
{
item = Menu_GetMatchingItemByNumber(menu, i, p);
if (item != NULL)
{
if (fadeOut)
{
item->window.flags |= (WINDOW_FADINGOUT | WINDOW_VISIBLE);
item->window.flags &= ~WINDOW_FADINGIN;
}
else
{
item->window.flags |= (WINDOW_VISIBLE | WINDOW_FADINGIN);
item->window.flags &= ~WINDOW_FADINGOUT;
}
}
}
}
/*
===============
Menu_ShowItemByName
===============
*/
void Menu_ShowItemByName(menuDef_t *menu, const char *p, qboolean bShow)
{
itemDef_t *item;
int i;
int count;
count = Menu_ItemsMatchingGroup(menu, p);
if (!count)
{
Com_Printf(S_COLOR_YELLOW"WARNING: Menu_ShowItemByName - unable to locate any items named :%s\n",p);
}
for (i = 0; i < count; i++)
{
item = Menu_GetMatchingItemByNumber(menu, i, p);
if (item != NULL)
{
if (bShow)
{
item->window.flags |= WINDOW_VISIBLE;
}
else
{
item->window.flags &= ~WINDOW_VISIBLE;
// stop cinematics playing in the window
if (item->window.cinematic >= 0)
{
DC->stopCinematic(item->window.cinematic);
item->window.cinematic = -1;
}
}
}
}
}
/*
===============
Menu_GetFocused
===============
*/
menuDef_t *Menu_GetFocused(void)
{
int i;
for (i = 0; i < menuCount; i++)
{
if ((Menus[i].window.flags & WINDOW_HASFOCUS) && (Menus[i].window.flags & WINDOW_VISIBLE))
{
return &Menus[i];
}
}
return NULL;
}
/*
===============
Menus_OpenByName
===============
*/
void Menus_OpenByName(const char *p)
{
Menus_ActivateByName(p);
}
/*
===============
Menus_FindByName
===============
*/
menuDef_t *Menus_FindByName(const char *p)
{
int i;
for (i = 0; i < menuCount; i++)
{
if (Q_stricmp(Menus[i].window.name, p) == 0)
{
return &Menus[i];
}
}
return NULL;
}
/*
===============
Menu_RunCloseScript
===============
*/
static void Menu_RunCloseScript(menuDef_t *menu)
{
if (menu && menu->window.flags & WINDOW_VISIBLE && menu->onClose)
{
itemDef_t item;
item.parent = menu;
Item_RunScript(&item, menu->onClose);
}
}
/*
===============
Item_ActivateByName
===============
*/
void Item_ActivateByName(const char *menuName,const char *itemName)
{
itemDef_t *item;
menuDef_t *menu;
menu = Menus_FindByName(menuName);
item = (itemDef_s *) Menu_FindItemByName((menuDef_t *) menu, itemName);
if (item != NULL)
{
item->window.flags &= ~WINDOW_INACTIVE;
}
}
/*
===============
Menus_CloseByName
===============
*/
void Menus_CloseByName(const char *p)
{
menuDef_t *menu = Menus_FindByName(p);
if (menu != NULL)
{
Menu_RunCloseScript(menu);
menu->window.flags &= ~(WINDOW_VISIBLE | WINDOW_HASFOCUS);
}
}
/*
===============
Menu_FindItemByName
===============
*/
itemDef_t *Menu_FindItemByName(menuDef_t *menu, const char *p)
{
int i;
if (menu == NULL || p == NULL)
{
return NULL;
}
for (i = 0; i < menu->itemCount; i++)
{
if (Q_stricmp(p, menu->items[i]->window.name) == 0)
{
return menu->items[i];
}
}
return NULL;
}
/*
=================
Menu_ClearFocus
=================
*/
itemDef_t *Menu_ClearFocus(menuDef_t *menu)
{
int i;
itemDef_t *ret = NULL;
if (menu == NULL)
{
return NULL;
}
for (i = 0; i < menu->itemCount; i++)
{
if (menu->items[i]->window.flags & WINDOW_HASFOCUS)
{
ret = menu->items[i];
}
menu->items[i]->window.flags &= ~WINDOW_HASFOCUS;
if (menu->items[i]->leaveFocus)
{
Item_RunScript(menu->items[i], menu->items[i]->leaveFocus);
}
}
return ret;
}
/*
=================
Menu_TransitionItemByName
=================
*/
void Menu_TransitionItemByName(menuDef_t *menu, const char *p, rectDef_t rectFrom, rectDef_t rectTo, int time, float amt)
{
itemDef_t *item;
int i;
int count = Menu_ItemsMatchingGroup(menu, p);
for (i = 0; i < count; i++)
{
item = Menu_GetMatchingItemByNumber(menu, i, p);
if (item != NULL)
{
item->window.flags |= (WINDOW_INTRANSITION | WINDOW_VISIBLE);
item->window.offsetTime = time;
memcpy(&item->window.rectClient, &rectFrom, sizeof(rectDef_t));
memcpy(&item->window.rectEffects, &rectTo, sizeof(rectDef_t));
item->window.rectEffects2.x = abs(rectTo.x - rectFrom.x) / amt;
item->window.rectEffects2.y = abs(rectTo.y - rectFrom.y) / amt;
item->window.rectEffects2.w = abs(rectTo.w - rectFrom.w) / amt;
item->window.rectEffects2.h = abs(rectTo.h - rectFrom.h) / amt;
Item_UpdatePosition(item);
}
}
}
/*
=================
Menu_OrbitItemByName
=================
*/
void Menu_OrbitItemByName(menuDef_t *menu, const char *p, float x, float y, float cx, float cy, int time)
{
itemDef_t *item;
int i;
int count = Menu_ItemsMatchingGroup(menu, p);
for (i = 0; i < count; i++)
{
item = Menu_GetMatchingItemByNumber(menu, i, p);
if (item != NULL)
{
item->window.flags |= (WINDOW_ORBITING | WINDOW_VISIBLE);
item->window.offsetTime = time;
item->window.rectEffects.x = cx;
item->window.rectEffects.y = cy;
item->window.rectClient.x = x;
item->window.rectClient.y = y;
Item_UpdatePosition(item);
}
}
}
/*
=================
Script_FadeIn
=================
*/
void Script_FadeIn(itemDef_t *item, const char **args)
{
const char *name;
if (String_Parse(args, &name))
{
Menu_FadeItemByName((menuDef_t *) item->parent, name, qfalse);
}
}
/*
=================
Script_FadeOut
=================
*/
void Script_FadeOut(itemDef_t *item, const char **args)
{
const char *name;
if (String_Parse(args, &name))
{
Menu_FadeItemByName((menuDef_t *) item->parent, name, qtrue);
}
}
/*
=================
Script_Show
=================
*/
void Script_Show(itemDef_t *item, const char **args)
{
const char *name;
if (String_Parse(args, &name))
{
Menu_ShowItemByName((menuDef_t *) item->parent, name, qtrue);
}
}
/*
=================
Script_ShowMenu
=================
*/
void Script_ShowMenu(itemDef_t *item, const char **args)
{
const char *name;
if (String_Parse(args, &name))
{
Menus_ShowItems(name);
}
}
/*
=================
Script_Hide
=================
*/
void Script_Hide(itemDef_t *item, const char **args)
{
const char *name;
if (String_Parse(args, &name))
{
Menu_ShowItemByName((menuDef_t *) item->parent, name, qfalse);
}
}
/*
=================
Script_SetColor
=================
*/
void Script_SetColor(itemDef_t *item, const char **args)
{
const char *name;
int i;
float f;
vec4_t *out;
// expecting type of color to set and 4 args for the color
if (String_Parse(args, &name))
{
out = NULL;
if (Q_stricmp(name, "backcolor") == 0)
{
out = &item->window.backColor;
item->window.flags |= WINDOW_BACKCOLORSET;
}
else if (Q_stricmp(name, "forecolor") == 0)
{
out = &item->window.foreColor;
item->window.flags |= WINDOW_FORECOLORSET;
}
else if (Q_stricmp(name, "bordercolor") == 0)
{
out = &item->window.borderColor;
}
if (out)
{
for (i = 0; i < 4; i++)
{
// if (!Float_Parse(args, &f))
if (COM_ParseFloat( args, &f))
{
return;
}
(*out)[i] = f;
}
}
}
}
/*
=================
Script_Open
=================
*/
void Script_Open(itemDef_t *item, const char **args)
{
const char *name;
if (String_Parse(args, &name))
{
Menus_OpenByName(name);
}
}
void Script_OpenGoToMenu(itemDef_t *item, const char **args)
{
Menus_OpenByName(GoToMenu); // Give warning
}
/*
=================
Script_Close
=================
*/
void Script_Close(itemDef_t *item, const char **args)
{
const char *name;
if (String_Parse(args, &name))
{
if (Q_stricmp(name, "all") == 0)
{
Menus_CloseAll();
}
else
{
Menus_CloseByName(name);
}
}
}
/*
=================
Script_Activate
=================
*/
void Script_Activate(itemDef_t *item, const char **args)
{
const char *name, *menu;
if (String_Parse(args, &menu))
{
if (String_Parse(args, &name))
{
Item_ActivateByName(menu,name);
}
}
}
/*
=================
Script_SetBackground
=================
*/
void Script_SetBackground(itemDef_t *item, const char **args)
{
const char *name;
// expecting name to set asset to
if (String_Parse(args, &name))
{
item->window.background = DC->registerShaderNoMip(name);
}
}
/*
=================
Script_SetAsset
=================
*/
void Script_SetAsset(itemDef_t *item, const char **args)
{
const char *name;
// expecting name to set asset to
if (String_Parse(args, &name))
{
// check for a model
if (item->type == ITEM_TYPE_MODEL)
{
}
}
}
/*
=================
Script_SetFocus
=================
*/
void Script_SetFocus(itemDef_t *item, const char **args)
{
const char *name;
itemDef_t *focusItem;
if (String_Parse(args, &name))
{
focusItem = (itemDef_s *) Menu_FindItemByName((menuDef_t *) item->parent, name);
if (focusItem && !(focusItem->window.flags & WINDOW_DECORATION) && !(focusItem->window.flags & WINDOW_HASFOCUS))
{
Menu_ClearFocus((menuDef_t *) item->parent);
focusItem->window.flags |= WINDOW_HASFOCUS;
if (focusItem->onFocus)
{
Item_RunScript(focusItem, focusItem->onFocus);
}
if (DC->Assets.itemFocusSound)
{
DC->startLocalSound( DC->Assets.itemFocusSound, CHAN_LOCAL_SOUND );
}
}
}
}
/*
=================
Script_SetItemFlag
=================
*/
void Script_SetItemFlag(itemDef_t *item, const char **args)
{
const char *itemName,*number;
if (String_Parse(args, &itemName))
{
item = (itemDef_s *) Menu_FindItemByName((menuDef_t *) item->parent, itemName);
if (String_Parse(args, &number))
{
int amount = atoi(number);
item->window.flags |= amount;
}
}
}
/*
=================
Script_SetItemColor
=================
*/
void Script_SetItemColor(itemDef_t *item, const char **args)
{
const char *itemname;
const char *name;
vec4_t color;
int i;
vec4_t *out;
// expecting type of color to set and 4 args for the color
if (String_Parse(args, &itemname) && String_Parse(args, &name))
{
itemDef_t *item2;
int j;
int count = Menu_ItemsMatchingGroup((menuDef_t *) item->parent, itemname);
// if (!Color_Parse(args, &color))
if (COM_ParseVec4(args, &color))
{
return;
}
for (j = 0; j < count; j++)
{
item2 = Menu_GetMatchingItemByNumber((menuDef_t *) item->parent, j, itemname);
if (item2 != NULL)
{
out = NULL;
if (Q_stricmp(name, "backcolor") == 0)
{
out = &item2->window.backColor;
}
else if (Q_stricmp(name, "forecolor") == 0)
{
out = &item2->window.foreColor;
item2->window.flags |= WINDOW_FORECOLORSET;
}
else if (Q_stricmp(name, "bordercolor") == 0)
{
out = &item2->window.borderColor;
}
if (out)
{
for (i = 0; i < 4; i++)
{
(*out)[i] = color[i];
}
}
}
}
}
}
/*
=================
Script_Transition
=================
*/
void Script_Transition(itemDef_t *item, const char **args)
{
const char *name;
rectDef_t rectFrom, rectTo;
int time;
float amt;
if (String_Parse(args, &name))
{
// if ( Rect_Parse(args, &rectFrom) && Rect_Parse(args, &rectTo) && Int_Parse(args, &time) && Float_Parse(args, &amt))
if ( !ParseRect(args, &rectFrom) && !ParseRect(args, &rectTo) && Int_Parse(args, &time) && !COM_ParseFloat(args, &amt))
{
Menu_TransitionItemByName((menuDef_t *) item->parent, name, rectFrom, rectTo, time, amt);
}
}
}
/*
=================
Script_SetCvar
=================
*/
void Script_SetCvar(itemDef_t *item, const char **args)
{
const char *cvar, *val;
if (String_Parse(args, &cvar) && String_Parse(args, &val))
{
DC->setCVar(cvar, val);
}
}
/*
=================
Script_Exec
=================
*/
void Script_Exec(itemDef_t *item, const char **args)
{
const char *val;
if (String_Parse(args, &val))
{
DC->executeText(EXEC_APPEND, va("%s ; ", val));
}
}
/*
=================
Script_Play
=================
*/
void Script_Play(itemDef_t *item, const char **args)
{
const char *val;
if (String_Parse(args, &val))
{
DC->startLocalSound(DC->registerSound(val, qfalse), CHAN_LOCAL_SOUND);
}
}
/*
=================
Script_playLooped
=================
*/
void Script_playLooped(itemDef_t *item, const char **args)
{
const char *val;
if (String_Parse(args, &val))
{
// FIXME BOB - is this needed?
// DC->stopBackgroundTrack();
// DC->startBackgroundTrack(val, val);
}
}
/*
=================
Script_Orbit
=================
*/
void Script_Orbit(itemDef_t *item, const char **args)
{
const char *name;
float cx, cy, x, y;
int time;
if (String_Parse(args, &name))
{
// if ( Float_Parse(args, &x) && Float_Parse(args, &y) && Float_Parse(args, &cx) && Float_Parse(args, &cy) && Int_Parse(args, &time) )
if ( !COM_ParseFloat(args, &x) && !COM_ParseFloat(args, &y) && !COM_ParseFloat(args, &cx) && !COM_ParseFloat(args, &cy) && Int_Parse(args, &time) )
{
Menu_OrbitItemByName((menuDef_t *) item->parent, name, x, y, cx, cy, time);
}
}
}
commandDef_t commandList[] =
{
{"activate", &Script_Activate}, // menu
{"close", &Script_Close}, // menu
{"exec", &Script_Exec}, // group/name
{"fadein", &Script_FadeIn}, // group/name
{"fadeout", &Script_FadeOut}, // group/name
{"hide", &Script_Hide}, // group/name
{"open", &Script_Open}, // menu
{"openGoToMenu", &Script_OpenGoToMenu}, //
{"orbit", &Script_Orbit}, // group/name
{"play", &Script_Play}, // group/name
{"playlooped", &Script_playLooped}, // group/name
{"setasset", &Script_SetAsset}, // works on this
{"setbackground", &Script_SetBackground}, // works on this
{"setcolor", &Script_SetColor}, // works on this
{"setcvar", &Script_SetCvar}, // group/name
{"setfocus", &Script_SetFocus}, // sets this background color to team color
{"setitemcolor", &Script_SetItemColor}, // group/name
{"setitemflag", &Script_SetItemFlag}, // name
{"show", &Script_Show}, // group/name
{"showMenu", &Script_ShowMenu}, // menu
{"transition", &Script_Transition}, // group/name
};
int scriptCommandCount = sizeof(commandList) / sizeof(commandDef_t);
/*
===============
Item_Init
===============
*/
void Item_Init(itemDef_t *item)
{
memset(item, 0, sizeof(itemDef_t));
item->textscale = 0.55f;
Window_Init(&item->window);
}
/*
===============
Item_Multi_Setting
===============
*/
const char *Item_Multi_Setting(itemDef_t *item)
{
char buff[1024];
float value = 0;
int i;
multiDef_t *multiPtr = (multiDef_t*)item->typeData;
if (multiPtr)
{
if (multiPtr->strDef)
{
if (item->cvar)
{
DC->getCVarString(item->cvar, buff, sizeof(buff));
}
else
{
}
}
else
{
if (item->cvar) // Was a cvar given?
{
value = DC->getCVarValue(item->cvar);
}
else
{
value = item->value;
}
}
for (i = 0; i < multiPtr->count; i++)
{
if (multiPtr->strDef)
{
if (Q_stricmp(buff, multiPtr->cvarStr[i]) == 0)
{
return multiPtr->cvarList[i];
}
}
else
{
if (multiPtr->cvarValue[i] == value)
{
return multiPtr->cvarList[i];
}
}
}
}
return "";
}
//---------------------------------------------------------------------------------------------------------
// Item Keyword Parse functions
//---------------------------------------------------------------------------------------------------------
/*
===============
ItemParse_name
name <string>
===============
*/
qboolean ItemParse_name( itemDef_t *item)
{
if (!PC_ParseStringMem((const char **)&item->window.name))
{
return qfalse;
}
return qtrue;
}
qboolean ItemParse_font( itemDef_t *item )
{
if (PC_ParseInt(&item->font))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_focusSound
name <string>
===============
*/
qboolean ItemParse_focusSound( itemDef_t *item)
{
const char *temp;
if (!PC_ParseStringMem((const char **)&temp))
{
return qfalse;
}
item->focusSound = DC->registerSound(temp, qfalse);
return qtrue;
}
/*
===============
ItemParse_text
text <string>
===============
*/
qboolean ItemParse_text( itemDef_t *item)
{
if (!PC_ParseStringMem((const char **) &item->text))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_descText
text <string>
===============
*/
qboolean ItemParse_descText( itemDef_t *item)
{
if (!PC_ParseStringMem((const char **) &item->descText))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_text
text <string>
===============
*/
qboolean ItemParse_text2( itemDef_t *item)
{
if (!PC_ParseStringMem((const char **) &item->text2))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_group
group <string>
===============
*/
qboolean ItemParse_group( itemDef_t *item)
{
if (!PC_ParseStringMem((const char **)&item->window.group))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_asset_model
asset_model <string>
===============
*/
qboolean ItemParse_asset_model( itemDef_t *item)
{
const char *temp;
modelDef_t *modelPtr;
Item_ValidateTypeData(item);
modelPtr = (modelDef_t*)item->typeData;
if (!PC_ParseStringMem((const char **)&temp))
{
return qfalse;
}
item->asset = DC->registerModel(temp);
modelPtr->angle = rand() % 360;
return qtrue;
}
/*
===============
ItemParse_asset_model
asset_shader <string>
===============
*/
qboolean ItemParse_asset_shader( itemDef_t *item)
{
const char *temp;
if (!PC_ParseStringMem((const char **)&temp))
{
return qfalse;
}
item->asset = DC->registerShaderNoMip(temp);
return qtrue;
}
/*
===============
ItemParse_asset_model
model_origin <number> <number> <number>
===============
*/
qboolean ItemParse_model_origin( itemDef_t *item)
{
modelDef_t *modelPtr;
Item_ValidateTypeData(item);
modelPtr = (modelDef_t*)item->typeData;
if (PC_ParseFloat(&modelPtr->origin[0]))
{
if (PC_ParseFloat(&modelPtr->origin[1]))
{
if (PC_ParseFloat(&modelPtr->origin[2]))
{
return qtrue;
}
}
}
return qfalse;
}
/*
===============
ItemParse_model_fovx
model_fovx <number>
===============
*/
qboolean ItemParse_model_fovx( itemDef_t *item)
{
modelDef_t *modelPtr;
Item_ValidateTypeData(item);
modelPtr = (modelDef_t*)item->typeData;
if (PC_ParseFloat(&modelPtr->fov_x))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_model_fovy
model_fovy <number>
===============
*/
qboolean ItemParse_model_fovy( itemDef_t *item)
{
modelDef_t *modelPtr;
Item_ValidateTypeData(item);
modelPtr = (modelDef_t*)item->typeData;
if (PC_ParseFloat(&modelPtr->fov_y))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_model_rotation
model_rotation <integer>
===============
*/
qboolean ItemParse_model_rotation( itemDef_t *item)
{
modelDef_t *modelPtr;
Item_ValidateTypeData(item);
modelPtr = (modelDef_t*)item->typeData;
if (PC_ParseInt(&modelPtr->rotationSpeed))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_model_angle
model_angle <integer>
===============
*/
qboolean ItemParse_model_angle( itemDef_t *item)
{
modelDef_t *modelPtr;
Item_ValidateTypeData(item);
modelPtr = (modelDef_t*)item->typeData;
if (PC_ParseInt(&modelPtr->angle))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_rect
rect <rectangle>
===============
*/
qboolean ItemParse_rect( itemDef_t *item)
{
if (!PC_ParseRect(&item->window.rectClient))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_flag
flag <integer>
===============
*/
qboolean ItemParse_flag( itemDef_t *item)
{
int i;
char *tempStr;
if (!PC_ParseStringMem((const char **) &tempStr))
{
return qfalse;
}
i=0;
while (itemFlags[i].string)
{
if (Q_stricmp(tempStr,itemFlags[i].string)==0)
{
item->window.flags |= itemFlags[i].value;
break;
}
i++;
}
if (itemFlags[i].string == NULL)
{
PC_ParseWarning(va("Unknown item flag value '%s'",tempStr));
}
return qtrue;
}
/*
===============
ItemParse_style
style <integer>
===============
*/
qboolean ItemParse_style( itemDef_t *item)
{
int i;
char *tempStr;
if (!PC_ParseStringMem((const char **) &tempStr))
{
return qfalse;
}
i=0;
while (styles[i])
{
if (Q_stricmp(tempStr,styles[i])==0)
{
item->window.style = i;
break;
}
i++;
}
if (styles[i] == NULL)
{
PC_ParseWarning(va("Unknown item style value '%s'",tempStr));
}
return qtrue;
}
/*
===============
ItemParse_decoration
decoration
===============
*/
qboolean ItemParse_decoration( itemDef_t *item )
{
item->window.flags |= WINDOW_DECORATION;
return qtrue;
}
/*
===============
ItemParse_notselectable
notselectable
===============
*/
qboolean ItemParse_notselectable( itemDef_t *item )
{
listBoxDef_t *listPtr;
Item_ValidateTypeData(item);
listPtr = (listBoxDef_t*)item->typeData;
if (item->type == ITEM_TYPE_LISTBOX && listPtr)
{
listPtr->notselectable = qtrue;
}
return qtrue;
}
/*
===============
ItemParse_wrapped
manually wrapped
===============
*/
qboolean ItemParse_wrapped( itemDef_t *item )
{
item->window.flags |= WINDOW_WRAPPED;
return qtrue;
}
/*
===============
ItemParse_autowrapped
auto wrapped
===============
*/
qboolean ItemParse_autowrapped( itemDef_t *item)
{
item->window.flags |= WINDOW_AUTOWRAPPED;
return qtrue;
}
/*
===============
ItemParse_horizontalscroll
horizontalscroll
===============
*/
qboolean ItemParse_horizontalscroll( itemDef_t *item )
{
item->window.flags |= WINDOW_HORIZONTAL;
return qtrue;
}
/*
===============
ItemParse_type
type <integer>
===============
*/
qboolean ItemParse_type( itemDef_t *item )
{
int i;
char *tempStr;
// if (PC_ParseInt(&item->type))
if (!PC_ParseStringMem((const char **) &tempStr))
{
return qfalse;
}
i=0;
while (types[i])
{
if (Q_stricmp(tempStr,types[i])==0)
{
item->type = i;
break;
}
i++;
}
if (types[i] == NULL)
{
PC_ParseWarning(va("Unknown item type value '%s'",tempStr));
}
else
{
Item_ValidateTypeData(item);
}
return qtrue;
}
/*
===============
ItemParse_elementwidth
elementwidth, used for listbox image elements
uses textalignx for storage
===============
*/
qboolean ItemParse_elementwidth( itemDef_t *item )
{
listBoxDef_t *listPtr;
Item_ValidateTypeData(item);
listPtr = (listBoxDef_t*)item->typeData;
if (PC_ParseFloat(&listPtr->elementWidth))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_elementheight
elementheight, used for listbox image elements
uses textaligny for storage
===============
*/
qboolean ItemParse_elementheight( itemDef_t *item )
{
listBoxDef_t *listPtr;
Item_ValidateTypeData(item);
listPtr = (listBoxDef_t*)item->typeData;
if (PC_ParseFloat(&listPtr->elementHeight))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_feeder
feeder <float>
===============
*/
qboolean ItemParse_feeder( itemDef_t *item )
{
if (PC_ParseFloat( &item->special))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_elementtype
elementtype, used to specify what type of elements a listbox contains
uses textstyle for storage
===============
*/
qboolean ItemParse_elementtype( itemDef_t *item )
{
listBoxDef_t *listPtr;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
listPtr = (listBoxDef_t*)item->typeData;
if (PC_ParseInt(&listPtr->elementStyle))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_columns
columns sets a number of columns and an x pos and width per..
===============
*/
qboolean ItemParse_columns( itemDef_t *item)
{
int num, i;
listBoxDef_t *listPtr;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
listPtr = (listBoxDef_t*)item->typeData;
if (!PC_ParseInt(&num))
{
if (num > MAX_LB_COLUMNS)
{
num = MAX_LB_COLUMNS;
}
listPtr->numColumns = num;
for (i = 0; i < num; i++)
{
int pos, width, maxChars;
if (!PC_ParseInt(&pos) && !PC_ParseInt(&width) && !PC_ParseInt(&maxChars))
{
listPtr->columnInfo[i].pos = pos;
listPtr->columnInfo[i].width = width;
listPtr->columnInfo[i].maxChars = maxChars;
}
else
{
return qfalse;
}
}
}
else
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_border
===============
*/
qboolean ItemParse_border( itemDef_t *item)
{
if (PC_ParseInt(&item->window.border))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_bordersize
===============
*/
qboolean ItemParse_bordersize( itemDef_t *item )
{
if (PC_ParseFloat(&item->window.borderSize))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_visible
===============
*/
qboolean ItemParse_visible( itemDef_t *item)
{
int i;
if (PC_ParseInt(&i))
{
return qfalse;
}
if (i)
{
item->window.flags |= WINDOW_VISIBLE;
}
return qtrue;
}
/*
===============
ItemParse_ownerdraw
===============
*/
qboolean ItemParse_ownerdraw( itemDef_t *item)
{
if (PC_ParseInt(&item->window.ownerDraw))
{
return qfalse;
}
item->type = ITEM_TYPE_OWNERDRAW;
return qtrue;
}
/*
===============
ItemParse_align
===============
*/
qboolean ItemParse_align( itemDef_t *item)
{
if (PC_ParseInt(&item->alignment))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_align
===============
*/
qboolean ItemParse_Appearance_slot( itemDef_t *item)
{
if (PC_ParseInt(&item->appearanceSlot))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_textalign
===============
*/
qboolean ItemParse_textalign( itemDef_t *item )
{
char *tempStr;
int i;
if (!PC_ParseStringMem((const char **) &tempStr))
{
return qfalse;
}
i=0;
while (alignment[i])
{
if (Q_stricmp(tempStr,alignment[i])==0)
{
item->textalignment = i;
break;
}
i++;
}
if (alignment[i] == NULL)
{
PC_ParseWarning(va("Unknown text alignment value '%s'",tempStr));
}
return qtrue;
}
/*
===============
ItemParse_text2alignx
===============
*/
qboolean ItemParse_text2alignx( itemDef_t *item)
{
if (PC_ParseFloat(&item->text2alignx))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_text2aligny
===============
*/
qboolean ItemParse_text2aligny( itemDef_t *item)
{
if (PC_ParseFloat(&item->text2aligny))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_textalignx
===============
*/
qboolean ItemParse_textalignx( itemDef_t *item)
{
if (PC_ParseFloat(&item->textalignx))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_textaligny
===============
*/
qboolean ItemParse_textaligny( itemDef_t *item)
{
if (PC_ParseFloat(&item->textaligny))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_textscale
===============
*/
qboolean ItemParse_textscale( itemDef_t *item )
{
if (PC_ParseFloat(&item->textscale))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_textstyle
===============
*/
qboolean ItemParse_textstyle( itemDef_t *item)
{
if (PC_ParseInt(&item->textStyle))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_backcolor
===============
*/
qboolean ItemParse_backcolor( itemDef_t *item)
{
int i;
float f;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
item->window.backColor[i] = f;
}
return qtrue;
}
/*
===============
ItemParse_forecolor
===============
*/
qboolean ItemParse_forecolor( itemDef_t *item)
{
int i;
float f;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
item->window.foreColor[i] = f;
item->window.flags |= WINDOW_FORECOLORSET;
}
return qtrue;
}
/*
===============
ItemParse_bordercolor
===============
*/
qboolean ItemParse_bordercolor( itemDef_t *item)
{
int i;
float f;
for (i = 0; i < 4; i++)
{
if (PC_ParseFloat(&f))
{
return qfalse;
}
item->window.borderColor[i] = f;
}
return qtrue;
}
/*
===============
ItemParse_outlinecolor
===============
*/
qboolean ItemParse_outlinecolor( itemDef_t *item)
{
if (PC_ParseColor(&item->window.outlineColor))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_background
===============
*/
qboolean ItemParse_background( itemDef_t *item)
{
const char *temp;
if (!PC_ParseStringMem((const char **) &temp))
{
return qfalse;
}
item->window.background = ui.R_RegisterShaderNoMip(temp);
return qtrue;
}
/*
===============
ItemParse_cinematic
===============
*/
qboolean ItemParse_cinematic( itemDef_t *item)
{
if (!PC_ParseStringMem((const char **) &item->window.cinematicName))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_doubleClick
===============
*/
qboolean ItemParse_doubleClick( itemDef_t *item)
{
listBoxDef_t *listPtr;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
listPtr = (listBoxDef_t*)item->typeData;
if (!PC_Script_Parse(&listPtr->doubleClick))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_onFocus
===============
*/
qboolean ItemParse_onFocus( itemDef_t *item)
{
if (!PC_Script_Parse(&item->onFocus))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_leaveFocus
===============
*/
qboolean ItemParse_leaveFocus( itemDef_t *item )
{
if (!PC_Script_Parse(&item->leaveFocus))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_mouseEnter
===============
*/
qboolean ItemParse_mouseEnter( itemDef_t *item)
{
if (!PC_Script_Parse(&item->mouseEnter))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_mouseExit
===============
*/
qboolean ItemParse_mouseExit( itemDef_t *item)
{
if (!PC_Script_Parse(&item->mouseExit))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_mouseEnterText
===============
*/
qboolean ItemParse_mouseEnterText( itemDef_t *item)
{
if (!PC_Script_Parse(&item->mouseEnterText))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_mouseExitText
===============
*/
qboolean ItemParse_mouseExitText( itemDef_t *item)
{
if (!PC_Script_Parse(&item->mouseExitText))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_action
===============
*/
qboolean ItemParse_action( itemDef_t *item)
{
if (!PC_Script_Parse(&item->action))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_stripedFile
===============
*/
qboolean ItemParse_stripedFile( itemDef_t *item)
{
char *tempStr;
if (!PC_ParseStringMem((const char **) &tempStr))
{
return qfalse;
}
Q_strncpyz( uiInfo.uiDC.Assets.stripedFile, tempStr, sizeof(uiInfo.uiDC.Assets.stripedFile) );
ui.SP_Register(uiInfo.uiDC.Assets.stripedFile, SP_REGISTER_REQUIRED|SP_REGISTER_MENU);
return qtrue;
}
/*
===============
ItemParse_special
===============
*/
qboolean ItemParse_special( itemDef_t *item)
{
if (PC_ParseFloat(&item->special))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_cvarTest
===============
*/
qboolean ItemParse_cvarTest( itemDef_t *item)
{
if (!PC_ParseStringMem((const char **) &item->cvarTest))
{
return qfalse;
}
return qtrue;
}
/*
===============
ItemParse_cvar
===============
*/
qboolean ItemParse_cvar( itemDef_t *item)
{
editFieldDef_t *editPtr;
Item_ValidateTypeData(item);
if (!PC_ParseStringMem(&item->cvar))
{
return qfalse;
}
if (item->typeData)
{
editPtr = (editFieldDef_t*)item->typeData;
editPtr->minVal = -1;
editPtr->maxVal = -1;
editPtr->defVal = -1;
}
return qtrue;
}
/*
===============
ItemParse_maxChars
===============
*/
qboolean ItemParse_maxChars( itemDef_t *item)
{
editFieldDef_t *editPtr;
int maxChars;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
if (PC_ParseInt(&maxChars))
{
return qfalse;
}
editPtr = (editFieldDef_t*)item->typeData;
editPtr->maxChars = maxChars;
return qtrue;
}
/*
===============
ItemParse_maxPaintChars
===============
*/
qboolean ItemParse_maxPaintChars( itemDef_t *item)
{
editFieldDef_t *editPtr;
int maxChars;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
if (PC_ParseInt(&maxChars))
{
return qfalse;
}
editPtr = (editFieldDef_t*)item->typeData;
editPtr->maxPaintChars = maxChars;
return qtrue;
}
/*
===============
ItemParse_cvarFloat
===============
*/
qboolean ItemParse_cvarFloat( itemDef_t *item)
{
editFieldDef_t *editPtr;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
editPtr = (editFieldDef_t*)item->typeData;
if (PC_ParseStringMem((const char **) &item->cvar) &&
!PC_ParseFloat(&editPtr->defVal) &&
!PC_ParseFloat(&editPtr->minVal) &&
!PC_ParseFloat(&editPtr->maxVal))
{
return qtrue;
}
return qfalse;
}
/*
===============
ItemParse_cvarStrList
===============
*/
qboolean ItemParse_cvarStrList( itemDef_t *item)
{
const char *token;
multiDef_t *multiPtr;
int pass;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
multiPtr = (multiDef_t*)item->typeData;
multiPtr->count = 0;
multiPtr->strDef = qtrue;
if (!PC_ParseStringMem(&token))
{
return qfalse;
}
if (*token != '{')
{
return qfalse;
}
pass = 0;
while ( 1 )
{
if (!PC_ParseStringMem(&token))
{
PC_ParseWarning("end of file inside menu item\n");
return qfalse;
}
if ((int)token > 0) //a normal StringAlloc ptr
{
if (*token == '}')
{
return qtrue;
}
if (*token == ',' || *token == ';')
{
continue;
}
}
if (pass == 0)
{
multiPtr->cvarList[multiPtr->count] = token;
pass = 1;
}
else
{
multiPtr->cvarStr[multiPtr->count] = token;
pass = 0;
multiPtr->count++;
if (multiPtr->count >= MAX_MULTI_CVARS)
{
return qfalse;
}
}
}
return qfalse;
}
/*
===============
ItemParse_cvarFloatList
===============
*/
qboolean ItemParse_cvarFloatList( itemDef_t *item)
{
const char *token;
multiDef_t *multiPtr;
Item_ValidateTypeData(item);
if (!item->typeData)
{
return qfalse;
}
multiPtr = (multiDef_t*)item->typeData;
multiPtr->count = 0;
multiPtr->strDef = qfalse;
if (!PC_ParseStringMem(&token))
{
return qfalse;
}
if (*token != '{')
{
return qfalse;
}
while ( 1 )
{
if (!PC_ParseStringMem(&token))
{
PC_ParseWarning("end of file inside menu item\n");
return qfalse;
}
if ((int)token > 0) //a normal StringAlloc ptr
{
if (*token == '}')
{
return qtrue;
}
if (*token == ',' || *token == ';')
{
continue;
}
}
multiPtr->cvarList[multiPtr->count] = token; //either a striped ID, or a StringAlloc ptr
if (PC_ParseFloat(&multiPtr->cvarValue[multiPtr->count]))
{
return qfalse;
}
multiPtr->count++;
if (multiPtr->count >= MAX_MULTI_CVARS)
{
return qfalse;
}
}
return qfalse;
}
/*
===============
ItemParse_addColorRange
===============
*/
qboolean ItemParse_addColorRange( itemDef_t *item)
{
colorRangeDef_t color;
if (PC_ParseFloat(&color.low) &&
PC_ParseFloat(&color.high) &&
PC_ParseColor(&color.color) )
{
if (item->numColors < MAX_COLOR_RANGES)
{
memcpy(&item->colorRanges[item->numColors], &color, sizeof(color));
item->numColors++;
}
return qtrue;
}
return qfalse;
}
/*
===============
ItemParse_ownerdrawFlag
===============
*/
qboolean ItemParse_ownerdrawFlag( itemDef_t *item )
{
int i;
if (PC_ParseInt(&i))
{
return qfalse;
}
item->window.ownerDrawFlags |= i;
return qtrue;
}
/*
===============
ItemParse_enableCvar
===============
*/
qboolean ItemParse_enableCvar( itemDef_t *item)
{
if (PC_Script_Parse(&item->enableCvar))
{
item->cvarFlags = CVAR_ENABLE;
return qtrue;
}
return qfalse;
}
/*
===============
ItemParse_disableCvar
===============
*/
qboolean ItemParse_disableCvar( itemDef_t *item )
{
if (PC_Script_Parse(&item->enableCvar))
{
item->cvarFlags = CVAR_DISABLE;
return qtrue;
}
return qfalse;
}
/*
===============
ItemParse_showCvar
===============
*/
qboolean ItemParse_showCvar( itemDef_t *item )
{
if (PC_Script_Parse(&item->enableCvar))
{
item->cvarFlags = CVAR_SHOW;
return qtrue;
}
return qfalse;
}
/*
===============
ItemParse_hideCvar
===============
*/
qboolean ItemParse_hideCvar( itemDef_t *item)
{
if (PC_Script_Parse(&item->enableCvar))
{
item->cvarFlags = CVAR_HIDE;
return qtrue;
}
return qfalse;
}
/*
===============
Item_ValidateTypeData
===============
*/
void Item_ValidateTypeData(itemDef_t *item)
{
if (item->typeData)
{
return;
}
if (item->type == ITEM_TYPE_LISTBOX)
{
item->typeData = UI_Alloc(sizeof(listBoxDef_t));
memset(item->typeData, 0, sizeof(listBoxDef_t));
}
else if (item->type == ITEM_TYPE_EDITFIELD || item->type == ITEM_TYPE_NUMERICFIELD || item->type == ITEM_TYPE_YESNO || item->type == ITEM_TYPE_BIND || item->type == ITEM_TYPE_SLIDER || item->type == ITEM_TYPE_TEXT)
{
item->typeData = UI_Alloc(sizeof(editFieldDef_t));
memset(item->typeData, 0, sizeof(editFieldDef_t));
if (item->type == ITEM_TYPE_EDITFIELD)
{
if (!((editFieldDef_t *) item->typeData)->maxPaintChars)
{
((editFieldDef_t *) item->typeData)->maxPaintChars = MAX_EDITFIELD;
}
}
}
else if (item->type == ITEM_TYPE_MULTI)
{
item->typeData = UI_Alloc(sizeof(multiDef_t));
}
else if (item->type == ITEM_TYPE_MODEL)
{
item->typeData = UI_Alloc(sizeof(modelDef_t));
}
}
keywordHash_t itemParseKeywords[] = {
{"action", ItemParse_action, },
{"addColorRange", ItemParse_addColorRange, },
{"align", ItemParse_align, },
{"appearance_slot", ItemParse_Appearance_slot, },
{"asset_model", ItemParse_asset_model, },
{"asset_shader", ItemParse_asset_shader, },
{"autowrapped", ItemParse_autowrapped, },
{"backcolor", ItemParse_backcolor, },
{"background", ItemParse_background, },
{"border", ItemParse_border, },
{"bordercolor", ItemParse_bordercolor, },
{"bordersize", ItemParse_bordersize, },
{"cinematic", ItemParse_cinematic, },
{"columns", ItemParse_columns, },
{"cvar", ItemParse_cvar, },
{"cvarFloat", ItemParse_cvarFloat, },
{"cvarFloatList", ItemParse_cvarFloatList, },
{"cvarStrList", ItemParse_cvarStrList, },
{"cvarTest", ItemParse_cvarTest, },
{"decoration", ItemParse_decoration, },
{"desctext", ItemParse_descText },
{"disableCvar", ItemParse_disableCvar, },
{"doubleclick", ItemParse_doubleClick, },
{"elementheight", ItemParse_elementheight, },
{"elementtype", ItemParse_elementtype, },
{"elementwidth", ItemParse_elementwidth, },
{"enableCvar", ItemParse_enableCvar, },
{"feeder", ItemParse_feeder, },
{"flag", ItemParse_flag, },
{"focusSound", ItemParse_focusSound, },
{"font", ItemParse_font, },
{"forecolor", ItemParse_forecolor, },
{"group", ItemParse_group, },
{"hideCvar", ItemParse_hideCvar, },
{"horizontalscroll",ItemParse_horizontalscroll, },
{"leaveFocus", ItemParse_leaveFocus, },
{"maxChars", ItemParse_maxChars, },
{"maxPaintChars", ItemParse_maxPaintChars, },
{"model_angle", ItemParse_model_angle, },
{"model_fovx", ItemParse_model_fovx, },
{"model_fovy", ItemParse_model_fovy, },
{"model_origin", ItemParse_model_origin, },
{"model_rotation", ItemParse_model_rotation, },
{"mouseEnter", ItemParse_mouseEnter, },
{"mouseEnterText", ItemParse_mouseEnterText, },
{"mouseExit", ItemParse_mouseExit, },
{"mouseExitText", ItemParse_mouseExitText, },
{"name", ItemParse_name },
{"notselectable", ItemParse_notselectable, },
{"onFocus", ItemParse_onFocus, },
{"outlinecolor", ItemParse_outlinecolor, },
{"ownerdraw", ItemParse_ownerdraw, },
{"ownerdrawFlag", ItemParse_ownerdrawFlag, },
{"rect", ItemParse_rect, },
{"showCvar", ItemParse_showCvar, },
{"special", ItemParse_special, },
{"stripedFile", ItemParse_stripedFile, },
{"style", ItemParse_style, },
{"text", ItemParse_text },
{"text2", ItemParse_text2 },
{"text2alignx", ItemParse_text2alignx, },
{"text2aligny", ItemParse_text2aligny, },
{"textalign", ItemParse_textalign, },
{"textalignx", ItemParse_textalignx, },
{"textaligny", ItemParse_textaligny, },
{"textscale", ItemParse_textscale, },
{"textstyle", ItemParse_textstyle, },
{"type", ItemParse_type, },
{"visible", ItemParse_visible, },
{"wrapped", ItemParse_wrapped, },
{NULL, NULL, }
};
keywordHash_t *itemParseKeywordHash[KEYWORDHASH_SIZE];
/*
===============
Item_SetupKeywordHash
===============
*/
void Item_SetupKeywordHash(void)
{
int i;
memset(itemParseKeywordHash, 0, sizeof(itemParseKeywordHash));
for (i = 0; itemParseKeywords[i].keyword; i++)
{
KeywordHash_Add(itemParseKeywordHash, &itemParseKeywords[i]);
}
}
/*
===============
Item_Parse
===============
*/
qboolean Item_Parse(itemDef_t *item)
{
keywordHash_t *key;
char *token;
if (!PC_ParseStringMem((const char **) &token))
{
return qfalse;
}
if (*token != '{')
{
return qfalse;
}
while ( 1 )
{
if (!PC_ParseStringMem((const char **) &token))
{
PC_ParseWarning("End of file inside menu item");
return qfalse;
}
if (*token == '}')
{
/* if (!item->window.name)
{
item->window.name = defaultString;
Com_Printf(S_COLOR_YELLOW"WARNING: Menu item has no name\n");
}
if (!item->window.group)
{
item->window.group = defaultString;
Com_Printf(S_COLOR_YELLOW"WARNING: Menu item has no group\n");
}
*/
return qtrue;
}
key = (keywordHash_s *) KeywordHash_Find(itemParseKeywordHash, token);
if (!key)
{
PC_ParseWarning(va("Unknown item keyword '%s'", token));
continue;
}
if ( !key->func(item) )
{
PC_ParseWarning(va("Couldn't parse item keyword '%s'", token));
return qfalse;
}
}
}
/*
===============
Item_InitControls
init's special control types
===============
*/
void Item_InitControls(itemDef_t *item)
{
if (item == NULL)
{
return;
}
if (item->type == ITEM_TYPE_LISTBOX)
{
listBoxDef_t *listPtr = (listBoxDef_t*)item->typeData;
item->cursorPos = 0;
if (listPtr)
{
listPtr->cursorPos = 0;
listPtr->startPos = 0;
listPtr->endPos = 0;
listPtr->cursorPos = 0;
}
}
}
/*
=================
Int_Parse
=================
*/
qboolean Int_Parse(const char **p, int *i)
{
char *token;
token = COM_ParseExt(p, qfalse);
if (token && token[0] != 0)
{
*i = atoi(token);
return qtrue;
}
else
{
return qfalse;
}
}
/*
=================
String_Parse
=================
*/
qboolean String_Parse(const char **p, const char **out)
{
char *token;
token = COM_ParseExt(p, qfalse);
if (token && token[0] != 0)
{
*(out) = String_Alloc(token);
return qtrue;
}
return qfalse;
}
/*
===============
Item_RunScript
===============
*/
void Item_RunScript(itemDef_t *item, const char *s)
{
char script[1024];
const char *p;
int i;
qboolean bRan;
memset(script, 0, sizeof(script));
if (item && s && s[0])
{
Q_strcat(script, 1024, s);
p = script;
while (1)
{
const char *command;
// expect command then arguments, ; ends command, NULL ends script
if (!String_Parse(&p, &command))
{
return;
}
if (command[0] == ';' && command[1] == '\0')
{
continue;
}
bRan = qfalse;
for (i = 0; i < scriptCommandCount; i++)
{
if (Q_stricmp(command, commandList[i].name) == 0)
{
(commandList[i].handler(item, &p));
bRan = qtrue;
break;
}
}
// not in our auto list, pass to handler
if (!bRan)
{
DC->runScript(&p);
}
}
}
}
/*
===============
Menu_SetupKeywordHash
===============
*/
void Menu_SetupKeywordHash(void)
{
int i;
memset(menuParseKeywordHash, 0, sizeof(menuParseKeywordHash));
for (i = 0; menuParseKeywords[i].keyword; i++)
{
KeywordHash_Add(menuParseKeywordHash, &menuParseKeywords[i]);
}
}
/*
===============
Menus_ActivateByName
===============
*/
menuDef_t *Menus_ActivateByName(const char *p)
{
int i;
menuDef_t *m = NULL;
menuDef_t *focus = Menu_GetFocused();
for (i = 0; i < menuCount; i++)
{
// Look for the name in the current list of windows
if (Q_stricmp(Menus[i].window.name, p) == 0)
{
m = &Menus[i];
Menus_Activate(m);
if (openMenuCount < MAX_OPEN_MENUS && focus != NULL)
{
menuStack[openMenuCount++] = focus;
}
}
else
{
Menus[i].window.flags &= ~WINDOW_HASFOCUS;
}
}
if (!m)
{
Com_Printf(S_COLOR_YELLOW"WARNING: Menus_ActivateByName: Unable to find menu '%s'\n",p);
}
return m;
}
/*
===============
Menus_Activate
===============
*/
void Menus_Activate(menuDef_t *menu)
{
menu->window.flags |= (WINDOW_HASFOCUS | WINDOW_VISIBLE);
if (menu->onOpen)
{
itemDef_t item;
item.parent = menu;
Item_RunScript(&item, menu->onOpen);
}
// menu->appearanceTime = DC->realTime + 1000;
menu->appearanceTime = 0;
menu->appearanceCnt = 0;
}
typedef struct {
char *command;
int id;
int defaultbind1;
int defaultbind2;
int bind1;
int bind2;
} bind_t;
static bind_t g_bindings[] =
{
{"+scores", K_TAB, -1, -1, -1},
{"+button1", K_F1, -1, -1, -1},
{"invuse", K_ENTER, -1, -1, -1},
{"force_throw", K_F1, -1, -1, -1},
{"force_pull", K_F2, -1, -1, -1},
{"force_speed", K_F5, -1, -1, -1},
{"force_heal", K_F3, -1, -1, -1},
{"+force_grip", K_F5, -1, -1, -1},
{"force_distract", K_F4, -1, -1, -1},
{"+force_lightning",K_F6, -1, -1, -1},
{"use_bacta", -1, -1, -1, -1},
{"use_electrobinoculars",-1, -1, -1, -1},
{"use_inquisitor", -1, -1, -1, -1},
{"use_lightamp_goggles",-1, -1, -1, -1},
{"use_sentry", -1, -1, -1, -1},
{"invnext", -1, -1, -1, -1},
{"invprev", -1, -1, -1, -1},
{"invuse", -1, -1, -1, -1},
{"forcenext", K_F6, -1, -1, -1},
{"forceprev", K_F6, -1, -1, -1},
{"+speed", K_SHIFT, -1, -1, -1},
{"+forward", K_UPARROW, -1, -1, -1},
{"+back", K_DOWNARROW, -1, -1, -1},
{"+moveleft", ',', -1, -1, -1},
{"+moveright", '.', -1, -1, -1},
{"+moveup", K_SPACE, -1, -1, -1},
{"+movedown", 'c', -1, -1, -1},
{"+left", K_LEFTARROW, -1, -1, -1},
{"+right", K_RIGHTARROW, -1, -1, -1},
{"+strafe", K_ALT, -1, -1, -1},
{"+lookup", K_PGDN, -1, -1, -1},
{"+lookdown", K_DEL, -1, -1, -1},
{"+mlook", '/', -1, -1, -1},
{"centerview", K_END, -1, -1, -1},
{"zoom", -1, -1, -1, -1},
{"weapon 1", '1', -1, -1, -1},
{"weapon 2", '2', -1, -1, -1},
{"weapon 3", '3', -1, -1, -1},
{"weapon 4", '4', -1, -1, -1},
{"weapon 5", '5', -1, -1, -1},
{"weapon 6", '6', -1, -1, -1},
{"weapon 7", '7', -1, -1, -1},
{"weapon 8", '8', -1, -1, -1},
{"weapon 9", '9', -1, -1, -1},
{"weapon 10", '0', -1, -1, -1},
{"weapon 11", -1, -1, -1, -1},
{"weapon 12", -1, -1, -1, -1},
{"weapon 13", -1, -1, -1, -1},
{"+attack", K_CTRL, -1, -1, -1},
{"+altattack", -1, -1, -1, -1},
{"weapprev", '[', -1, -1, -1},
{"weapnext", ']', -1, -1, -1},
{"+button3", K_MOUSE3, -1, -1, -1},
{"prevTeamMember", 'w', -1, -1, -1},
{"nextTeamMember", 'r', -1, -1, -1},
{"nextOrder", 't', -1, -1, -1},
{"confirmOrder", 'y', -1, -1, -1},
{"denyOrder", 'n', -1, -1, -1},
{"taskOffense", 'o', -1, -1, -1},
{"taskDefense", 'd', -1, -1, -1},
{"taskPatrol", 'p', -1, -1, -1},
{"taskCamp", 'c', -1, -1, -1},
{"taskFollow", 'f', -1, -1, -1},
{"taskRetrieve", 'v', -1, -1, -1},
{"taskEscort", 'e', -1, -1, -1},
{"taskOwnFlag", 'i', -1, -1, -1},
{"taskSuicide", 'k', -1, -1, -1},
{"tauntKillInsult", K_F1, -1, -1, -1},
{"tauntPraise", K_F2, -1, -1, -1},
{"tauntTaunt", K_F3, -1, -1, -1},
{"tauntDeathInsult", K_F4, -1, -1, -1},
{"tauntGauntlet", K_F5, -1, -1, -1},
{"scoresUp", K_KP_PGUP, -1, -1, -1},
{"scoresDown", K_KP_PGDN, -1, -1, -1},
{"messagemode", '-1', -1, -1, -1},
{"messagemode2", -1, -1, -1, -1},
{"messagemode3", -1, -1, -1, -1},
{"messagemode4", -1, -1, -1, -1},
{"+use", -1, -1, -1, -1},
{"+useforce", -1, -1, -1, -1},
{"datapad", -1, -1, -1, -1},
{"save quik*", -1, -1, -1, -1},
{"thirdperson", -1, -1, -1, -1},
{"exitview", -1, -1, -1, -1}
};
static const int g_bindCount = sizeof(g_bindings) / sizeof(bind_t);
/*
=================
Controls_GetKeyAssignment
=================
*/
static void Controls_GetKeyAssignment (char *command, int *twokeys)
{
int count;
int j;
char b[256];
twokeys[0] = twokeys[1] = -1;
count = 0;
for ( j = 0; j < 256; j++ )
{
DC->getBindingBuf( j, b, 256 );
if ( *b == 0 )
{
continue;
}
if ( !Q_stricmp( b, command ) )
{
twokeys[count] = j;
count++;
if (count == 2)
{
break;
}
}
}
}
/*
=================
Controls_GetConfig
=================
*/
void Controls_GetConfig( void )
{
int i;
int twokeys[2];
// iterate each command, get its numeric binding
for (i=0; i < g_bindCount; i++)
{
Controls_GetKeyAssignment(g_bindings[i].command, twokeys);
g_bindings[i].bind1 = twokeys[0];
g_bindings[i].bind2 = twokeys[1];
}
}
/*
===============
Item_SetScreenCoords
===============
*/
void Item_SetScreenCoords(itemDef_t *item, float x, float y)
{
if (item == NULL)
{
return;
}
if (item->window.border != 0)
{
x += item->window.borderSize;
y += item->window.borderSize;
}
item->window.rect.x = x + item->window.rectClient.x;
item->window.rect.y = y + item->window.rectClient.y;
item->window.rect.w = item->window.rectClient.w;
item->window.rect.h = item->window.rectClient.h;
// force the text rects to recompute
item->textRect.w = 0;
item->textRect.h = 0;
}
/*
===============
Menu_Reset
===============
*/
void Menu_Reset(void)
{
menuCount = 0;
}
/*
===============
Menu_UpdatePosition
===============
*/
void Menu_UpdatePosition(menuDef_t *menu)
{
int i;
float x, y;
if (menu == NULL)
{
return;
}
x = menu->window.rect.x;
y = menu->window.rect.y;
if (menu->window.border != 0)
{
x += menu->window.borderSize;
y += menu->window.borderSize;
}
for (i = 0; i < menu->itemCount; i++)
{
Item_SetScreenCoords(menu->items[i], x, y);
}
}
/*
===============
Menu_PostParse
===============
*/
void Menu_PostParse(menuDef_t *menu)
{
if (menu == NULL)
{
return;
}
if (menu->fullScreen)
{
menu->window.rect.x = 0;
menu->window.rect.y = 0;
menu->window.rect.w = 640;
menu->window.rect.h = 480;
}
Menu_UpdatePosition(menu);
}
/*
===============
Menu_Init
===============
*/
void Menu_Init(menuDef_t *menu)
{
memset(menu, 0, sizeof(menuDef_t));
menu->cursorItem = -1;
UI_Cursor_Show(qtrue);
if (DC)
{
menu->fadeAmount = DC->Assets.fadeAmount;
menu->fadeClamp = DC->Assets.fadeClamp;
menu->fadeCycle = DC->Assets.fadeCycle;
}
Window_Init(&menu->window);
}
/*
===============
Menu_Parse
===============
*/
qboolean Menu_Parse(char *buffer, menuDef_t *menu)
{
// pc_token_t token;
keywordHash_t *key;
char *token2;
token2 = PC_ParseExt();
if (!token2)
{
return qfalse;
}
if (*token2 != '{')
{
PC_ParseWarning("Misplaced {");
return qfalse;
}
while ( 1 )
{
token2 = PC_ParseExt();
if (!token2)
{
PC_ParseWarning("End of file inside menu.");
return qfalse;
}
if (*token2 == '}')
{
return qtrue;
}
key = KeywordHash_Find(menuParseKeywordHash, token2);
if (!key)
{
PC_ParseWarning(va("Unknown menu keyword %s",token2));
continue;
}
if ( !key->func((itemDef_t*)menu) )
{
PC_ParseWarning(va("Couldn't parse menu keyword %s as %s",token2, key->keyword));
return qfalse;
}
}
}
/*
===============
Menu_New
===============
*/
void Menu_New(char *buffer)
{
menuDef_t *menu = &Menus[menuCount];
if (menuCount < MAX_MENUS)
{
Menu_Init(menu);
if (Menu_Parse(buffer, menu))
{
Menu_PostParse(menu);
menuCount++;
}
}
}
/*
===============
Menus_CloseAll
===============
*/
void Menus_CloseAll(void)
{
int i;
for (i = 0; i < menuCount; i++)
{
Menu_RunCloseScript(&Menus[i]);
Menus[i].window.flags &= ~(WINDOW_HASFOCUS | WINDOW_VISIBLE);
}
}
/*
===============
PC_StartParseSession
===============
*/
int PC_StartParseSession(const char *fileName,char **buffer)
{
int len;
// Try to open file and read it in.
len = ui.FS_ReadFile( fileName,(void **) buffer );
// Not there?
if ( len>0 )
{
strncpy(parseData.fileName, fileName, MAX_QPATH);
parseData.bufferStart = *buffer;
parseData.bufferCurrent = *buffer;
COM_BeginParseSession();
}
return len;
}
/*
===============
PC_EndParseSession
===============
*/
void PC_EndParseSession(char *buffer)
{
ui.FS_FreeFile( buffer ); //let go of the buffer
}
/*
===============
PC_ParseWarning
===============
*/
void PC_ParseWarning(const char *message)
{
ui.Printf(S_COLOR_YELLOW "WARNING: %s Line #%d of File '%s'\n", message,parseData.com_lines,parseData.fileName);
}
char *PC_ParseExt(void)
{
return (COM_ParseExt(&parseData.bufferCurrent, qtrue));
}
qboolean PC_ParseString(const char **string)
{
int hold;
hold = COM_ParseString(&parseData.bufferCurrent,string);
while (hold==0 && **string == 0)
{
hold = COM_ParseString(&parseData.bufferCurrent,string);
}
return(hold);
}
qboolean PC_ParseInt(int *number)
{
return(COM_ParseInt(&parseData.bufferCurrent,number));
}
qboolean PC_ParseFloat(float *number)
{
return(COM_ParseFloat(&parseData.bufferCurrent,number));
}
qboolean PC_ParseColor(vec4_t *color)
{
return(COM_ParseVec4(&parseData.bufferCurrent, color));
}
/*
=================
Controls_GetConfig2
=================
*/
void Controls_GetConfig2( void )
{
int i;
int twokeys[2];
// iterate each command, get its numeric binding
for (i=0; i < g_bindCount; i++)
{
Controls_GetKeyAssignment(g_bindings[i].command, twokeys);
g_bindings[i].bind1 = twokeys[0];
g_bindings[i].bind2 = twokeys[1];
}
//s_controls.invertmouse.curvalue = DC->getCVarValue( "m_pitch" ) < 0;
//s_controls.smoothmouse.curvalue = UI_ClampCvar( 0, 1, Controls_GetCvarValue( "m_filter" ) );
//s_controls.alwaysrun.curvalue = UI_ClampCvar( 0, 1, Controls_GetCvarValue( "cl_run" ) );
//s_controls.autoswitch.curvalue = UI_ClampCvar( 0, 1, Controls_GetCvarValue( "cg_autoswitch" ) );
//s_controls.sensitivity.curvalue = UI_ClampCvar( 2, 30, Controls_GetCvarValue( "sensitivity" ) );
//s_controls.joyenable.curvalue = UI_ClampCvar( 0, 1, Controls_GetCvarValue( "in_joystick" ) );
//s_controls.joythreshold.curvalue = UI_ClampCvar( 0.05, 0.75, Controls_GetCvarValue( "joy_threshold" ) );
//s_controls.freelook.curvalue = UI_ClampCvar( 0, 1, Controls_GetCvarValue( "cl_freelook" ) );
}
/*
=================
Controls_SetConfig2
=================
*/
void Controls_SetConfig2(qboolean restart)
{
// int i;
// FIXME BOB - make this work
// iterate each command, get its numeric binding
/*
for (i=0; i < g_bindCount; i++)
{
if (g_bindingsNew[i].bind1 != -1)
{
DC->setBinding( g_bindingsNew[i].bind1, g_bindingsNew[i].command );
if (g_bindings[i].bind2 != -1)
DC->setBinding( g_bindingsNew[i].bind2, g_bindingsNew[i].command );
}
}
DC->executeText(EXEC_APPEND, "in_restart\n");
*/
}
/*
=================
Menu_Count
=================
*/
int Menu_Count(void)
{
return menuCount;
}
/*
=================
Menu_PaintAll
=================
*/
void Menu_PaintAll(void)
{
int i;
if (captureFunc)
{
captureFunc(captureData);
}
for (i = 0; i < Menu_Count(); i++)
{
Menu_Paint(&Menus[i], qfalse);
}
if (uis.debugMode)
{
vec4_t v = {1, 1, 1, 1};
DC->drawText(5, 25, .75, v, va("(%d,%d)",DC->cursorx,DC->cursory), 0, 0, DC->Assets.qhMediumFont);
DC->drawText(5, 10, .75, v, va("fps: %f", DC->FPS), 0, 0, DC->Assets.qhMediumFont);
}
}
/*
=================
Menu_Paint
=================
*/
void Menu_Paint(menuDef_t *menu, qboolean forcePaint)
{
int i;
if (menu == NULL)
{
return;
}
if (!(menu->window.flags & WINDOW_VISIBLE) && !forcePaint)
{
return;
}
// if (menu->window.ownerDrawFlags && DC->ownerDrawVisible && !DC->ownerDrawVisible(menu->window.ownerDrawFlags))
// {
// return;
// }
if (forcePaint)
{
menu->window.flags |= WINDOW_FORCED;
}
// draw the background if necessary
if (menu->fullScreen)
{
vec3_t color;
color[0] = menu->window.backColor[0];
color[1] = menu->window.backColor[1];
color[2] = menu->window.backColor[2];
ui.R_SetColor( color);
if (menu->window.background==0) // No background shader given? Make it blank
{
menu->window.background = uis.whiteShader;
}
DC->drawHandlePic( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, menu->window.background );
}
else if (menu->window.background)
{
// this allows a background shader without being full screen
//UI_DrawHandlePic(menu->window.rect.x, menu->window.rect.y, menu->window.rect.w, menu->window.rect.h, menu->backgroundShader);
}
// paint the background and or border
Window_Paint(&menu->window, menu->fadeAmount, menu->fadeClamp, menu->fadeCycle );
// Loop through all items for the menu and paint them
for (i = 0; i < menu->itemCount; i++)
{
if (!menu->items[i]->appearanceSlot)
{
Item_Paint(menu->items[i]);
}
else // Timed order of appearance
{
if (menu->appearanceTime < DC->realTime) // Time to show another item
{
menu->appearanceTime = DC->realTime + menu->appearanceIncrement;
menu->appearanceCnt++;
}
if (menu->items[i]->appearanceSlot<=menu->appearanceCnt)
{
Item_Paint(menu->items[i]);
}
}
}
if (uis.debugMode)
{
vec4_t color;
color[0] = color[2] = color[3] = 1;
color[1] = 0;
DC->drawRect(menu->window.rect.x, menu->window.rect.y, menu->window.rect.w, menu->window.rect.h, 1, color);
}
}
/*
=================
Item_EnableShowViaCvar
=================
*/
qboolean Item_EnableShowViaCvar(itemDef_t *item, int flag)
{
char script[1024];
const char *p;
memset(script, 0, sizeof(script));
if (item && item->enableCvar && *item->enableCvar && item->cvarTest && *item->cvarTest)
{
char buff[1024];
DC->getCVarString(item->cvarTest, buff, sizeof(buff));
Q_strcat(script, 1024, item->enableCvar);
p = script;
while (1)
{
const char *val;
// expect value then ; or NULL, NULL ends list
if (!String_Parse(&p, &val))
{
return (item->cvarFlags & flag) ? qfalse : qtrue;
}
if (val[0] == ';' && val[1] == '\0')
{
continue;
}
// enable it if any of the values are true
if (item->cvarFlags & flag)
{
if (Q_stricmp(buff, val) == 0)
{
return qtrue;
}
}
else
{
// disable it if any of the values are true
if (Q_stricmp(buff, val) == 0)
{
return qfalse;
}
}
}
return (item->cvarFlags & flag) ? qfalse : qtrue;
}
return qtrue;
}
/*
=================
Item_SetTextExtents
=================
*/
void Item_SetTextExtents(itemDef_t *item, int *width, int *height, const char *text)
{
const char *textPtr = (text) ? text : item->text;
if (textPtr == NULL )
{
return;
}
*width = item->textRect.w;
*height = item->textRect.h;
// keeps us from computing the widths and heights more than once
if (*width == 0 || (item->type == ITEM_TYPE_OWNERDRAW && item->textalignment == ITEM_ALIGN_CENTER)
|| ((int)item->text<0 && item->asset != sp_language->modificationCount ) //string package language changed
)
{
int originalWidth;
originalWidth = DC->textWidth(textPtr, item->textscale, item->font);
if (item->type == ITEM_TYPE_OWNERDRAW && (item->textalignment == ITEM_ALIGN_CENTER || item->textalignment == ITEM_ALIGN_RIGHT))
{
originalWidth += DC->ownerDrawWidth(item->window.ownerDraw, item->textscale);
}
else if (item->type == ITEM_TYPE_EDITFIELD && item->textalignment == ITEM_ALIGN_CENTER && item->cvar)
{
char buff[256];
DC->getCVarString(item->cvar, buff, 256);
originalWidth += DC->textWidth(buff, item->textscale, item->font);
}
*width = DC->textWidth(textPtr, item->textscale, item->font);
*height = DC->textHeight(textPtr, item->textscale, item->font);
item->textRect.w = *width;
item->textRect.h = *height;
item->textRect.x = item->textalignx;
item->textRect.y = item->textaligny;
if (item->textalignment == ITEM_ALIGN_RIGHT)
{
item->textRect.x = item->textalignx - originalWidth;
}
else if (item->textalignment == ITEM_ALIGN_CENTER)
{
item->textRect.x = item->textalignx - originalWidth / 2;
}
ToWindowCoords(&item->textRect.x, &item->textRect.y, &item->window);
if ((int)item->text<0 )//string package
{//mark language
item->asset = sp_language->modificationCount;
}
}
}
/*
=================
Item_TextColor
=================
*/
void Item_TextColor(itemDef_t *item, vec4_t *newColor)
{
vec4_t lowLight;
const vec4_t greyColor = { .5, .5, .5, 1};
menuDef_t *parent = (menuDef_t*)item->parent;
Fade(&item->window.flags, &item->window.foreColor[3], parent->fadeClamp, &item->window.nextTime, parent->fadeCycle, qtrue, parent->fadeAmount);
if (item->window.flags & WINDOW_HASFOCUS)
{
lowLight[0] = 0.8 * parent->focusColor[0];
lowLight[1] = 0.8 * parent->focusColor[1];
lowLight[2] = 0.8 * parent->focusColor[2];
lowLight[3] = 0.8 * parent->focusColor[3];
LerpColor(parent->focusColor,lowLight,*newColor,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
/* else if (item->textStyle == ITEM_TEXTSTYLE_BLINK && !((DC->realTime/BLINK_DIVISOR) & 1))
{
lowLight[0] = 0.8 * item->window.foreColor[0];
lowLight[1] = 0.8 * item->window.foreColor[1];
lowLight[2] = 0.8 * item->window.foreColor[2];
lowLight[3] = 0.8 * item->window.foreColor[3];
LerpColor(item->window.foreColor,lowLight,*newColor,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
*/ else
{
memcpy(newColor, &item->window.foreColor, sizeof(vec4_t));
}
// items can be enabled and disabled based on cvars
if (item->enableCvar && *item->enableCvar && item->cvarTest && *item->cvarTest)
{
if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE))
{
memcpy(newColor, &parent->disableColor, sizeof(vec4_t));
}
}
if (item->window.flags & WINDOW_INACTIVE)
{
memcpy(newColor, &greyColor, sizeof(vec4_t));
}
}
/*
=================
Item_Text_Wrapped_Paint
=================
*/
void Item_Text_Wrapped_Paint(itemDef_t *item)
{
char text[1024];
const char *p, *start, *textPtr;
char buff[1024];
int width, height;
float x, y;
vec4_t color;
// now paint the text and/or any optional images
// default to left
if (item->text == NULL)
{
if (item->cvar == NULL)
{
return;
}
else
{
DC->getCVarString(item->cvar, text, sizeof(text));
textPtr = text;
}
}
else
{
textPtr = item->text;
if ((int)textPtr < 0) //string package ID
{
textPtr = SP_GetStringText(-(int)textPtr);
}
}
if (*textPtr == '\0')
{
return;
}
Item_TextColor(item, &color);
Item_SetTextExtents(item, &width, &height, textPtr);
x = item->textRect.x;
y = item->textRect.y;
start = textPtr;
p = strchr(textPtr, '\r');
while (p && *p)
{
strncpy(buff, start, p-start+1);
buff[p-start] = '\0';
DC->drawText(x, y, item->textscale, color, buff, 0, item->textStyle, item->font);
y += height + 5;
start += p - start + 1;
p = strchr(p+1, '\r');
}
DC->drawText(x, y, item->textscale, color, start, 0, item->textStyle, item->font);
}
/*
=================
Menu_Paint
=================
*/
void Item_Text_Paint(itemDef_t *item)
{
char text[1024];
const char *textPtr;
int height, width;
vec4_t color;
if (item->window.flags & WINDOW_WRAPPED)
{
Item_Text_Wrapped_Paint(item);
return;
}
if (item->window.flags & WINDOW_AUTOWRAPPED)
{
Item_Text_AutoWrapped_Paint(item);
return;
}
if (item->text == NULL)
{
if (item->cvar == NULL)
{
return;
}
else
{
DC->getCVarString(item->cvar, text, sizeof(text));
textPtr = text;
}
}
else
{
textPtr = item->text;
if ((int)textPtr < 0)
{
textPtr = SP_GetStringText(-(int)textPtr);
}
}
// this needs to go here as it sets extents for cvar types as well
Item_SetTextExtents(item, &width, &height, textPtr);
if (*textPtr == '\0')
{
return;
}
Item_TextColor(item, &color);
DC->drawText(item->textRect.x, item->textRect.y, item->textscale, color, textPtr, 0, item->textStyle, item->font);
if (item->text2) // Is there a second line of text?
{
if ((int)item->text2 < 0)
{
textPtr = SP_GetStringText(-(int)item->text2);
}
else
{
textPtr = item->text2;
}
Item_TextColor(item, &color);
DC->drawText(item->textRect.x + item->text2alignx, item->textRect.y + item->text2aligny, item->textscale, color, textPtr, 0, item->textStyle, item->font);
}
}
/*
=================
Item_UpdatePosition
=================
*/
// FIXME: consolidate this with nearby stuff
void Item_UpdatePosition(itemDef_t *item)
{
float x, y;
menuDef_t *menu;
if (item == NULL || item->parent == NULL)
{
return;
}
menu = (menuDef_t *) item->parent;
x = menu->window.rect.x;
y = menu->window.rect.y;
if (menu->window.border != 0)
{
x += menu->window.borderSize;
y += menu->window.borderSize;
}
Item_SetScreenCoords(item, x, y);
}
/*
=================
Item_CorrectedTextRect
=================
*/
static rectDef_t *Item_CorrectedTextRect(itemDef_t *item)
{
static rectDef_t rect;
if (item)
{
if (item->type==ITEM_TYPE_BUTTON)
{
return &item->window.rect;
}
memset(&rect, 0, sizeof(rectDef_t));
rect = item->textRect;
if (rect.w)
{
rect.y -= rect.h;
}
}
return &rect;
}
/*
=================
Item_TextField_Paint
=================
*/
void Item_TextField_Paint(itemDef_t *item)
{
char buff[1024];
vec4_t newColor, lowLight;
int offset;
menuDef_t *parent = (menuDef_t*)item->parent;
editFieldDef_t *editPtr = (editFieldDef_t*)item->typeData;
Item_Text_Paint(item);
buff[0] = '\0';
if (item->cvar)
{
DC->getCVarString(item->cvar, buff, sizeof(buff));
}
parent = (menuDef_t*)item->parent;
if (item->window.flags & WINDOW_HASFOCUS)
{
lowLight[0] = 0.8 * parent->focusColor[0];
lowLight[1] = 0.8 * parent->focusColor[1];
lowLight[2] = 0.8 * parent->focusColor[2];
lowLight[3] = 0.8 * parent->focusColor[3];
LerpColor(parent->focusColor,lowLight,newColor,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
else
{
memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
}
offset = 8;//(item->text && *item->text) ? 8 : 0;
if (item->window.flags & WINDOW_HASFOCUS && g_editingField)
{
char cursor = DC->getOverstrikeMode() ? '_' : '|';
DC->drawTextWithCursor(item->textRect.x + item->textRect.w + offset, item->textRect.y, item->textscale, newColor, buff + editPtr->paintOffset, item->cursorPos - editPtr->paintOffset , cursor, /*editPtr->maxPaintChars*/ item->window.rect.w, item->textStyle, item->font);
}
else
{
DC->drawText(item->textRect.x + item->textRect.w + offset, item->textRect.y, item->textscale, newColor, buff + editPtr->paintOffset, /*editPtr->maxPaintChars*/ item->window.rect.w, item->textStyle, item->font);
}
}
/*
=================
Item_ListBox_Paint
=================
*/
void Item_ListBox_Paint(itemDef_t *item)
{
float x, y, size;
int count, i, thumb;
qhandle_t image;
qhandle_t optionalImage;
listBoxDef_t *listPtr = (listBoxDef_t*)item->typeData;
// the listbox is horizontal or vertical and has a fixed size scroll bar going either direction
// elements are enumerated from the DC and either text or image handles are acquired from the DC as well
// textscale is used to size the text, textalignx and textaligny are used to size image elements
// there is no clipping available so only the last completely visible item is painted
count = DC->feederCount(item->special);
// default is vertical if horizontal flag is not here
if (item->window.flags & WINDOW_HORIZONTAL)
{
// draw scrollbar in bottom of the window
// bar
x = item->window.rect.x + 1;
y = item->window.rect.y + item->window.rect.h - SCROLLBAR_SIZE - 1;
DC->drawHandlePic(x, y, SCROLLBAR_SIZE, SCROLLBAR_SIZE, DC->Assets.scrollBarArrowLeft);
x += SCROLLBAR_SIZE - 1;
size = item->window.rect.w - (SCROLLBAR_SIZE * 2);
DC->drawHandlePic(x, y, size+1, SCROLLBAR_SIZE, DC->Assets.scrollBar);
x += size - 1;
DC->drawHandlePic(x, y, SCROLLBAR_SIZE, SCROLLBAR_SIZE, DC->Assets.scrollBarArrowRight);
// thumb
thumb = Item_ListBox_ThumbDrawPosition(item);//Item_ListBox_ThumbPosition(item);
if (thumb > x - SCROLLBAR_SIZE - 1)
{
thumb = x - SCROLLBAR_SIZE - 1;
}
DC->drawHandlePic(thumb, y, SCROLLBAR_SIZE, SCROLLBAR_SIZE, DC->Assets.scrollBarThumb);
//
listPtr->endPos = listPtr->startPos;
size = item->window.rect.w - 2;
// items
// size contains max available space
if (listPtr->elementStyle == LISTBOX_IMAGE)
{
// fit = 0;
x = item->window.rect.x + 1;
y = item->window.rect.y + 1;
for (i = listPtr->startPos; i < count; i++)
{
// always draw at least one
// which may overdraw the box if it is too small for the element
image = DC->feederItemImage(item->special, i);
if (image)
{
DC->drawHandlePic(x+1, y+1, listPtr->elementWidth - 2, listPtr->elementHeight - 2, image);
}
if (i == item->cursorPos)
{
DC->drawRect(x, y, listPtr->elementWidth-1, listPtr->elementHeight-1, item->window.borderSize, item->window.borderColor);
}
size -= listPtr->elementWidth;
if (size < listPtr->elementWidth)
{
listPtr->drawPadding = size; //listPtr->elementWidth - size;
break;
}
x += listPtr->elementWidth;
listPtr->endPos++;
// fit++;
}
}
else
{
//
}
}
else
{
// draw scrollbar to right side of the window
x = item->window.rect.x + item->window.rect.w - SCROLLBAR_SIZE - 1;
y = item->window.rect.y + 1;
DC->drawHandlePic(x, y, SCROLLBAR_SIZE, SCROLLBAR_SIZE, DC->Assets.scrollBarArrowUp);
y += SCROLLBAR_SIZE - 1;
listPtr->endPos = listPtr->startPos;
size = item->window.rect.h - (SCROLLBAR_SIZE * 2);
DC->drawHandlePic(x, y, SCROLLBAR_SIZE, size+1, DC->Assets.scrollBar);
y += size - 1;
DC->drawHandlePic(x, y, SCROLLBAR_SIZE, SCROLLBAR_SIZE, DC->Assets.scrollBarArrowDown);
// thumb
thumb = Item_ListBox_ThumbDrawPosition(item);//Item_ListBox_ThumbPosition(item);
if (thumb > y - SCROLLBAR_SIZE - 1)
{
thumb = y - SCROLLBAR_SIZE - 1;
}
DC->drawHandlePic(x, thumb, SCROLLBAR_SIZE, SCROLLBAR_SIZE, DC->Assets.scrollBarThumb);
// adjust size for item painting
size = item->window.rect.h - 2;
if (listPtr->elementStyle == LISTBOX_IMAGE)
{
// fit = 0;
x = item->window.rect.x + 1;
y = item->window.rect.y + 1;
for (i = listPtr->startPos; i < count; i++)
{
// always draw at least one
// which may overdraw the box if it is too small for the element
image = DC->feederItemImage(item->special, i);
if (image)
{
DC->drawHandlePic(x+1, y+1, listPtr->elementWidth - 2, listPtr->elementHeight - 2, image);
}
if (i == item->cursorPos)
{
DC->drawRect(x, y, listPtr->elementWidth - 1, listPtr->elementHeight - 1, item->window.borderSize, item->window.borderColor);
}
listPtr->endPos++;
size -= listPtr->elementWidth;
if (size < listPtr->elementHeight)
{
listPtr->drawPadding = listPtr->elementHeight - size;
break;
}
y += listPtr->elementHeight;
// fit++;
}
}
else
{
x = item->window.rect.x + 1;
y = item->window.rect.y + 1 - listPtr->elementHeight;
for (i = listPtr->startPos; i < count; i++)
{
const char *text;
// always draw at least one
// which may overdraw the box if it is too small for the element
if (listPtr->numColumns > 0)
{
int j;
for (j = 0; j < listPtr->numColumns; j++)
{
text = DC->feederItemText(item->special, i, j, &optionalImage);
if (optionalImage >= 0)
{
DC->drawHandlePic(x + 4 + listPtr->columnInfo[j].pos, y - 1 + listPtr->elementHeight / 2, listPtr->columnInfo[j].width, listPtr->columnInfo[j].width, optionalImage);
}
else if (text)
{
DC->drawText(x + 4 + listPtr->columnInfo[j].pos, y + listPtr->elementHeight, item->textscale, item->window.foreColor, text, listPtr->columnInfo[j].maxChars, item->textStyle, item->font);
}
}
}
else
{
text = DC->feederItemText(item->special, i, 0, &optionalImage);
if (optionalImage >= 0)
{
//DC->drawHandlePic(x + 4 + listPtr->elementHeight, y, listPtr->columnInfo[j].width, listPtr->columnInfo[j].width, optionalImage);
}
else if (text)
{
DC->drawText(x + 4, y + listPtr->elementHeight, item->textscale, item->window.foreColor, text, 0, item->textStyle, item->font);
}
}
// The chosen text
if (i == item->cursorPos)
{
DC->fillRect(x + 2, y + listPtr->elementHeight+6, item->window.rect.w - SCROLLBAR_SIZE - 4, listPtr->elementHeight, item->window.outlineColor);
}
size -= listPtr->elementHeight;
if (size < listPtr->elementHeight)
{
listPtr->drawPadding = listPtr->elementHeight - size;
break;
}
listPtr->endPos++;
y += listPtr->elementHeight;
// fit++;
}
}
}
}
char g_nameBind1[32];
char g_nameBind2[32];
typedef struct
{
char* name;
float defaultvalue;
float value;
} configcvar_t;
/*
=================
BindingFromName
=================
*/
void BindingFromName(const char *cvar)
{
int i, b1, b2;
// iterate each command, set its default binding
for (i=0; i < g_bindCount; i++)
{
if (Q_stricmp(cvar, g_bindings[i].command) == 0) {
b1 = g_bindings[i].bind1;
if (b1 == -1)
{
break;
}
DC->keynumToStringBuf( b1, g_nameBind1, 32 );
Q_strupr(g_nameBind1);
b2 = g_bindings[i].bind2;
if (b2 != -1)
{
DC->keynumToStringBuf( b2, g_nameBind2, 32 );
Q_strupr(g_nameBind2);
strcat( g_nameBind1, va(" %s ",ui.SP_GetStringTextString(va("%s_OR",uiInfo.uiDC.Assets.stripedFile))) );
strcat( g_nameBind1, g_nameBind2 );
}
return;
}
}
strcpy(g_nameBind1, "???");
}
/*
=================
Item_Bind_Paint
=================
*/
void Item_Bind_Paint(itemDef_t *item)
{
vec4_t newColor, lowLight;
float value;
int maxChars = 0;
menuDef_t *parent = (menuDef_t*)item->parent;
editFieldDef_t *editPtr = (editFieldDef_t*)item->typeData;
if (editPtr)
{
maxChars = editPtr->maxPaintChars;
}
value = (item->cvar) ? DC->getCVarValue(item->cvar) : 0;
if (item->window.flags & WINDOW_HASFOCUS)
{
if (g_bindItem == item)
{
lowLight[0] = 0.8f * 1.0f;
lowLight[1] = 0.8f * 0.0f;
lowLight[2] = 0.8f * 0.0f;
lowLight[3] = 0.8f * 1.0f;
}
else
{
lowLight[0] = 0.8f * parent->focusColor[0];
lowLight[1] = 0.8f * parent->focusColor[1];
lowLight[2] = 0.8f * parent->focusColor[2];
lowLight[3] = 0.8f * parent->focusColor[3];
}
LerpColor(parent->focusColor,lowLight,newColor,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
else
{
Item_TextColor( item,&newColor);
}
if (item->text)
{
Item_Text_Paint(item);
BindingFromName(item->cvar);
DC->drawText(item->textRect.x + item->textRect.w + 8, item->textRect.y, item->textscale, newColor, g_nameBind1, /*maxChars*/item->textRect.w, item->textStyle, item->font);
}
else
{
DC->drawText(item->textRect.x, item->textRect.y, item->textscale, newColor, (value != 0) ? "FIXME 1" : "FIXME 0", /*maxChars*/item->textRect.w, item->textStyle, item->font);
}
}
/*
=================
Item_Model_Paint
=================
*/
void Item_Model_Paint(itemDef_t *item)
{
float x, y, w, h;
refdef_t refdef;
refEntity_t ent;
vec3_t mins, maxs, origin;
vec3_t angles;
modelDef_t *modelPtr = (modelDef_t*)item->typeData;
if (modelPtr == NULL)
{
return;
}
// setup the refdef
memset( &refdef, 0, sizeof( refdef ) );
refdef.rdflags = RDF_NOWORLDMODEL;
AxisClear( refdef.viewaxis );
x = item->window.rect.x+1;
y = item->window.rect.y+1;
w = item->window.rect.w-2;
h = item->window.rect.h-2;
refdef.x = x;
refdef.y = y;
refdef.width = w;
refdef.height = h;
DC->modelBounds( item->asset, mins, maxs );
origin[2] = -0.5 * ( mins[2] + maxs[2] );
origin[1] = 0.5 * ( mins[1] + maxs[1] );
// calculate distance so the model nearly fills the box
if (qtrue)
{
float len = 0.5 * ( maxs[2] - mins[2] );
origin[0] = len / 0.268; // len / tan( fov/2 )
//origin[0] = len / tan(w/2);
}
else
{
origin[0] = item->textscale;
}
refdef.fov_x = (modelPtr->fov_x) ? modelPtr->fov_x : w;
refdef.fov_x = (modelPtr->fov_y) ? modelPtr->fov_y : h;
//refdef.fov_x = (int)((float)refdef.width / 640.0f * 90.0f);
//xx = refdef.width / tan( refdef.fov_x / 360 * M_PI );
//refdef.fov_y = atan2( refdef.height, xx );
//refdef.fov_y *= ( 360 / M_PI );
// FIXME : Bob make this work
// DC->clearScene();
refdef.time = DC->realTime;
// add the model
memset( &ent, 0, sizeof(ent) );
//adjust = 5.0 * sin( (float)uis.realtime / 500 );
//adjust = 360 % (int)((float)uis.realtime / 1000);
//VectorSet( angles, 0, 0, 1 );
// use item storage to track
if (modelPtr->rotationSpeed)
{
if (DC->realTime > item->window.nextTime)
{
item->window.nextTime = DC->realTime + modelPtr->rotationSpeed;
modelPtr->angle = (int)(modelPtr->angle + 1) % 360;
}
}
VectorSet( angles, 0, modelPtr->angle, 0 );
AnglesToAxis( angles, ent.axis );
ent.hModel = item->asset;
VectorCopy( origin, ent.origin );
VectorCopy( origin, ent.lightingOrigin );
ent.renderfx = RF_LIGHTING_ORIGIN | RF_NOSHADOW;
VectorCopy( ent.origin, ent.oldorigin );
// DC->addRefEntityToScene( &ent );
DC->renderScene( &refdef );
}
/*
=================
Item_CorrectedTextRect
=================
*/
void Item_OwnerDraw_Paint(itemDef_t *item)
{
menuDef_t *parent;
if (item == NULL)
{
return;
}
parent = (menuDef_t*)item->parent;
if (DC->ownerDrawItem)
{
vec4_t color, lowLight;
menuDef_t *parent = (menuDef_t*)item->parent;
Fade(&item->window.flags, &item->window.foreColor[3], parent->fadeClamp, &item->window.nextTime, parent->fadeCycle, qtrue, parent->fadeAmount);
memcpy(&color, &item->window.foreColor, sizeof(color));
if (item->numColors > 0 && DC->getValue)
{
// if the value is within one of the ranges then set color to that, otherwise leave at default
int i;
float f = DC->getValue(item->window.ownerDraw);
for (i = 0; i < item->numColors; i++)
{
if (f >= item->colorRanges[i].low && f <= item->colorRanges[i].high)
{
memcpy(&color, &item->colorRanges[i].color, sizeof(color));
break;
}
}
}
if (item->window.flags & WINDOW_HASFOCUS)
{
lowLight[0] = 0.8 * parent->focusColor[0];
lowLight[1] = 0.8 * parent->focusColor[1];
lowLight[2] = 0.8 * parent->focusColor[2];
lowLight[3] = 0.8 * parent->focusColor[3];
LerpColor(parent->focusColor,lowLight,color,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
else if (item->textStyle == ITEM_TEXTSTYLE_BLINK && !((DC->realTime/BLINK_DIVISOR) & 1))
{
lowLight[0] = 0.8 * item->window.foreColor[0];
lowLight[1] = 0.8 * item->window.foreColor[1];
lowLight[2] = 0.8 * item->window.foreColor[2];
lowLight[3] = 0.8 * item->window.foreColor[3];
LerpColor(item->window.foreColor,lowLight,color,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE))
{
memcpy(color, parent->disableColor, sizeof(vec4_t));
}
if (item->text)
{
Item_Text_Paint(item);
if (item->text[0])
{
// +8 is an offset kludge to properly align owner draw items that have text combined with them
DC->ownerDrawItem(item->textRect.x + item->textRect.w + 8, item->window.rect.y, item->window.rect.w, item->window.rect.h, 0, item->textaligny, item->window.ownerDraw, item->window.ownerDrawFlags, item->alignment, item->special, item->textscale, color, item->window.background, item->textStyle, item->font );
}
else
{
DC->ownerDrawItem(item->textRect.x + item->textRect.w, item->window.rect.y, item->window.rect.w, item->window.rect.h, 0, item->textaligny, item->window.ownerDraw, item->window.ownerDrawFlags, item->alignment, item->special, item->textscale, color, item->window.background, item->textStyle, item->font );
}
}
else
{
DC->ownerDrawItem(item->window.rect.x, item->window.rect.y, item->window.rect.w, item->window.rect.h, item->textalignx, item->textaligny, item->window.ownerDraw, item->window.ownerDrawFlags, item->alignment, item->special, item->textscale, color, item->window.background, item->textStyle, item->font );
}
}
}
void Item_YesNo_Paint(itemDef_t *item)
{
vec4_t newColor, lowLight;
float value;
menuDef_t *parent = (menuDef_t*)item->parent;
value = (item->cvar) ? DC->getCVarValue(item->cvar) : 0;
if (item->window.flags & WINDOW_HASFOCUS)
{
lowLight[0] = 0.8 * parent->focusColor[0];
lowLight[1] = 0.8 * parent->focusColor[1];
lowLight[2] = 0.8 * parent->focusColor[2];
lowLight[3] = 0.8 * parent->focusColor[3];
LerpColor(parent->focusColor,lowLight,newColor,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
else
{
memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
}
if (item->text)
{
Item_Text_Paint(item);
DC->drawText(item->textRect.x + item->textRect.w + 8, item->textRect.y, item->textscale, newColor, (value != 0) ? "Yes" : "No", 0, item->textStyle, item->font);
}
else
{
DC->drawText(item->textRect.x, item->textRect.y, item->textscale, newColor, (value != 0) ? "Yes" : "No", 0, item->textStyle, item->font);
}
}
/*
=================
Item_Multi_Paint
=================
*/
void Item_Multi_Paint(itemDef_t *item)
{
vec4_t newColor, lowLight;
const char *text = "";
menuDef_t *parent = (menuDef_t*)item->parent;
if (item->window.flags & WINDOW_HASFOCUS)
{
lowLight[0] = 0.8 * parent->focusColor[0];
lowLight[1] = 0.8 * parent->focusColor[1];
lowLight[2] = 0.8 * parent->focusColor[2];
lowLight[3] = 0.8 * parent->focusColor[3];
LerpColor(parent->focusColor,lowLight,newColor,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
else
{
memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
}
text = Item_Multi_Setting(item);
if ((int)text < 0) //it's a striped ID
{
text = SP_GetStringText(-(int)text);
}
if (item->text)
{
Item_Text_Paint(item);
DC->drawText(item->textRect.x + item->textRect.w + 8, item->textRect.y, item->textscale, newColor, text, 0, item->textStyle, item->font);
}
else
{
DC->drawText(item->textRect.x, item->textRect.y, item->textscale, newColor, text, 0, item->textStyle, item->font);
}
}
/*
=================
Item_Slider_ThumbPosition
=================
*/
int Item_ListBox_ThumbDrawPosition(itemDef_t *item)
{
int min, max;
if (itemCapture == item)
{
if (item->window.flags & WINDOW_HORIZONTAL)
{
min = item->window.rect.x + SCROLLBAR_SIZE + 1;
max = item->window.rect.x + item->window.rect.w - 2*SCROLLBAR_SIZE - 1;
if (DC->cursorx >= min + SCROLLBAR_SIZE/2 && DC->cursorx <= max + SCROLLBAR_SIZE/2)
{
return DC->cursorx - SCROLLBAR_SIZE/2;
}
else
{
return Item_ListBox_ThumbPosition(item);
}
}
else
{
min = item->window.rect.y + SCROLLBAR_SIZE + 1;
max = item->window.rect.y + item->window.rect.h - 2*SCROLLBAR_SIZE - 1;
if (DC->cursory >= min + SCROLLBAR_SIZE/2 && DC->cursory <= max + SCROLLBAR_SIZE/2)
{
return DC->cursory - SCROLLBAR_SIZE/2;
}
else
{
return Item_ListBox_ThumbPosition(item);
}
}
}
else
{
return Item_ListBox_ThumbPosition(item);
}
}
/*
=================
Item_Slider_ThumbPosition
=================
*/
float Item_Slider_ThumbPosition(itemDef_t *item)
{
float value, range, x;
editFieldDef_t *editDef = (editFieldDef_t *) item->typeData;
if (item->text)
{
x = item->textRect.x + item->textRect.w + 8;
}
else
{
x = item->window.rect.x;
}
if (editDef == NULL && item->cvar)
{
return x;
}
value = DC->getCVarValue(item->cvar);
if (value < editDef->minVal)
{
value = editDef->minVal;
}
else if (value > editDef->maxVal)
{
value = editDef->maxVal;
}
range = editDef->maxVal - editDef->minVal;
value -= editDef->minVal;
value /= range;
//value /= (editDef->maxVal - editDef->minVal);
value *= SLIDER_WIDTH;
x += value;
// vm fuckage
//x = x + (((float)value / editDef->maxVal) * SLIDER_WIDTH);
return x;
}
/*
=================
Item_Slider_Paint
=================
*/
void Item_Slider_Paint(itemDef_t *item)
{
vec4_t newColor, lowLight;
float x, y, value;
menuDef_t *parent = (menuDef_t*)item->parent;
value = (item->cvar) ? DC->getCVarValue(item->cvar) : 0;
if (item->window.flags & WINDOW_HASFOCUS)
{
lowLight[0] = 0.8 * parent->focusColor[0];
lowLight[1] = 0.8 * parent->focusColor[1];
lowLight[2] = 0.8 * parent->focusColor[2];
lowLight[3] = 0.8 * parent->focusColor[3];
LerpColor(parent->focusColor,lowLight,newColor,0.5+0.5*sin(DC->realTime / PULSE_DIVISOR));
}
else
{
memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
}
y = item->window.rect.y;
if (item->text)
{
Item_Text_Paint(item);
x = item->textRect.x + item->textRect.w + 8;
}
else
{
x = item->window.rect.x;
}
DC->setColor(newColor);
DC->drawHandlePic( x, y+7, SLIDER_WIDTH, SLIDER_HEIGHT, DC->Assets.sliderBar );
x = Item_Slider_ThumbPosition(item);
// DC->drawHandlePic( x - (SLIDER_THUMB_WIDTH / 2), y - 2, SLIDER_THUMB_WIDTH, SLIDER_THUMB_HEIGHT, DC->Assets.sliderThumb );
DC->drawHandlePic( x - (SLIDER_THUMB_WIDTH / 2), y+7, SLIDER_THUMB_WIDTH, SLIDER_THUMB_HEIGHT, DC->Assets.sliderThumb );
}
/*
=================
Item_Paint
=================
*/
void Item_Paint(itemDef_t *item)
{
int xPos,textWidth;
vec4_t red;
menuDef_t *parent = (menuDef_t*)item->parent;
red[0] = red[3] = 1;
red[1] = red[2] = 0;
if (item == NULL)
{
return;
}
if (item->window.flags & WINDOW_MOUSEOVER)
{
if (item->descText)
{
const char *textPtr;
if ((int)item->descText < 0)
{
textPtr = SP_GetStringText(-(int)item->descText);
}
else
{
textPtr = item->descText;
}
textWidth = DC->textWidth(textPtr,(float) item->textscale, item->font);
if (parent->descAlignment == ITEM_ALIGN_RIGHT)
{
xPos = parent->descX - textWidth; // Right justify
}
else if (parent->descAlignment == ITEM_ALIGN_CENTER)
{
xPos = parent->descX - (textWidth/2); // Center justify
}
else // Left justify
{
xPos = parent->descX;
}
vec4_t color = {1, 1, 1, 1};
Item_TextColor(item, &color);
if (!parent->descScale)
{
parent->descScale = 1;
}
DC->drawText(xPos, parent->descY, (float) parent->descScale, parent->descColor, textPtr, 0, 0, item->font);
}
}
if (item->window.flags & WINDOW_ORBITING)
{
if (DC->realTime > item->window.nextTime)
{
float rx, ry, a, c, s, w, h;
item->window.nextTime = DC->realTime + item->window.offsetTime;
// translate
w = item->window.rectClient.w / 2;
h = item->window.rectClient.h / 2;
rx = item->window.rectClient.x + w - item->window.rectEffects.x;
ry = item->window.rectClient.y + h - item->window.rectEffects.y;
a = (float) (3 * M_PI / 180);
c = cos(a);
s = sin(a);
item->window.rectClient.x = (rx * c - ry * s) + item->window.rectEffects.x - w;
item->window.rectClient.y = (rx * s + ry * c) + item->window.rectEffects.y - h;
Item_UpdatePosition(item);
}
}
if (item->window.flags & WINDOW_INTRANSITION)
{
if (DC->realTime > item->window.nextTime)
{
int done = 0;
item->window.nextTime = DC->realTime + item->window.offsetTime;
// transition the x,y
if (item->window.rectClient.x == item->window.rectEffects.x)
{
done++;
}
else
{
if (item->window.rectClient.x < item->window.rectEffects.x)
{
item->window.rectClient.x += item->window.rectEffects2.x;
if (item->window.rectClient.x > item->window.rectEffects.x)
{
item->window.rectClient.x = item->window.rectEffects.x;
done++;
}
}
else
{
item->window.rectClient.x -= item->window.rectEffects2.x;
if (item->window.rectClient.x < item->window.rectEffects.x)
{
item->window.rectClient.x = item->window.rectEffects.x;
done++;
}
}
}
if (item->window.rectClient.y == item->window.rectEffects.y)
{
done++;
}
else
{
if (item->window.rectClient.y < item->window.rectEffects.y)
{
item->window.rectClient.y += item->window.rectEffects2.y;
if (item->window.rectClient.y > item->window.rectEffects.y)
{
item->window.rectClient.y = item->window.rectEffects.y;
done++;
}
}
else
{
item->window.rectClient.y -= item->window.rectEffects2.y;
if (item->window.rectClient.y < item->window.rectEffects.y)
{
item->window.rectClient.y = item->window.rectEffects.y;
done++;
}
}
}
if (item->window.rectClient.w == item->window.rectEffects.w)
{
done++;
}
else
{
if (item->window.rectClient.w < item->window.rectEffects.w)
{
item->window.rectClient.w += item->window.rectEffects2.w;
if (item->window.rectClient.w > item->window.rectEffects.w)
{
item->window.rectClient.w = item->window.rectEffects.w;
done++;
}
}
else
{
item->window.rectClient.w -= item->window.rectEffects2.w;
if (item->window.rectClient.w < item->window.rectEffects.w)
{
item->window.rectClient.w = item->window.rectEffects.w;
done++;
}
}
}
if (item->window.rectClient.h == item->window.rectEffects.h)
{
done++;
}
else
{
if (item->window.rectClient.h < item->window.rectEffects.h)
{
item->window.rectClient.h += item->window.rectEffects2.h;
if (item->window.rectClient.h > item->window.rectEffects.h)
{
item->window.rectClient.h = item->window.rectEffects.h;
done++;
}
}
else
{
item->window.rectClient.h -= item->window.rectEffects2.h;
if (item->window.rectClient.h < item->window.rectEffects.h)
{
item->window.rectClient.h = item->window.rectEffects.h;
done++;
}
}
}
Item_UpdatePosition(item);
if (done == 4)
{
item->window.flags &= ~WINDOW_INTRANSITION;
}
}
}
if (item->window.ownerDrawFlags && DC->ownerDrawVisible)
{
if (!DC->ownerDrawVisible(item->window.ownerDrawFlags))
{
item->window.flags &= ~WINDOW_VISIBLE;
}
else
{
item->window.flags |= WINDOW_VISIBLE;
}
}
if (item->cvarFlags & (CVAR_SHOW | CVAR_HIDE))
{
if (!Item_EnableShowViaCvar(item, CVAR_SHOW))
{
return;
}
}
if (item->window.flags & WINDOW_TIMEDVISIBLE)
{
}
if (!(item->window.flags & WINDOW_VISIBLE))
{
return;
}
// paint the rect first..
Window_Paint(&item->window, parent->fadeAmount , parent->fadeClamp, parent->fadeCycle);
if (uis.debugMode)
{
vec4_t color;
rectDef_t *r = Item_CorrectedTextRect(item);
color[1] = color[3] = 1;
color[0] = color[2] = 0;
DC->drawRect(r->x, r->y, r->w, r->h, 1, color);
}
//DC->drawRect(item->window.rect.x, item->window.rect.y, item->window.rect.w, item->window.rect.h, 1, red);
switch (item->type)
{
case ITEM_TYPE_OWNERDRAW:
Item_OwnerDraw_Paint(item);
break;
case ITEM_TYPE_TEXT:
case ITEM_TYPE_BUTTON:
Item_Text_Paint(item);
break;
case ITEM_TYPE_RADIOBUTTON:
break;
case ITEM_TYPE_CHECKBOX:
break;
case ITEM_TYPE_EDITFIELD:
case ITEM_TYPE_NUMERICFIELD:
Item_TextField_Paint(item);
break;
case ITEM_TYPE_COMBO:
break;
case ITEM_TYPE_LISTBOX:
Item_ListBox_Paint(item);
break;
case ITEM_TYPE_MODEL:
Item_Model_Paint(item);
break;
case ITEM_TYPE_YESNO:
Item_YesNo_Paint(item);
break;
case ITEM_TYPE_MULTI:
Item_Multi_Paint(item);
break;
case ITEM_TYPE_BIND:
Item_Bind_Paint(item);
break;
case ITEM_TYPE_SLIDER:
Item_Slider_Paint(item);
break;
default:
break;
}
}
/*
=================
LerpColor
=================
*/
void LerpColor(vec4_t a, vec4_t b, vec4_t c, float t)
{
int i;
// lerp and clamp each component
for (i=0; i<4; i++)
{
c[i] = a[i] + t*(b[i]-a[i]);
if (c[i] < 0)
{
c[i] = 0;
}
else if (c[i] > 1.0)
{
c[i] = 1.0;
}
}
}
/*
=================
Fade
=================
*/
void Fade(int *flags, float *f, float clamp, int *nextTime, int offsetTime, qboolean bFlags, float fadeAmount)
{
if (*flags & (WINDOW_FADINGOUT | WINDOW_FADINGIN))
{
if (DC->realTime > *nextTime)
{
*nextTime = DC->realTime + offsetTime;
if (*flags & WINDOW_FADINGOUT)
{
*f -= fadeAmount;
if (bFlags && *f <= 0.0)
{
*flags &= ~(WINDOW_FADINGOUT | WINDOW_VISIBLE);
}
}
else
{
*f += fadeAmount;
if (*f >= clamp)
{
*f = clamp;
if (bFlags)
{
*flags &= ~WINDOW_FADINGIN;
}
}
}
}
}
}
/*
=================
GradientBar_Paint
=================
*/
void GradientBar_Paint(rectDef_t *rect, vec4_t color)
{
// gradient bar takes two paints
DC->setColor( color );
DC->drawHandlePic(rect->x, rect->y, rect->w, rect->h, DC->Assets.gradientBar);
DC->setColor( NULL );
}
/*
=================
Window_Paint
=================
*/
void Window_Paint(Window *w, float fadeAmount, float fadeClamp, float fadeCycle)
{
//float bordersize = 0;
vec4_t color;
rectDef_t fillRect = w->rect;
if (uis.debugMode)
{
color[0] = color[1] = color[2] = color[3] = 1;
DC->drawRect(w->rect.x, w->rect.y, w->rect.w, w->rect.h, 1, color);
}
if (w == NULL || (w->style == 0 && w->border == 0))
{
return;
}
if (w->border != 0)
{
fillRect.x += w->borderSize;
fillRect.y += w->borderSize;
fillRect.w -= w->borderSize + 1;
fillRect.h -= w->borderSize + 1;
}
if (w->style == WINDOW_STYLE_FILLED)
{
// box, but possible a shader that needs filled
if (w->background)
{
Fade(&w->flags, &w->backColor[3], fadeClamp, &w->nextTime, fadeCycle, qtrue, fadeAmount);
DC->setColor(w->backColor);
DC->drawHandlePic(fillRect.x, fillRect.y, fillRect.w, fillRect.h, w->background);
DC->setColor(NULL);
}
else
{
DC->fillRect(fillRect.x, fillRect.y, fillRect.w, fillRect.h, w->backColor);
}
}
else if (w->style == WINDOW_STYLE_GRADIENT)
{
GradientBar_Paint(&fillRect, w->backColor);
// gradient bar
}
else if (w->style == WINDOW_STYLE_SHADER)
{
if (w->flags & WINDOW_FORECOLORSET)
{
DC->setColor(w->foreColor);
}
DC->drawHandlePic(fillRect.x, fillRect.y, fillRect.w, fillRect.h, w->background);
DC->setColor(NULL);
}
if (w->border == WINDOW_BORDER_FULL)
{
// full
// HACK HACK HACK
if (w->style == WINDOW_STYLE_TEAMCOLOR)
{
if (color[0] > 0)
{
// red
color[0] = 1;
color[1] = color[2] = .5;
}
else
{
color[2] = 1;
color[0] = color[1] = .5;
}
color[3] = 1;
DC->drawRect(w->rect.x, w->rect.y, w->rect.w, w->rect.h, w->borderSize, color);
}
else
{
DC->drawRect(w->rect.x, w->rect.y, w->rect.w, w->rect.h, w->borderSize, w->borderColor);
}
}
else if (w->border == WINDOW_BORDER_HORZ)
{
// top/bottom
DC->setColor(w->borderColor);
DC->drawTopBottom(w->rect.x, w->rect.y, w->rect.w, w->rect.h, w->borderSize);
DC->setColor( NULL );
}
else if (w->border == WINDOW_BORDER_VERT)
{
// left right
DC->setColor(w->borderColor);
DC->drawSides(w->rect.x, w->rect.y, w->rect.w, w->rect.h, w->borderSize);
DC->setColor( NULL );
}
else if (w->border == WINDOW_BORDER_KCGRADIENT)
{
// this is just two gradient bars along each horz edge
rectDef_t r = w->rect;
r.h = w->borderSize;
GradientBar_Paint(&r, w->borderColor);
r.y = w->rect.y + w->rect.h - 1;
GradientBar_Paint(&r, w->borderColor);
}
}
/*
=================
Display_KeyBindPending
=================
*/
qboolean Display_KeyBindPending(void)
{
return g_waitingForKey;
}
/*
=================
ToWindowCoords
=================
*/
void ToWindowCoords(float *x, float *y, windowDef_t *window)
{
if (window->border != 0)
{
*x += window->borderSize;
*y += window->borderSize;
}
*x += window->rect.x;
*y += window->rect.y;
}
/*
=================
Item_Text_AutoWrapped_Paint
=================
*/
void Item_Text_AutoWrapped_Paint(itemDef_t *item)
{
char text[1024];
const char *p, *textPtr, *newLinePtr;
char buff[1024];
int width, height, len, textWidth, newLine, newLineWidth;
float y;
vec4_t color;
textWidth = 0;
newLinePtr = NULL;
if (item->text == NULL)
{
if (item->cvar == NULL)
{
return;
}
else
{
DC->getCVarString(item->cvar, text, sizeof(text));
textPtr = text;
}
}
else
{
textPtr = item->text;
if ((int)textPtr < 0) //string package ID
{
textPtr = SP_GetStringText(-(int)textPtr);
}
}
if (*textPtr == '\0')
{
return;
}
Item_TextColor(item, &color);
Item_SetTextExtents(item, &width, &height, textPtr);
y = item->textaligny;
len = 0;
buff[0] = '\0';
newLine = 0;
newLineWidth = 0;
p = textPtr;
while (p)
{
if (*p == ' ' || *p == '\t' || *p == '\n' || *p == '\0')
{
newLine = len;
newLinePtr = p+1;
newLineWidth = textWidth;
}
textWidth = DC->textWidth(buff, item->textscale, 0);
if ( (newLine && textWidth > item->window.rect.w) || *p == '\n' || *p == '\0')
{
if (len)
{
if (item->textalignment == ITEM_ALIGN_LEFT)
{
item->textRect.x = item->textalignx;
}
else if (item->textalignment == ITEM_ALIGN_RIGHT)
{
item->textRect.x = item->textalignx - newLineWidth;
}
else if (item->textalignment == ITEM_ALIGN_CENTER)
{
item->textRect.x = item->textalignx - newLineWidth / 2;
}
item->textRect.y = y;
ToWindowCoords(&item->textRect.x, &item->textRect.y, &item->window);
//
buff[newLine] = '\0';
DC->drawText(item->textRect.x, item->textRect.y, item->textscale, color, buff, 0, item->textStyle, item->font);
}
if (*p == '\0')
{
break;
}
//
y += height + 5;
p = newLinePtr;
len = 0;
newLine = 0;
newLineWidth = 0;
continue;
}
buff[len++] = *p++;
buff[len] = '\0';
}
}
/*
=================
Rect_ContainsPoint
=================
*/
qboolean Rect_ContainsPoint(rectDef_t *rect, float x, float y)
{
if (rect)
{
// if ((x > rect->x) && (x < (rect->x + rect->w)) && (y > rect->y) && (y < (rect->y + rect->h)))
if ((x > rect->x) && (x < (rect->x + rect->w)))
{
if ((y > rect->y) && (y < (rect->y + rect->h)))
{
return qtrue;
}
}
}
return qfalse;
}
/*
=================
Item_ListBox_MaxScroll
=================
*/
int Item_ListBox_MaxScroll(itemDef_t *item)
{
listBoxDef_t *listPtr = (listBoxDef_t*)item->typeData;
int count = DC->feederCount(item->special);
int max;
if (item->window.flags & WINDOW_HORIZONTAL)
{
max = count - (item->window.rect.w / listPtr->elementWidth) + 1;
}
else
{
max = count - (item->window.rect.h / listPtr->elementHeight) + 1;
}
if (max < 0)
{
return 0;
}
return max;
}
/*
=================
Item_ListBox_ThumbPosition
=================
*/
int Item_ListBox_ThumbPosition(itemDef_t *item)
{
float max, pos, size;
listBoxDef_t *listPtr = (listBoxDef_t*)item->typeData;
max = Item_ListBox_MaxScroll(item);
if (item->window.flags & WINDOW_HORIZONTAL) {
size = item->window.rect.w - (SCROLLBAR_SIZE * 2) - 2;
if (max > 0)
{
pos = (size-SCROLLBAR_SIZE) / (float) max;
}
else
{
pos = 0;
}
pos *= listPtr->startPos;
return item->window.rect.x + 1 + SCROLLBAR_SIZE + pos;
}
else
{
size = item->window.rect.h - (SCROLLBAR_SIZE * 2) - 2;
if (max > 0)
{
pos = (size-SCROLLBAR_SIZE) / (float) max;
}
else
{
pos = 0;
}
pos *= listPtr->startPos;
return item->window.rect.y + 1 + SCROLLBAR_SIZE + pos;
}
}
/*
=================
Item_ListBox_OverLB
=================
*/
int Item_ListBox_OverLB(itemDef_t *item, float x, float y)
{
rectDef_t r;
listBoxDef_t *listPtr;
int thumbstart;
int count;
count = DC->feederCount(item->special);
listPtr = (listBoxDef_t*)item->typeData;
if (item->window.flags & WINDOW_HORIZONTAL)
{
// check if on left arrow
r.x = item->window.rect.x;
r.y = item->window.rect.y + item->window.rect.h - SCROLLBAR_SIZE;
r.h = r.w = SCROLLBAR_SIZE;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_LEFTARROW;
}
// check if on right arrow
r.x = item->window.rect.x + item->window.rect.w - SCROLLBAR_SIZE;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_RIGHTARROW;
}
// check if on thumb
thumbstart = Item_ListBox_ThumbPosition(item);
r.x = thumbstart;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_THUMB;
}
r.x = item->window.rect.x + SCROLLBAR_SIZE;
r.w = thumbstart - r.x;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_PGUP;
}
r.x = thumbstart + SCROLLBAR_SIZE;
r.w = item->window.rect.x + item->window.rect.w - SCROLLBAR_SIZE;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_PGDN;
}
}
else
{
r.x = item->window.rect.x + item->window.rect.w - SCROLLBAR_SIZE;
r.y = item->window.rect.y;
r.h = r.w = SCROLLBAR_SIZE;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_LEFTARROW;
}
r.y = item->window.rect.y + item->window.rect.h - SCROLLBAR_SIZE;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_RIGHTARROW;
}
thumbstart = Item_ListBox_ThumbPosition(item);
r.y = thumbstart;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_THUMB;
}
r.y = item->window.rect.y + SCROLLBAR_SIZE;
r.h = thumbstart - r.y;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_PGUP;
}
r.y = thumbstart + SCROLLBAR_SIZE;
r.h = item->window.rect.y + item->window.rect.h - SCROLLBAR_SIZE;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_PGDN;
}
}
return 0;
}
/*
=================
Item_ListBox_MouseEnter
=================
*/
void Item_ListBox_MouseEnter(itemDef_t *item, float x, float y)
{
rectDef_t r;
listBoxDef_t *listPtr = (listBoxDef_t*)item->typeData;
item->window.flags &= ~(WINDOW_LB_LEFTARROW | WINDOW_LB_RIGHTARROW | WINDOW_LB_THUMB | WINDOW_LB_PGUP | WINDOW_LB_PGDN);
item->window.flags |= Item_ListBox_OverLB(item, x, y);
if (item->window.flags & WINDOW_HORIZONTAL)
{
if (!(item->window.flags & (WINDOW_LB_LEFTARROW | WINDOW_LB_RIGHTARROW | WINDOW_LB_THUMB | WINDOW_LB_PGUP | WINDOW_LB_PGDN)))
{
// check for selection hit as we have exausted buttons and thumb
if (listPtr->elementStyle == LISTBOX_IMAGE)
{
r.x = item->window.rect.x;
r.y = item->window.rect.y;
r.h = item->window.rect.h - SCROLLBAR_SIZE;
r.w = item->window.rect.w - listPtr->drawPadding;
if (Rect_ContainsPoint(&r, x, y))
{
listPtr->cursorPos = (int)((x - r.x) / listPtr->elementWidth) + listPtr->startPos;
if (listPtr->cursorPos >= listPtr->endPos)
{
listPtr->cursorPos = listPtr->endPos;
}
}
}
else
{
// text hit..
}
}
}
else if (!(item->window.flags & (WINDOW_LB_LEFTARROW | WINDOW_LB_RIGHTARROW | WINDOW_LB_THUMB | WINDOW_LB_PGUP | WINDOW_LB_PGDN)))
{
r.x = item->window.rect.x;
r.y = item->window.rect.y;
r.w = item->window.rect.w - SCROLLBAR_SIZE;
r.h = item->window.rect.h - listPtr->drawPadding;
if (Rect_ContainsPoint(&r, x, y))
{
listPtr->cursorPos = (int)((y - 2 - r.y) / listPtr->elementHeight) + listPtr->startPos;
if (listPtr->cursorPos > listPtr->endPos)
{
listPtr->cursorPos = listPtr->endPos;
}
}
}
}
/*
=================
Item_MouseEnter
=================
*/
void Item_MouseEnter(itemDef_t *item, float x, float y)
{
rectDef_t r;
if (item)
{
r = item->textRect;
// r.y -= r.h; // NOt sure why this is here, but I commented it out.
// in the text rect?
// items can be enabled and disabled based on cvars
if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE))
{
return;
}
if (item->cvarFlags & (CVAR_SHOW | CVAR_HIDE) && !Item_EnableShowViaCvar(item, CVAR_SHOW))
{
return;
}
if (Rect_ContainsPoint(&r, x, y))
{
if (!(item->window.flags & WINDOW_MOUSEOVERTEXT))
{
Item_RunScript(item, item->mouseEnterText);
item->window.flags |= WINDOW_MOUSEOVERTEXT;
}
if (!(item->window.flags & WINDOW_MOUSEOVER))
{
Item_RunScript(item, item->mouseEnter);
item->window.flags |= WINDOW_MOUSEOVER;
}
}
else
{
// not in the text rect
if (item->window.flags & WINDOW_MOUSEOVERTEXT)
{
// if we were
Item_RunScript(item, item->mouseExitText);
item->window.flags &= ~WINDOW_MOUSEOVERTEXT;
}
if (!(item->window.flags & WINDOW_MOUSEOVER))
{
Item_RunScript(item, item->mouseEnter);
item->window.flags |= WINDOW_MOUSEOVER;
}
if (item->type == ITEM_TYPE_LISTBOX)
{
Item_ListBox_MouseEnter(item, x, y);
}
}
}
}
/*
=================
Item_SetFocus
=================
*/
// will optionaly set focus to this item
qboolean Item_SetFocus(itemDef_t *item, float x, float y)
{
int i;
itemDef_t *oldFocus;
sfxHandle_t *sfx = &DC->Assets.itemFocusSound;
qboolean playSound = qfalse;
menuDef_t *parent = (menuDef_t*)item->parent;
// sanity check, non-null, not a decoration and does not already have the focus
if (item == NULL || item->window.flags & WINDOW_DECORATION || item->window.flags & WINDOW_HASFOCUS || !(item->window.flags & WINDOW_VISIBLE))
{
return qfalse;
}
// items can be enabled and disabled based on cvars
if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE))
{
return qfalse;
}
if (item->cvarFlags & (CVAR_SHOW | CVAR_HIDE) && !Item_EnableShowViaCvar(item, CVAR_SHOW))
{
return qfalse;
}
oldFocus = Menu_ClearFocus((menuDef_t *) item->parent);
if (item->type == ITEM_TYPE_TEXT)
{
rectDef_t r;
r = item->textRect;
r.y -= r.h;
if (Rect_ContainsPoint(&r, x, y))
{
item->window.flags |= WINDOW_HASFOCUS;
if (item->focusSound)
{
sfx = &item->focusSound;
}
playSound = qtrue;
}
else
{
if (oldFocus)
{
oldFocus->window.flags |= WINDOW_HASFOCUS;
if (oldFocus->onFocus)
{
Item_RunScript(oldFocus, oldFocus->onFocus);
}
}
}
}
else
{
item->window.flags |= WINDOW_HASFOCUS;
if (item->onFocus)
{
Item_RunScript(item, item->onFocus);
}
if (item->focusSound)
{
sfx = &item->focusSound;
}
playSound = qtrue;
}
if (playSound && sfx)
{
DC->startLocalSound( *sfx, CHAN_LOCAL_SOUND );
}
for (i = 0; i < parent->itemCount; i++)
{
if (parent->items[i] == item)
{
parent->cursorItem = i;
break;
}
}
return qtrue;
}
/*
=================
IsVisible
=================
*/
qboolean IsVisible(int flags)
{
return (flags & WINDOW_VISIBLE && !(flags & WINDOW_FADINGOUT));
}
/*
=================
Item_MouseLeave
=================
*/
void Item_MouseLeave(itemDef_t *item)
{
if (item)
{
if (item->window.flags & WINDOW_MOUSEOVERTEXT)
{
Item_RunScript(item, item->mouseExitText);
item->window.flags &= ~WINDOW_MOUSEOVERTEXT;
}
Item_RunScript(item, item->mouseExit);
item->window.flags &= ~(WINDOW_LB_RIGHTARROW | WINDOW_LB_LEFTARROW);
}
}
/*
=================
Item_SetMouseOver
=================
*/
void Item_SetMouseOver(itemDef_t *item, qboolean focus)
{
if (item)
{
if (focus)
{
item->window.flags |= WINDOW_MOUSEOVER;
}
else
{
item->window.flags &= ~WINDOW_MOUSEOVER;
}
}
}
/*
=================
Menu_HandleMouseMove
=================
*/
void Menu_HandleMouseMove(menuDef_t *menu, float x, float y)
{
int i, pass;
qboolean focusSet = qfalse;
itemDef_t *overItem;
if (menu == NULL)
{
return;
}
if (!(menu->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED)))
{
return;
}
if (itemCapture)
{
//Item_MouseMove(itemCapture, x, y);
return;
}
if (g_waitingForKey || g_editingField)
{
return;
}
// FIXME: this is the whole issue of focus vs. mouse over..
// need a better overall solution as i don't like going through everything twice
for (pass = 0; pass < 2; pass++)
{
for (i = 0; i < menu->itemCount; i++)
{
// turn off focus each item
// menu->items[i].window.flags &= ~WINDOW_HASFOCUS;
if (!(menu->items[i]->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED)))
{
continue;
}
if (menu->items[i]->window.flags & WINDOW_INACTIVE)
{
continue;
}
// items can be enabled and disabled based on cvars
if (menu->items[i]->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(menu->items[i], CVAR_ENABLE))
{
continue;
}
if (menu->items[i]->cvarFlags & (CVAR_SHOW | CVAR_HIDE) && !Item_EnableShowViaCvar(menu->items[i], CVAR_SHOW))
{
continue;
}
if (Rect_ContainsPoint(&menu->items[i]->window.rect, x, y))
{
if (pass == 1)
{
overItem = menu->items[i];
if (overItem->type == ITEM_TYPE_TEXT && overItem->text)
{
if (!Rect_ContainsPoint(Item_CorrectedTextRect(overItem), x, y))
{
continue;
}
}
// if we are over an item
if (IsVisible(overItem->window.flags))
{
// different one
Item_MouseEnter(overItem, x, y);
// Item_SetMouseOver(overItem, qtrue);
// if item is not a decoration see if it can take focus
if (!focusSet)
{
focusSet = Item_SetFocus(overItem, x, y);
}
}
}
} else if (menu->items[i]->window.flags & WINDOW_MOUSEOVER)
{
Item_MouseLeave(menu->items[i]);
Item_SetMouseOver(menu->items[i], qfalse);
}
}
}
}
/*
=================
Display_MouseMove
=================
*/
qboolean Display_MouseMove(void *p, int x, int y)
{
int i;
menuDef_t *menu = (menuDef_t *) p;
if (menu == NULL)
{
menu = Menu_GetFocused();
if (menu)
{
if (menu->window.flags & WINDOW_POPUP)
{
Menu_HandleMouseMove(menu, x, y);
return qtrue;
}
}
for (i = 0; i < menuCount; i++)
{
Menu_HandleMouseMove(&Menus[i], x, y);
}
}
else
{
menu->window.rect.x += x;
menu->window.rect.y += y;
Menu_UpdatePosition(menu);
}
return qtrue;
}
/*
=================
Menus_AnyFullScreenVisible
=================
*/
qboolean Menus_AnyFullScreenVisible(void)
{
int i;
for (i = 0; i < menuCount; i++)
{
if (Menus[i].window.flags & WINDOW_VISIBLE && Menus[i].fullScreen)
{
return qtrue;
}
}
return qfalse;
}
/*
=================
BindingIDFromName
=================
*/
int BindingIDFromName(const char *name)
{
int i;
for (i=0; i < g_bindCount; i++)
{
if (Q_stricmp(name, g_bindings[i].command) == 0)
{
return i;
}
}
return -1;
}
/*
=================
Controls_SetConfig
=================
*/
void Controls_SetConfig(qboolean restart)
{
int i;
// iterate each command, get its numeric binding
for (i=0; i < g_bindCount; i++)
{
if (g_bindings[i].bind1 != -1)
{
DC->setBinding( g_bindings[i].bind1, g_bindings[i].command );
if (g_bindings[i].bind2 != -1)
DC->setBinding( g_bindings[i].bind2, g_bindings[i].command );
}
}
//if ( s_controls.invertmouse.curvalue )
// DC->setCVar("m_pitch", va("%f),-fabs( DC->getCVarValue( "m_pitch" ) ) );
//else
// trap_Cvar_SetValue( "m_pitch", fabs( trap_Cvar_VariableValue( "m_pitch" ) ) );
//trap_Cvar_SetValue( "m_filter", s_controls.smoothmouse.curvalue );
//trap_Cvar_SetValue( "cl_run", s_controls.alwaysrun.curvalue );
//trap_Cvar_SetValue( "cg_autoswitch", s_controls.autoswitch.curvalue );
//trap_Cvar_SetValue( "sensitivity", s_controls.sensitivity.curvalue );
//trap_Cvar_SetValue( "in_joystick", s_controls.joyenable.curvalue );
//trap_Cvar_SetValue( "joy_threshold", s_controls.joythreshold.curvalue );
//trap_Cvar_SetValue( "cl_freelook", s_controls.freelook.curvalue );
DC->executeText(EXEC_APPEND, "in_restart\n");
}
void Item_Bind_Ungrey(itemDef_t *item)
{
menuDef_t *menu;
int i;
menu = (menuDef_t *) item->parent;
for (i=0;i<menu->itemCount;++i)
{
if (menu->items[i] == item)
{
continue;
}
menu->items[i]->window.flags &= ~WINDOW_INACTIVE;
}
}
/*
=================
Item_Bind_HandleKey
=================
*/
qboolean Item_Bind_HandleKey(itemDef_t *item, int key, qboolean down)
{
int id;
int i;
menuDef_t *menu;
if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory) && !g_waitingForKey)
{
if (down && (key == K_MOUSE1 || key == K_ENTER))
{
g_waitingForKey = qtrue;
g_bindItem = item;
// Set all others in the menu to grey
menu = (menuDef_t *) item->parent;
for (i=0;i<menu->itemCount;++i)
{
if (menu->items[i] == item)
{
continue;
}
menu->items[i]->window.flags |= WINDOW_INACTIVE;
}
}
return qtrue;
}
else
{
if (!g_waitingForKey || g_bindItem == NULL)
{
return qtrue;
}
if (key & K_CHAR_FLAG)
{
return qtrue;
}
switch (key)
{
case K_ESCAPE:
g_waitingForKey = qfalse;
Item_Bind_Ungrey(item);
return qtrue;
case K_BACKSPACE:
id = BindingIDFromName(item->cvar);
if (id != -1)
{
DC->setBinding( g_bindings[id].bind1, "" );
DC->setBinding( g_bindings[id].bind2, "" );
g_bindings[id].bind1 = -1;
g_bindings[id].bind2 = -1;
}
Controls_SetConfig(qtrue);
g_waitingForKey = qfalse;
g_bindItem = NULL;
Item_Bind_Ungrey(item);
return qtrue;
break;
case '`':
return qtrue;
}
}
// Is the same key being bound to something else?
if (key != -1)
{
for (i=0; i < g_bindCount; i++)
{
// The second binding matches the key
if (g_bindings[i].bind2 == key)
{
g_bindings[i].bind2 = -1; // NULL it out
}
if (g_bindings[i].bind1 == key)
{
g_bindings[i].bind1 = g_bindings[i].bind2;
g_bindings[i].bind2 = -1;
}
}
}
id = BindingIDFromName(item->cvar);
if (id != -1)
{
if (key == -1)
{
if( g_bindings[id].bind1 != -1 )
{
DC->setBinding( g_bindings[id].bind1, "" );
g_bindings[id].bind1 = -1;
}
if( g_bindings[id].bind2 != -1 )
{
DC->setBinding( g_bindings[id].bind2, "" );
g_bindings[id].bind2 = -1;
}
}
else if (g_bindings[id].bind1 == -1)
{
g_bindings[id].bind1 = key;
}
else if (g_bindings[id].bind1 != key && g_bindings[id].bind2 == -1)
{
g_bindings[id].bind2 = key;
}
else
{
DC->setBinding( g_bindings[id].bind1, "" );
DC->setBinding( g_bindings[id].bind2, "" );
g_bindings[id].bind1 = key;
g_bindings[id].bind2 = -1;
}
}
Controls_SetConfig(qtrue);
g_waitingForKey = qfalse;
Item_Bind_Ungrey(item);
return qtrue;
}
/*
=================
Menu_SetNextCursorItem
=================
*/
itemDef_t *Menu_SetNextCursorItem(menuDef_t *menu)
{
qboolean wrapped = qfalse;
int oldCursor = menu->cursorItem;
if (menu->cursorItem == -1)
{
menu->cursorItem = 0;
wrapped = qtrue;
}
while (menu->cursorItem < menu->itemCount)
{
menu->cursorItem++;
if (menu->cursorItem >= menu->itemCount && !wrapped)
{
wrapped = qtrue;
menu->cursorItem = 0;
}
if (Item_SetFocus(menu->items[menu->cursorItem], DC->cursorx, DC->cursory))
{
Menu_HandleMouseMove(menu, menu->items[menu->cursorItem]->window.rect.x + 1, menu->items[menu->cursorItem]->window.rect.y + 1);
return menu->items[menu->cursorItem];
}
}
menu->cursorItem = oldCursor;
return NULL;
}
/*
=================
Menu_SetPrevCursorItem
=================
*/
itemDef_t *Menu_SetPrevCursorItem(menuDef_t *menu)
{
qboolean wrapped = qfalse;
int oldCursor = menu->cursorItem;
if (menu->cursorItem < 0)
{
menu->cursorItem = menu->itemCount-1;
wrapped = qtrue;
}
while (menu->cursorItem > -1)
{
menu->cursorItem--;
if (menu->cursorItem < 0 && !wrapped)
{
wrapped = qtrue;
menu->cursorItem = menu->itemCount -1;
}
if (Item_SetFocus(menu->items[menu->cursorItem], DC->cursorx, DC->cursory))
{
Menu_HandleMouseMove(menu, menu->items[menu->cursorItem]->window.rect.x + 1, menu->items[menu->cursorItem]->window.rect.y + 1);
return menu->items[menu->cursorItem];
}
}
menu->cursorItem = oldCursor;
return NULL;
}
/*
=================
Item_TextField_HandleKey
=================
*/
qboolean Item_TextField_HandleKey(itemDef_t *item, int key)
{
char buff[1024];
int len;
itemDef_t *newItem = NULL;
editFieldDef_t *editPtr = (editFieldDef_t*)item->typeData;
if (item->cvar)
{
memset(buff, 0, sizeof(buff));
DC->getCVarString(item->cvar, buff, sizeof(buff));
len = strlen(buff);
if (editPtr->maxChars && len > editPtr->maxChars)
{
len = editPtr->maxChars;
}
if ( key & K_CHAR_FLAG )
{
key &= ~K_CHAR_FLAG;
if (key == 'h' - 'a' + 1 )
{ // ctrl-h is backspace
if ( item->cursorPos > 0 )
{
memmove( &buff[item->cursorPos - 1], &buff[item->cursorPos], len + 1 - item->cursorPos);
item->cursorPos--;
if (item->cursorPos < editPtr->paintOffset)
{
editPtr->paintOffset--;
}
}
DC->setCVar(item->cvar, buff);
return qtrue;
}
//
// ignore any non printable chars
//
if ( key < 32 || !item->cvar)
{
return qtrue;
}
if (item->type == ITEM_TYPE_NUMERICFIELD)
{
if (key < '0' || key > '9')
{
return qfalse;
}
}
if (!DC->getOverstrikeMode())
{
if (( len == MAX_EDITFIELD - 1 ) || (editPtr->maxChars && len >= editPtr->maxChars))
{
return qtrue;
}
memmove( &buff[item->cursorPos + 1], &buff[item->cursorPos], len + 1 - item->cursorPos );
}
else
{
if (editPtr->maxChars && item->cursorPos >= editPtr->maxChars)
{
return qtrue;
}
}
buff[item->cursorPos] = key;
DC->setCVar(item->cvar, buff);
if (item->cursorPos < len + 1)
{
item->cursorPos++;
if (editPtr->maxPaintChars && item->cursorPos > editPtr->maxPaintChars)
{
editPtr->paintOffset++;
}
}
}
else
{
if ( key == K_DEL || key == K_KP_DEL )
{
if ( item->cursorPos < len )
{
memmove( buff + item->cursorPos, buff + item->cursorPos + 1, len - item->cursorPos);
DC->setCVar(item->cvar, buff);
}
return qtrue;
}
if ( key == K_RIGHTARROW || key == K_KP_RIGHTARROW )
{
if (editPtr->maxPaintChars && item->cursorPos >= editPtr->maxPaintChars && item->cursorPos < len)
{
item->cursorPos++;
editPtr->paintOffset++;
return qtrue;
}
if (item->cursorPos < len)
{
item->cursorPos++;
}
return qtrue;
}
if ( key == K_LEFTARROW || key == K_KP_LEFTARROW )
{
if ( item->cursorPos > 0 )
{
item->cursorPos--;
}
if (item->cursorPos < editPtr->paintOffset)
{
editPtr->paintOffset--;
}
return qtrue;
}
if ( key == K_HOME || key == K_KP_HOME)
{
item->cursorPos = 0;
editPtr->paintOffset = 0;
return qtrue;
}
if ( key == K_END || key == K_KP_END)
{
item->cursorPos = len;
if(item->cursorPos > editPtr->maxPaintChars)
{
editPtr->paintOffset = len - editPtr->maxPaintChars;
}
return qtrue;
}
if ( key == K_INS || key == K_KP_INS )
{
DC->setOverstrikeMode(!DC->getOverstrikeMode());
return qtrue;
}
}
if (key == K_TAB || key == K_DOWNARROW || key == K_KP_DOWNARROW)
{
newItem = Menu_SetNextCursorItem((menuDef_t *) item->parent);
if (newItem && (newItem->type == ITEM_TYPE_EDITFIELD || newItem->type == ITEM_TYPE_NUMERICFIELD))
{
g_editItem = newItem;
}
}
if (key == K_UPARROW || key == K_KP_UPARROW)
{
newItem = Menu_SetPrevCursorItem((menuDef_t *) item->parent);
if (newItem && (newItem->type == ITEM_TYPE_EDITFIELD || newItem->type == ITEM_TYPE_NUMERICFIELD))
{
g_editItem = newItem;
}
}
if ( key == K_ENTER || key == K_KP_ENTER || key == K_ESCAPE)
{
return qfalse;
}
return qtrue;
}
return qfalse;
}
/*
=================
Menu_OverActiveItem
=================
*/
static qboolean Menu_OverActiveItem(menuDef_t *menu, float x, float y)
{
if (menu && menu->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED))
{
if (Rect_ContainsPoint(&menu->window.rect, x, y))
{
int i;
for (i = 0; i < menu->itemCount; i++)
{
// turn off focus each item
// menu->items[i].window.flags &= ~WINDOW_HASFOCUS;
if (!(menu->items[i]->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED)))
{
continue;
}
if (menu->items[i]->window.flags & WINDOW_DECORATION)
{
continue;
}
if (Rect_ContainsPoint(&menu->items[i]->window.rect, x, y))
{
itemDef_t *overItem = menu->items[i];
if (overItem->type == ITEM_TYPE_TEXT && overItem->text)
{
if (Rect_ContainsPoint(Item_CorrectedTextRect(overItem), x, y))
{
return qtrue;
}
else
{
continue;
}
}
else
{
return qtrue;
}
}
}
}
}
return qfalse;
}
/*
=================
Display_VisibleMenuCount
=================
*/
int Display_VisibleMenuCount(void)
{
int i, count;
count = 0;
for (i = 0; i < menuCount; i++)
{
if (Menus[i].window.flags & (WINDOW_FORCED | WINDOW_VISIBLE))
{
count++;
}
}
return count;
}
/*
=================
Window_CloseCinematic
=================
*/
static void Window_CloseCinematic(windowDef_t *window)
{
if (window->style == WINDOW_STYLE_CINEMATIC && window->cinematic >= 0)
{
DC->stopCinematic(window->cinematic);
window->cinematic = -1;
}
}
/*
=================
Menu_CloseCinematics
=================
*/
static void Menu_CloseCinematics(menuDef_t *menu)
{
if (menu)
{
int i;
Window_CloseCinematic(&menu->window);
for (i = 0; i < menu->itemCount; i++)
{
Window_CloseCinematic(&menu->items[i]->window);
if (menu->items[i]->type == ITEM_TYPE_OWNERDRAW)
{
DC->stopCinematic(0-menu->items[i]->window.ownerDraw);
}
}
}
}
/*
=================
Display_CloseCinematics
=================
*/
static void Display_CloseCinematics()
{
int i;
for (i = 0; i < menuCount; i++)
{
Menu_CloseCinematics(&Menus[i]);
}
}
/*
=================
Menus_HandleOOBClick
=================
*/
void Menus_HandleOOBClick(menuDef_t *menu, int key, qboolean down)
{
if (menu)
{
int i;
// basically the behaviour we are looking for is if there are windows in the stack.. see if
// the cursor is within any of them.. if not close them otherwise activate them and pass the
// key on.. force a mouse move to activate focus and script stuff
if (down && menu->window.flags & WINDOW_OOB_CLICK)
{
Menu_RunCloseScript(menu);
menu->window.flags &= ~(WINDOW_HASFOCUS | WINDOW_VISIBLE);
}
for (i = 0; i < menuCount; i++)
{
if (Menu_OverActiveItem(&Menus[i], DC->cursorx, DC->cursory))
{
Menu_RunCloseScript(menu);
menu->window.flags &= ~(WINDOW_HASFOCUS | WINDOW_VISIBLE);
Menus_Activate(&Menus[i]);
Menu_HandleMouseMove(&Menus[i], DC->cursorx, DC->cursory);
Menu_HandleKey(&Menus[i], key, down);
}
}
if (Display_VisibleMenuCount() == 0)
{
if (DC->Pause)
{
DC->Pause(qfalse);
}
}
Display_CloseCinematics();
}
}
/*
=================
Item_StopCapture
=================
*/
void Item_StopCapture(itemDef_t *item)
{
}
/*
=================
Item_ListBox_HandleKey
=================
*/
qboolean Item_ListBox_HandleKey(itemDef_t *item, int key, qboolean down, qboolean force)
{
listBoxDef_t *listPtr = (listBoxDef_t*)item->typeData;
int count = DC->feederCount(item->special);
int max, viewmax;
if (force || (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory) && item->window.flags & WINDOW_HASFOCUS))
{
max = Item_ListBox_MaxScroll(item);
if (item->window.flags & WINDOW_HORIZONTAL)
{
viewmax = (item->window.rect.w / listPtr->elementWidth);
if ( key == K_LEFTARROW || key == K_KP_LEFTARROW )
{
if (!listPtr->notselectable)
{
listPtr->cursorPos--;
if (listPtr->cursorPos < 0)
{
listPtr->cursorPos = 0;
}
if (listPtr->cursorPos < listPtr->startPos)
{
listPtr->startPos = listPtr->cursorPos;
}
if (listPtr->cursorPos >= listPtr->startPos + viewmax)
{
listPtr->startPos = listPtr->cursorPos - viewmax + 1;
}
item->cursorPos = listPtr->cursorPos;
DC->feederSelection(item->special, item->cursorPos);
}
else
{
listPtr->startPos--;
if (listPtr->startPos < 0)
{
listPtr->startPos = 0;
}
}
return qtrue;
}
if ( key == K_RIGHTARROW || key == K_KP_RIGHTARROW )
{
if (!listPtr->notselectable)
{
listPtr->cursorPos++;
if (listPtr->cursorPos < listPtr->startPos)
{
listPtr->startPos = listPtr->cursorPos;
}
if (listPtr->cursorPos >= count)
{
listPtr->cursorPos = count-1;
}
if (listPtr->cursorPos >= listPtr->startPos + viewmax)
{
listPtr->startPos = listPtr->cursorPos - viewmax + 1;
}
item->cursorPos = listPtr->cursorPos;
DC->feederSelection(item->special, item->cursorPos);
}
else
{
listPtr->startPos++;
if (listPtr->startPos >= count)
{
listPtr->startPos = count-1;
}
}
return qtrue;
}
}
else
{
viewmax = (item->window.rect.h / listPtr->elementHeight);
if ( key == K_UPARROW || key == K_KP_UPARROW )
{
if (!listPtr->notselectable)
{
listPtr->cursorPos--;
if (listPtr->cursorPos < 0)
{
listPtr->cursorPos = 0;
}
if (listPtr->cursorPos < listPtr->startPos)
{
listPtr->startPos = listPtr->cursorPos;
}
if (listPtr->cursorPos >= listPtr->startPos + viewmax)
{
listPtr->startPos = listPtr->cursorPos - viewmax + 1;
}
item->cursorPos = listPtr->cursorPos;
DC->feederSelection(item->special, item->cursorPos);
}
else
{
listPtr->startPos--;
if (listPtr->startPos < 0)
{
listPtr->startPos = 0;
}
}
return qtrue;
}
if ( key == K_DOWNARROW || key == K_KP_DOWNARROW )
{
if (!listPtr->notselectable)
{
listPtr->cursorPos++;
if (listPtr->cursorPos < listPtr->startPos)
{
listPtr->startPos = listPtr->cursorPos;
}
if (listPtr->cursorPos >= count)
{
listPtr->cursorPos = count-1;
}
if (listPtr->cursorPos >= listPtr->startPos + viewmax)
{
listPtr->startPos = listPtr->cursorPos - viewmax + 1;
}
item->cursorPos = listPtr->cursorPos;
DC->feederSelection(item->special, item->cursorPos);
}
else
{
listPtr->startPos++;
if (listPtr->startPos > max)
{
listPtr->startPos = max;
}
}
return qtrue;
}
}
// mouse hit
if (key == K_MOUSE1 || key == K_MOUSE2)
{
if (item->window.flags & WINDOW_LB_LEFTARROW)
{
listPtr->startPos--;
if (listPtr->startPos < 0)
{
listPtr->startPos = 0;
}
}
else if (item->window.flags & WINDOW_LB_RIGHTARROW)
{
// one down
listPtr->startPos++;
if (listPtr->startPos > max)
{
listPtr->startPos = max;
}
}
else if (item->window.flags & WINDOW_LB_PGUP)
{
// page up
listPtr->startPos -= viewmax;
if (listPtr->startPos < 0)
{
listPtr->startPos = 0;
}
}
else if (item->window.flags & WINDOW_LB_PGDN)
{
// page down
listPtr->startPos += viewmax;
if (listPtr->startPos > max)
{
listPtr->startPos = max;
}
}
else if (item->window.flags & WINDOW_LB_THUMB)
{
// Display_SetCaptureItem(item);
}
else
{
// select an item
if (DC->realTime < lastListBoxClickTime && listPtr->doubleClick)
{
Item_RunScript(item, listPtr->doubleClick);
}
lastListBoxClickTime = DC->realTime + DOUBLE_CLICK_DELAY;
if (item->cursorPos != listPtr->cursorPos)
{
item->cursorPos = listPtr->cursorPos;
DC->feederSelection(item->special, item->cursorPos);
}
}
return qtrue;
}
if ( key == K_HOME || key == K_KP_HOME)
{
// home
listPtr->startPos = 0;
return qtrue;
}
if ( key == K_END || key == K_KP_END)
{
// end
listPtr->startPos = max;
return qtrue;
}
if (key == K_PGUP || key == K_KP_PGUP )
{
// page up
if (!listPtr->notselectable)
{
listPtr->cursorPos -= viewmax;
if (listPtr->cursorPos < 0)
{
listPtr->cursorPos = 0;
}
if (listPtr->cursorPos < listPtr->startPos)
{
listPtr->startPos = listPtr->cursorPos;
}
if (listPtr->cursorPos >= listPtr->startPos + viewmax)
{
listPtr->startPos = listPtr->cursorPos - viewmax + 1;
}
item->cursorPos = listPtr->cursorPos;
DC->feederSelection(item->special, item->cursorPos);
}
else
{
listPtr->startPos -= viewmax;
if (listPtr->startPos < 0)
{
listPtr->startPos = 0;
}
}
return qtrue;
}
if ( key == K_PGDN || key == K_KP_PGDN )
{
// page down
if (!listPtr->notselectable)
{
listPtr->cursorPos += viewmax;
if (listPtr->cursorPos < listPtr->startPos)
{
listPtr->startPos = listPtr->cursorPos;
}
if (listPtr->cursorPos >= count)
{
listPtr->cursorPos = count-1;
}
if (listPtr->cursorPos >= listPtr->startPos + viewmax)
{
listPtr->startPos = listPtr->cursorPos - viewmax + 1;
}
item->cursorPos = listPtr->cursorPos;
DC->feederSelection(item->special, item->cursorPos);
}
else
{
listPtr->startPos += viewmax;
if (listPtr->startPos > max)
{
listPtr->startPos = max;
}
}
return qtrue;
}
}
return qfalse;
}
/*
=================
Scroll_ListBox_AutoFunc
=================
*/
static void Scroll_ListBox_AutoFunc(void *p)
{
scrollInfo_t *si = (scrollInfo_t*)p;
if (DC->realTime > si->nextScrollTime)
{
// need to scroll which is done by simulating a click to the item
// this is done a bit sideways as the autoscroll "knows" that the item is a listbox
// so it calls it directly
Item_ListBox_HandleKey(si->item, si->scrollKey, qtrue, qfalse);
si->nextScrollTime = DC->realTime + si->adjustValue;
}
if (DC->realTime > si->nextAdjustTime)
{
si->nextAdjustTime = DC->realTime + SCROLL_TIME_ADJUST;
if (si->adjustValue > SCROLL_TIME_FLOOR)
{
si->adjustValue -= SCROLL_TIME_ADJUSTOFFSET;
}
}
}
/*
=================
Scroll_ListBox_ThumbFunc
=================
*/
static void Scroll_ListBox_ThumbFunc(void *p)
{
scrollInfo_t *si = (scrollInfo_t*)p;
rectDef_t r;
int pos, max;
listBoxDef_t *listPtr = (listBoxDef_t*)si->item->typeData;
if (si->item->window.flags & WINDOW_HORIZONTAL)
{
if (DC->cursorx == si->xStart)
{
return;
}
r.x = si->item->window.rect.x + SCROLLBAR_SIZE + 1;
r.y = si->item->window.rect.y + si->item->window.rect.h - SCROLLBAR_SIZE - 1;
r.h = SCROLLBAR_SIZE;
r.w = si->item->window.rect.w - (SCROLLBAR_SIZE*2) - 2;
max = Item_ListBox_MaxScroll(si->item);
//
pos = (DC->cursorx - r.x - SCROLLBAR_SIZE/2) * max / (r.w - SCROLLBAR_SIZE);
if (pos < 0)
{
pos = 0;
}
else if (pos > max)
{
pos = max;
}
listPtr->startPos = pos;
si->xStart = DC->cursorx;
}
else if (DC->cursory != si->yStart)
{
r.x = si->item->window.rect.x + si->item->window.rect.w - SCROLLBAR_SIZE - 1;
r.y = si->item->window.rect.y + SCROLLBAR_SIZE + 1;
r.h = si->item->window.rect.h - (SCROLLBAR_SIZE*2) - 2;
r.w = SCROLLBAR_SIZE;
max = Item_ListBox_MaxScroll(si->item);
//
pos = (DC->cursory - r.y - SCROLLBAR_SIZE/2) * max / (r.h - SCROLLBAR_SIZE);
if (pos < 0)
{
pos = 0;
}
else if (pos > max)
{
pos = max;
}
listPtr->startPos = pos;
si->yStart = DC->cursory;
}
if (DC->realTime > si->nextScrollTime)
{
// need to scroll which is done by simulating a click to the item
// this is done a bit sideways as the autoscroll "knows" that the item is a listbox
// so it calls it directly
Item_ListBox_HandleKey(si->item, si->scrollKey, qtrue, qfalse);
si->nextScrollTime = DC->realTime + si->adjustValue;
}
if (DC->realTime > si->nextAdjustTime)
{
si->nextAdjustTime = DC->realTime + SCROLL_TIME_ADJUST;
if (si->adjustValue > SCROLL_TIME_FLOOR)
{
si->adjustValue -= SCROLL_TIME_ADJUSTOFFSET;
}
}
}
/*
=================
Item_Slider_OverSlider
=================
*/
int Item_Slider_OverSlider(itemDef_t *item, float x, float y)
{
rectDef_t r;
r.x = Item_Slider_ThumbPosition(item) - (SLIDER_THUMB_WIDTH / 2);
r.y = item->window.rect.y - 2;
r.w = SLIDER_THUMB_WIDTH;
r.h = SLIDER_THUMB_HEIGHT;
if (Rect_ContainsPoint(&r, x, y))
{
return WINDOW_LB_THUMB;
}
return 0;
}
/*
=================
Scroll_Slider_ThumbFunc
=================
*/
static void Scroll_Slider_ThumbFunc(void *p)
{
float x, value, cursorx;
scrollInfo_t *si = (scrollInfo_t*)p;
editFieldDef_t *editDef = (struct editFieldDef_s *) si->item->typeData;
if (si->item->text)
{
x = si->item->textRect.x + si->item->textRect.w + 8;
}
else
{
x = si->item->window.rect.x;
}
cursorx = DC->cursorx;
if (cursorx < x)
{
cursorx = x;
}
else if (cursorx > x + SLIDER_WIDTH)
{
cursorx = x + SLIDER_WIDTH;
}
value = cursorx - x;
value /= SLIDER_WIDTH;
value *= (editDef->maxVal - editDef->minVal);
value += editDef->minVal;
DC->setCVar(si->item->cvar, va("%f", value));
}
/*
=================
Item_StopCapture
=================
*/
void Item_StartCapture(itemDef_t *item, int key)
{
int flags;
switch (item->type)
{
case ITEM_TYPE_EDITFIELD:
case ITEM_TYPE_NUMERICFIELD:
case ITEM_TYPE_LISTBOX:
{
flags = Item_ListBox_OverLB(item, DC->cursorx, DC->cursory);
if (flags & (WINDOW_LB_LEFTARROW | WINDOW_LB_RIGHTARROW))
{
scrollInfo.nextScrollTime = DC->realTime + SCROLL_TIME_START;
scrollInfo.nextAdjustTime = DC->realTime + SCROLL_TIME_ADJUST;
scrollInfo.adjustValue = SCROLL_TIME_START;
scrollInfo.scrollKey = key;
scrollInfo.scrollDir = (flags & WINDOW_LB_LEFTARROW) ? qtrue : qfalse;
scrollInfo.item = item;
captureData = &scrollInfo;
captureFunc = &Scroll_ListBox_AutoFunc;
itemCapture = item;
}
else if (flags & WINDOW_LB_THUMB)
{
scrollInfo.scrollKey = key;
scrollInfo.item = item;
scrollInfo.xStart = DC->cursorx;
scrollInfo.yStart = DC->cursory;
captureData = &scrollInfo;
captureFunc = &Scroll_ListBox_ThumbFunc;
itemCapture = item;
}
break;
}
case ITEM_TYPE_SLIDER:
{
flags = Item_Slider_OverSlider(item, DC->cursorx, DC->cursory);
if (flags & WINDOW_LB_THUMB)
{
scrollInfo.scrollKey = key;
scrollInfo.item = item;
scrollInfo.xStart = DC->cursorx;
scrollInfo.yStart = DC->cursory;
captureData = &scrollInfo;
captureFunc = &Scroll_Slider_ThumbFunc;
itemCapture = item;
}
break;
}
}
}
/*
=================
Item_YesNo_HandleKey
=================
*/
qboolean Item_YesNo_HandleKey(itemDef_t *item, int key)
{
if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory) && item->window.flags & WINDOW_HASFOCUS && item->cvar)
{
if (key == K_MOUSE1 || key == K_ENTER || key == K_MOUSE2 || key == K_MOUSE3)
{
DC->setCVar(item->cvar, va("%i", !DC->getCVarValue(item->cvar)));
return qtrue;
}
}
return qfalse;
}
/*
=================
Item_Multi_FindCvarByValue
=================
*/
int Item_Multi_FindCvarByValue(itemDef_t *item)
{
char buff[1024];
float value = 0;
int i;
multiDef_t *multiPtr = (multiDef_t*)item->typeData;
if (multiPtr)
{
if (multiPtr->strDef)
{
DC->getCVarString(item->cvar, buff, sizeof(buff));
}
else
{
value = DC->getCVarValue(item->cvar);
}
for (i = 0; i < multiPtr->count; i++)
{
if (multiPtr->strDef)
{
if (Q_stricmp(buff, multiPtr->cvarStr[i]) == 0)
{
return i;
}
}
else
{
if (multiPtr->cvarValue[i] == value)
{
return i;
}
}
}
}
return 0;
}
/*
=================
Item_Multi_CountSettings
=================
*/
int Item_Multi_CountSettings(itemDef_t *item)
{
multiDef_t *multiPtr = (multiDef_t*)item->typeData;
if (multiPtr == NULL)
{
return 0;
}
return multiPtr->count;
}
/*
=================
Item_OwnerDraw_HandleKey
=================
*/
qboolean Item_OwnerDraw_HandleKey(itemDef_t *item, int key)
{
if (item && DC->ownerDrawHandleKey)
{
return DC->ownerDrawHandleKey(item->window.ownerDraw, item->window.ownerDrawFlags, &item->special, key);
}
return qfalse;
}
/*
=================
Item_Multi_HandleKey
=================
*/
qboolean Item_Multi_HandleKey(itemDef_t *item, int key)
{
multiDef_t *multiPtr = (multiDef_t*)item->typeData;
if (multiPtr)
{
// if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory) && item->window.flags & WINDOW_HASFOCUS && item->cvar)
if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory) && item->window.flags & WINDOW_HASFOCUS)
{
if (key == K_MOUSE1 || key == K_ENTER || key == K_MOUSE2 || key == K_MOUSE3)
{
if (item->cvar)
{
int current = Item_Multi_FindCvarByValue(item) + 1;
int max = Item_Multi_CountSettings(item);
if ( current < 0 || current >= max )
{
current = 0;
}
if (multiPtr->strDef)
{
DC->setCVar(item->cvar, multiPtr->cvarStr[current]);
}
else
{
float value = multiPtr->cvarValue[current];
if (((float)((int) value)) == value)
{
DC->setCVar(item->cvar, va("%i", (int) value ));
}
else
{
DC->setCVar(item->cvar, va("%f", value ));
}
}
}
else
{
item->value++;
int max = Item_Multi_CountSettings(item);
if ( item->value < 0 || item->value >= max )
{
item->value = 0;
}
}
return qtrue;
}
}
}
return qfalse;
}
/*
=================
Item_Slider_HandleKey
=================
*/
qboolean Item_Slider_HandleKey(itemDef_t *item, int key, qboolean down)
{
float x, value, width, work;
//DC->Print("slider handle key\n");
if (item->window.flags & WINDOW_HASFOCUS && item->cvar && Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory))
{
if (key == K_MOUSE1 || key == K_ENTER || key == K_MOUSE2 || key == K_MOUSE3)
{
editFieldDef_t *editDef = (editFieldDef_s *) item->typeData;
if (editDef)
{
rectDef_t testRect;
width = SLIDER_WIDTH;
if (item->text)
{
x = item->textRect.x + item->textRect.w + 8;
}
else
{
x = item->window.rect.x;
}
testRect = item->window.rect;
testRect.x = x;
value = (float)SLIDER_THUMB_WIDTH / 2;
testRect.x -= value;
//DC->Print("slider x: %f\n", testRect.x);
testRect.w = (SLIDER_WIDTH + (float)SLIDER_THUMB_WIDTH / 2);
//DC->Print("slider w: %f\n", testRect.w);
if (Rect_ContainsPoint(&testRect, DC->cursorx, DC->cursory))
{
work = DC->cursorx - x;
value = work / width;
value *= (editDef->maxVal - editDef->minVal);
// vm fuckage
// value = (((float)(DC->cursorx - x)/ SLIDER_WIDTH) * (editDef->maxVal - editDef->minVal));
value += editDef->minVal;
DC->setCVar(item->cvar, va("%f", value));
return qtrue;
}
}
}
}
DC->Print("slider handle key exit\n");
return qfalse;
}
/*
=================
Item_HandleKey
=================
*/
qboolean Item_HandleKey(itemDef_t *item, int key, qboolean down)
{
if (itemCapture)
{
Item_StopCapture(itemCapture);
itemCapture = NULL;
captureFunc = NULL;
captureData = NULL;
}
else
{
if (down && key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3)
{
Item_StartCapture(item, key);
}
}
if (!down)
{
return qfalse;
}
switch (item->type)
{
case ITEM_TYPE_BUTTON:
return qfalse;
break;
case ITEM_TYPE_RADIOBUTTON:
return qfalse;
break;
case ITEM_TYPE_CHECKBOX:
return qfalse;
break;
case ITEM_TYPE_EDITFIELD:
case ITEM_TYPE_NUMERICFIELD:
//return Item_TextField_HandleKey(item, key);
return qfalse;
break;
case ITEM_TYPE_COMBO:
return qfalse;
break;
case ITEM_TYPE_LISTBOX:
return Item_ListBox_HandleKey(item, key, down, qfalse);
break;
case ITEM_TYPE_YESNO:
return Item_YesNo_HandleKey(item, key);
break;
case ITEM_TYPE_MULTI:
return Item_Multi_HandleKey(item, key);
break;
case ITEM_TYPE_OWNERDRAW:
return Item_OwnerDraw_HandleKey(item, key);
break;
case ITEM_TYPE_BIND:
return Item_Bind_HandleKey(item, key, down);
break;
case ITEM_TYPE_SLIDER:
return Item_Slider_HandleKey(item, key, down);
break;
default:
return qfalse;
break;
}
//return qfalse;
}
/*
=================
Item_Action
=================
*/
void Item_Action(itemDef_t *item)
{
if (item)
{
Item_RunScript(item, item->action);
}
}
/*
=================
Menu_HandleKey
=================
*/
void Menu_HandleKey(menuDef_t *menu, int key, qboolean down)
{
int i;
itemDef_t *item = NULL;
qboolean inHandler = qfalse;
if (inHandler)
{
return;
}
inHandler = qtrue;
if (g_waitingForKey && down)
{
Item_Bind_HandleKey(g_bindItem, key, down);
inHandler = qfalse;
return;
}
if (g_editingField && down)
{
if (!Item_TextField_HandleKey(g_editItem, key))
{
g_editingField = qfalse;
g_editItem = NULL;
inHandler = qfalse;
return;
}
else if (key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3)
{
g_editingField = qfalse;
g_editItem = NULL;
Display_MouseMove(NULL, DC->cursorx, DC->cursory);
}
else if (key == K_TAB || key == K_UPARROW || key == K_DOWNARROW)
{
return;
}
}
if (menu == NULL)
{
inHandler = qfalse;
return;
}
// see if the mouse is within the window bounds and if so is this a mouse click
if (down && !(menu->window.flags & WINDOW_POPUP) && !Rect_ContainsPoint(&menu->window.rect, DC->cursorx, DC->cursory))
{
static qboolean inHandleKey = qfalse;
if (!inHandleKey && key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3)
{
inHandleKey = qtrue;
Menus_HandleOOBClick(menu, key, down);
inHandleKey = qfalse;
inHandler = qfalse;
return;
}
}
// get the item with focus
for (i = 0; i < menu->itemCount; i++)
{
if (menu->items[i]->window.flags & WINDOW_HASFOCUS)
{
item = menu->items[i];
}
}
if (item != NULL)
{
if (Item_HandleKey(item, key, down))
{
Item_Action(item);
inHandler = qfalse;
return;
}
}
if (!down)
{
inHandler = qfalse;
return;
}
// default handling
switch ( key )
{
case K_F11:
if (DC->getCVarValue("developer"))
{
uis.debugMode ^= 1;
}
break;
case K_F12:
if (DC->getCVarValue("developer"))
{
DC->executeText(EXEC_APPEND, "screenshot\n");
}
break;
case K_KP_UPARROW:
case K_UPARROW:
Menu_SetPrevCursorItem(menu);
break;
case K_ESCAPE:
if (!g_waitingForKey && menu->onESC)
{
itemDef_t it;
it.parent = menu;
Item_RunScript(&it, menu->onESC);
}
break;
case K_TAB:
case K_KP_DOWNARROW:
case K_DOWNARROW:
Menu_SetNextCursorItem(menu);
break;
case K_MOUSE1:
case K_MOUSE2:
if (item)
{
if (item->type == ITEM_TYPE_TEXT)
{
if (Rect_ContainsPoint(Item_CorrectedTextRect(item), DC->cursorx, DC->cursory))
{
Item_Action(item);
}
}
else if (item->type == ITEM_TYPE_EDITFIELD || item->type == ITEM_TYPE_NUMERICFIELD)
{
if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory))
{
item->cursorPos = 0;
g_editingField = qtrue;
g_editItem = item;
DC->setOverstrikeMode(qtrue);
}
}
else
{
if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory))
{
Item_Action(item);
}
}
}
break;
case K_JOY1:
case K_JOY2:
case K_JOY3:
case K_JOY4:
case K_AUX1:
case K_AUX2:
case K_AUX3:
case K_AUX4:
case K_AUX5:
case K_AUX6:
case K_AUX7:
case K_AUX8:
case K_AUX9:
case K_AUX10:
case K_AUX11:
case K_AUX12:
case K_AUX13:
case K_AUX14:
case K_AUX15:
case K_AUX16:
break;
case K_KP_ENTER:
case K_ENTER:
if (item)
{
if (item->type == ITEM_TYPE_EDITFIELD || item->type == ITEM_TYPE_NUMERICFIELD)
{
item->cursorPos = 0;
g_editingField = qtrue;
g_editItem = item;
DC->setOverstrikeMode(qtrue);
}
else
{
Item_Action(item);
}
}
break;
}
inHandler = qfalse;
}
/*
=================
ParseRect
=================
*/
qboolean ParseRect(const char **p, rectDef_t *r)
{
if (!COM_ParseFloat(p, &r->x))
{
if (!COM_ParseFloat(p, &r->y))
{
if (!COM_ParseFloat(p, &r->w))
{
if (!COM_ParseFloat(p, &r->h))
{
return qtrue;
}
}
}
}
return qfalse;
}
/*
=================
Menus_HideItems
=================
*/
void Menus_HideItems(const char *menuName)
{
menuDef_t *menu;
int i;
menu =Menus_FindByName(menuName); // Get menu
if (!menu)
{
Com_Printf(S_COLOR_YELLOW"WARNING: No menu was found. Could not hide items.\n");
return;
}
menu->window.flags &= ~(WINDOW_HASFOCUS | WINDOW_VISIBLE);
for (i = 0; i < menu->itemCount; i++)
{
menu->items[i]->cvarFlags = CVAR_HIDE;
}
}
/*
=================
Menus_ShowItems
=================
*/
void Menus_ShowItems(const char *menuName)
{
menuDef_t *menu;
int i;
menu =Menus_FindByName(menuName); // Get menu
if (!menu)
{
Com_Printf(S_COLOR_YELLOW"WARNING: No menu was found. Could not show items.\n");
return;
}
menu->window.flags |= (WINDOW_HASFOCUS | WINDOW_VISIBLE);
for (i = 0; i < menu->itemCount; i++)
{
menu->items[i]->cvarFlags = CVAR_SHOW;
}
}
/*
=================
UI_Cursor_Show
=================
*/
void UI_Cursor_Show(qboolean flag)
{
DC->cursorShow = flag;
if ((DC->cursorShow != qtrue) && (DC->cursorShow != qfalse))
{
DC->cursorShow = qtrue;
}
}