summaryrefslogtreecommitdiff
path: root/src/ui/ui_shared.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/ui/ui_shared.c')
-rw-r--r--src/ui/ui_shared.c11444
1 files changed, 6558 insertions, 4886 deletions
diff --git a/src/ui/ui_shared.c b/src/ui/ui_shared.c
index 5640e0e..8b6225e 100644
--- a/src/ui/ui_shared.c
+++ b/src/ui/ui_shared.c
@@ -1,13 +1,14 @@
/*
===========================================================================
Copyright (C) 1999-2005 Id Software, Inc.
-Copyright (C) 2000-2006 Tim Angus
+Copyright (C) 2000-2013 Darklegion Development
+Copyright (C) 2015-2019 GrangerHub
This file is part of Tremulous.
Tremulous is free software; you can redistribute it
and/or modify it under the terms of the GNU General Public License as
-published by the Free Software Foundation; either version 2 of the License,
+published by the Free Software Foundation; either version 3 of the License,
or (at your option) any later version.
Tremulous is distributed in the hope that it will be
@@ -16,38 +17,40 @@ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
-along with Tremulous; if not, write to the Free Software
-Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+along with Tremulous; if not, see <https://www.gnu.org/licenses/>
+
===========================================================================
*/
#include "ui_shared.h"
-#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;
+#define SCROLL_TIME_START 500
+#define SCROLL_TIME_ADJUST 150
+#define SCROLL_TIME_ADJUSTOFFSET 40
+#define SCROLL_TIME_FLOOR 20
+
+typedef struct {
+ int nextScrollTime;
+ int nextAdjustTime;
+ int adjustValue;
+ int scrollKey;
+ float xStart;
+ float yStart;
+ itemDef_t *item;
+ qboolean scrollDir;
} scrollInfo_t;
static scrollInfo_t scrollInfo;
-//TA: hack to prevent compiler warnings
-void voidFunction( void *var ) { return; }
-qboolean voidFunction2( itemDef_t *var1, int var2 ) { return qfalse; }
+// prevent compiler warnings
+void voidFunction(void *var) { return; }
-static void (*captureFunc) (void *p) = voidFunction;
+qboolean voidFunction2(itemDef_t *var1, int var2) { return qfalse; }
+
+static CaptureFunc *captureFunc = voidFunction;
+static int captureFuncExpiry = 0;
static void *captureData = NULL;
-static itemDef_t *itemCapture = NULL; // item that has the mouse captured ( if any )
+static itemDef_t *itemCapture = NULL; // item that has the mouse captured ( if any )
displayContextDef_t *DC = NULL;
@@ -56,20 +59,23 @@ static qboolean g_editingField = qfalse;
static itemDef_t *g_bindItem = NULL;
static itemDef_t *g_editItem = NULL;
+static itemDef_t *g_comboBoxItem = NULL;
-menuDef_t Menus[MAX_MENUS]; // defined menus
-int menuCount = 0; // how many
+menuDef_t Menus[MAX_MENUS]; // defined menus
+int menuCount = 0; // how many
menuDef_t *menuStack[MAX_OPEN_MENUS];
int openMenuCount = 0;
-static qboolean debugMode = qfalse;
-
#define DOUBLE_CLICK_DELAY 300
static int lastListBoxClickTime = 0;
+itemDataType_t Item_DataType(itemDef_t *item);
void Item_RunScript(itemDef_t *item, const char *s);
void Item_SetupKeywordHash(void);
+static ID_INLINE qboolean Item_IsEditField(itemDef_t *item);
+static ID_INLINE qboolean Item_IsListBox(itemDef_t *item);
+static void Item_ListBox_SetStartPos(itemDef_t *item, int startPos);
void Menu_SetupKeywordHash(void);
int BindingIDFromName(const char *name);
qboolean Item_Bind_HandleKey(itemDef_t *item, int key, qboolean down);
@@ -77,40 +83,68 @@ itemDef_t *Menu_SetPrevCursorItem(menuDef_t *menu);
itemDef_t *Menu_SetNextCursorItem(menuDef_t *menu);
static qboolean Menu_OverActiveItem(menuDef_t *menu, float x, float y);
+/*
+===============
+UI_InstallCaptureFunc
+===============
+*/
+void UI_InstallCaptureFunc(CaptureFunc *f, void *data, int timeout)
+{
+ captureFunc = f;
+ captureData = data;
+
+ if (timeout > 0)
+ captureFuncExpiry = DC->realTime + timeout;
+ else
+ captureFuncExpiry = 0;
+}
+
+/*
+===============
+UI_RemoveCaptureFunc
+===============
+*/
+void UI_RemoveCaptureFunc(void)
+{
+ captureFunc = voidFunction;
+ captureData = NULL;
+ captureFuncExpiry = 0;
+}
+
#ifdef CGAME
-#define MEM_POOL_SIZE 128 * 1024
+#define MEM_POOL_SIZE 128 * 1024
#else
-#define MEM_POOL_SIZE 1024 * 1024
+#define MEM_POOL_SIZE 1024 * 1024
#endif
-//TA: hacked variable name to avoid conflict with new cgame Alloc
-static char UI_memoryPool[MEM_POOL_SIZE];
-static int allocPoint, outOfMemory;
+static char UI_memoryPool[MEM_POOL_SIZE];
+static int allocPoint, outOfMemory;
/*
===============
UI_Alloc
===============
*/
-void *UI_Alloc( int size )
+void *UI_Alloc(int size)
{
- char *p;
+ char *p;
+
+ if (allocPoint + size > MEM_POOL_SIZE)
+ {
+ outOfMemory = qtrue;
- if( allocPoint + size > MEM_POOL_SIZE )
- {
- outOfMemory = qtrue;
+ if (DC->Print)
+ DC->Print("UI_Alloc: Failure. Out of memory!\n");
- if( DC->Print )
- DC->Print( "UI_Alloc: Failure. Out of memory!\n" );
- //DC->trap_Print(S_COLOR_YELLOW"WARNING: UI Out of Memory!\n");
- return NULL;
- }
+ // DC->trap_Print(S_COLOR_YELLOW"WARNING: UI Out of Memory!\n");
+ return NULL;
+ }
- p = &UI_memoryPool[ allocPoint ];
+ p = &UI_memoryPool[allocPoint];
- allocPoint += ( size + 15 ) & ~15;
+ allocPoint += (size + 15) & ~15;
- return p;
+ return p;
}
/*
@@ -118,20 +152,13 @@ void *UI_Alloc( int size )
UI_InitMemory
===============
*/
-void UI_InitMemory( void )
+void UI_InitMemory(void)
{
- allocPoint = 0;
- outOfMemory = qfalse;
+ allocPoint = 0;
+ outOfMemory = qfalse;
}
-qboolean UI_OutOfMemory( )
-{
- return outOfMemory;
-}
-
-
-
-
+qboolean UI_OutOfMemory() { return outOfMemory; }
#define HASH_TABLE_SIZE 2048
/*
@@ -139,25 +166,29 @@ qboolean UI_OutOfMemory( )
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;
+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;
}
typedef struct stringDef_s {
- struct stringDef_s *next;
- const char *str;
+ struct stringDef_s *next;
+ const char *str;
} stringDef_t;
static int strPoolIndex = 0;
@@ -166,69 +197,81 @@ static char strPool[STRING_POOL_SIZE];
static int strHandleCount = 0;
static stringDef_t *strHandle[HASH_TABLE_SIZE];
+// Make a copy of a string for later use. Can safely be called on the
+// same string repeatedly. Redundant on string literals or global
+// constants.
+const char *String_Alloc(const char *p)
+{
+ int len;
+ long hash;
+ stringDef_t *str, *last;
-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 == NULL) {
- return NULL;
- }
+ if (*p == 0)
+ return "";
- if (*p == 0) {
- return staticNULL;
- }
+ hash = hashForString(p);
- hash = hashForString(p);
+ str = strHandle[hash];
+
+ while (str)
+ {
+ if (strcmp(p, str->str) == 0)
+ return str->str;
- str = strHandle[hash];
- while (str) {
- if (strcmp(p, str->str) == 0) {
- return str->str;
+ str = str->next;
}
- str = str->next;
- }
- len = strlen(p);
- if (len + strPoolIndex + 1 < STRING_POOL_SIZE) {
- int ph = strPoolIndex;
- strcpy(&strPool[strPoolIndex], p);
- strPoolIndex += len + 1;
+ len = strlen(p);
- str = strHandle[hash];
- last = str;
- while (str && str->next) {
- last = str;
- str = str->next;
- }
-
- str = 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;
-}
-
-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);
+ 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 = UI_Alloc(sizeof(stringDef_t));
+ str->next = NULL;
+ str->str = &strPool[ph];
+
+ if (last)
+ last->next = str;
+ else
+ strHandle[hash] = str;
+
+ return &strPool[ph];
+ }
+ else
+ {
+ Com_Error(ERR_DROP, "String_Alloc( %s ): string pool full!", p);
+ return NULL; // not that we return at all
+ }
+}
+
+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);
}
/*
@@ -236,22 +279,29 @@ void String_Report( void ) {
String_Init
=================
*/
-void String_Init( void )
+void String_Init(void)
{
- int i;
- for( i = 0; i < HASH_TABLE_SIZE; i++ )
- strHandle[ i ] = 0;
+ int i;
- strHandleCount = 0;
- strPoolIndex = 0;
- menuCount = 0;
- openMenuCount = 0;
- UI_InitMemory( );
- Item_SetupKeywordHash( );
- Menu_SetupKeywordHash( );
+ for (i = 0; i < HASH_TABLE_SIZE; i++)
+ strHandle[i] = 0;
- if( DC && DC->getBindingBuf )
- Controls_GetConfig( );
+ strHandleCount = 0;
+
+ strPoolIndex = 0;
+
+ menuCount = 0;
+
+ openMenuCount = 0;
+
+ UI_InitMemory();
+
+ Item_SetupKeywordHash();
+
+ Menu_SetupKeywordHash();
+
+ if (DC && DC->getBindingBuf)
+ Controls_GetConfig();
}
/*
@@ -259,21 +309,22 @@ void String_Init( void )
PC_SourceWarning
=================
*/
-void PC_SourceWarning(int handle, char *format, ...) {
- int line;
- char filename[128];
- va_list argptr;
- static char string[4096];
+__attribute__((format(printf, 2, 3))) void PC_SourceWarning(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);
+ va_start(argptr, format);
+ Q_vsnprintf(string, sizeof(string), format, argptr);
+ va_end(argptr);
- filename[0] = '\0';
- line = 0;
- trap_Parse_SourceFileAndLine(handle, filename, &line);
+ filename[0] = '\0';
+ line = 0;
+ trap_Parse_SourceFileAndLine(handle, filename, &line);
- Com_Printf(S_COLOR_YELLOW "WARNING: %s, line %d: %s\n", filename, line, string);
+ Com_Printf(S_COLOR_YELLOW "WARNING: %s, line %d: %s\n", filename, line, string);
}
/*
@@ -281,21 +332,22 @@ void PC_SourceWarning(int handle, char *format, ...) {
PC_SourceError
=================
*/
-void PC_SourceError(int handle, char *format, ...) {
- int line;
- char filename[128];
- va_list argptr;
- static char string[4096];
+__attribute__((format(printf, 2, 3))) 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);
+ va_start(argptr, format);
+ Q_vsnprintf(string, sizeof(string), format, argptr);
+ va_end(argptr);
- filename[0] = '\0';
- line = 0;
- trap_Parse_SourceFileAndLine(handle, filename, &line);
+ filename[0] = '\0';
+ line = 0;
+ trap_Parse_SourceFileAndLine(handle, filename, &line);
- Com_Printf(S_COLOR_RED "ERROR: %s, line %d: %s\n", filename, line, string);
+ Com_Printf(S_COLOR_RED "ERROR: %s, line %d: %s\n", filename, line, string);
}
/*
@@ -305,17 +357,19 @@ LerpColor
*/
void LerpColor(vec4_t a, vec4_t b, vec4_t c, float t)
{
- int i;
+ 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;
- }
+ // 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;
+ }
}
/*
@@ -323,15 +377,271 @@ void LerpColor(vec4_t a, vec4_t b, vec4_t c, float t)
Float_Parse
=================
*/
-qboolean Float_Parse(char **p, float *f) {
- char *token;
- token = COM_ParseExt(p, qfalse);
- if (token && token[0] != 0) {
- *f = atof(token);
+qboolean Float_Parse(char **p, float *f)
+{
+ char *token;
+ token = COM_ParseExt(p, qfalse);
+
+ if (token && token[0] != 0)
+ {
+ *f = atof(token);
+ return qtrue;
+ }
+ else
+ return qfalse;
+}
+
+#define MAX_EXPR_ELEMENTS 32
+
+typedef enum { EXPR_OPERATOR, EXPR_VALUE } exprType_t;
+
+typedef struct exprToken_s {
+ exprType_t type;
+ union {
+ char op;
+ float val;
+ } u;
+} exprToken_t;
+
+typedef struct exprList_s {
+ exprToken_t l[MAX_EXPR_ELEMENTS];
+ int f, b;
+} exprList_t;
+
+/*
+=================
+OpPrec
+
+Return a value reflecting operator precedence
+=================
+*/
+static ID_INLINE int OpPrec(char op)
+{
+ switch (op)
+ {
+ case '*':
+ return 4;
+
+ case '/':
+ return 3;
+
+ case '+':
+ return 2;
+
+ case '-':
+ return 1;
+
+ case '(':
+ return 0;
+
+ default:
+ return -1;
+ }
+}
+
+/*
+=================
+PC_Expression_Parse
+=================
+*/
+static qboolean PC_Expression_Parse(int handle, float *f)
+{
+ pc_token_t token;
+ int unmatchedParentheses = 0;
+ exprList_t stack, fifo;
+ exprToken_t value;
+ qboolean expectingNumber = qtrue;
+
+#define FULL(a) (a.b >= (MAX_EXPR_ELEMENTS - 1))
+#define EMPTY(a) (a.f > a.b)
+
+#define PUSH_VAL(a, v) \
+ { \
+ if (FULL(a)) \
+ return qfalse; \
+ a.b++; \
+ a.l[a.b].type = EXPR_VALUE; \
+ a.l[a.b].u.val = v; \
+ }
+
+#define PUSH_OP(a, o) \
+ { \
+ if (FULL(a)) \
+ return qfalse; \
+ a.b++; \
+ a.l[a.b].type = EXPR_OPERATOR; \
+ a.l[a.b].u.op = o; \
+ }
+
+#define POP_STACK(a) \
+ { \
+ if (EMPTY(a)) \
+ return qfalse; \
+ value = a.l[a.b]; \
+ a.b--; \
+ }
+
+#define PEEK_STACK_OP(a) (a.l[a.b].u.op)
+#define PEEK_STACK_VAL(a) (a.l[a.b].u.val)
+
+#define POP_FIFO(a) \
+ { \
+ if (EMPTY(a)) \
+ return qfalse; \
+ value = a.l[a.f]; \
+ a.f++; \
+ }
+
+ stack.f = fifo.f = 0;
+ stack.b = fifo.b = -1;
+
+ while (trap_Parse_ReadToken(handle, &token))
+ {
+ if (!unmatchedParentheses && token.string[0] == ')')
+ break;
+
+ // Special case to catch negative numbers
+ if (expectingNumber && token.string[0] == '-')
+ {
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
+
+ token.floatvalue = -token.floatvalue;
+ }
+
+ if (token.type == TT_NUMBER)
+ {
+ if (!expectingNumber)
+ return qfalse;
+
+ expectingNumber = !expectingNumber;
+
+ PUSH_VAL(fifo, token.floatvalue);
+ }
+ else
+ {
+ switch (token.string[0])
+ {
+ case '(':
+ unmatchedParentheses++;
+ PUSH_OP(stack, '(');
+ break;
+
+ case ')':
+ unmatchedParentheses--;
+
+ if (unmatchedParentheses < 0)
+ return qfalse;
+
+ while (!EMPTY(stack) && PEEK_STACK_OP(stack) != '(')
+ {
+ POP_STACK(stack);
+ PUSH_OP(fifo, value.u.op);
+ }
+
+ // Pop the '('
+ POP_STACK(stack);
+
+ break;
+
+ case '*':
+ case '/':
+ case '+':
+ case '-':
+ if (expectingNumber)
+ return qfalse;
+
+ expectingNumber = !expectingNumber;
+
+ if (EMPTY(stack))
+ {
+ PUSH_OP(stack, token.string[0]);
+ }
+ else
+ {
+ while (!EMPTY(stack) && OpPrec(token.string[0]) < OpPrec(PEEK_STACK_OP(stack)))
+ {
+ POP_STACK(stack);
+ PUSH_OP(fifo, value.u.op);
+ }
+
+ PUSH_OP(stack, token.string[0]);
+ }
+
+ break;
+
+ default:
+ // Unknown token
+ return qfalse;
+ }
+ }
+ }
+
+ while (!EMPTY(stack))
+ {
+ POP_STACK(stack);
+ PUSH_OP(fifo, value.u.op);
+ }
+
+ while (!EMPTY(fifo))
+ {
+ POP_FIFO(fifo);
+
+ if (value.type == EXPR_VALUE)
+ {
+ PUSH_VAL(stack, value.u.val);
+ }
+ else if (value.type == EXPR_OPERATOR)
+ {
+ char op = value.u.op;
+ float operand1, operand2, result;
+
+ POP_STACK(stack);
+ operand2 = value.u.val;
+ POP_STACK(stack);
+ operand1 = value.u.val;
+
+ switch (op)
+ {
+ case '*':
+ result = operand1 * operand2;
+ break;
+
+ case '/':
+ result = operand1 / operand2;
+ break;
+
+ case '+':
+ result = operand1 + operand2;
+ break;
+
+ case '-':
+ result = operand1 - operand2;
+ break;
+
+ default:
+ Com_Error(ERR_FATAL, "Unknown operator '%c' in postfix string", op);
+ return qfalse;
+ }
+
+ PUSH_VAL(stack, result);
+ }
+ }
+
+ POP_STACK(stack);
+
+ *f = value.u.val;
+
return qtrue;
- } else {
- return qfalse;
- }
+
+#undef FULL
+#undef EMPTY
+#undef PUSH_VAL
+#undef PUSH_OP
+#undef POP_STACK
+#undef PEEK_STACK_OP
+#undef PEEK_STACK_VAL
+#undef POP_FIFO
}
/*
@@ -339,26 +649,37 @@ qboolean Float_Parse(char **p, float *f) {
PC_Float_Parse
=================
*/
-qboolean PC_Float_Parse(int handle, float *f) {
- pc_token_t token;
- int negative = qfalse;
+qboolean PC_Float_Parse(int handle, float *f)
+{
+ pc_token_t token;
+ int negative = qfalse;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- if (token.string[0] == '-') {
if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- negative = qtrue;
- }
- if (token.type != TT_NUMBER) {
- PC_SourceError(handle, "expected float but found %s\n", token.string);
- return qfalse;
- }
- if (negative)
- *f = -token.floatvalue;
- else
- *f = token.floatvalue;
- return qtrue;
+ return qfalse;
+
+ if (token.string[0] == '(')
+ return PC_Expression_Parse(handle, f);
+
+ if (token.string[0] == '-')
+ {
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
+
+ negative = qtrue;
+ }
+
+ if (token.type != TT_NUMBER)
+ {
+ PC_SourceError(handle, "expected float but found %s\n", token.string);
+ return qfalse;
+ }
+
+ if (negative)
+ *f = -token.floatvalue;
+ else
+ *f = token.floatvalue;
+
+ return qtrue;
}
/*
@@ -366,17 +687,20 @@ qboolean PC_Float_Parse(int handle, float *f) {
Color_Parse
=================
*/
-qboolean Color_Parse(char **p, vec4_t *c) {
- int i;
- float f;
+qboolean Color_Parse(char **p, vec4_t *c)
+{
+ int i;
+ float f;
- for (i = 0; i < 4; i++) {
- if (!Float_Parse(p, &f)) {
- return qfalse;
+ for (i = 0; i < 4; i++)
+ {
+ if (!Float_Parse(p, &f))
+ return qfalse;
+
+ (*c)[i] = f;
}
- (*c)[i] = f;
- }
- return qtrue;
+
+ return qtrue;
}
/*
@@ -384,17 +708,20 @@ qboolean Color_Parse(char **p, vec4_t *c) {
PC_Color_Parse
=================
*/
-qboolean PC_Color_Parse(int handle, vec4_t *c) {
- int i;
- float f;
+qboolean PC_Color_Parse(int handle, vec4_t *c)
+{
+ int i;
+ float f;
+
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
+ (*c)[i] = f;
}
- (*c)[i] = f;
- }
- return qtrue;
+
+ return qtrue;
}
/*
@@ -402,16 +729,18 @@ qboolean PC_Color_Parse(int handle, vec4_t *c) {
Int_Parse
=================
*/
-qboolean Int_Parse(char **p, int *i) {
- char *token;
- token = COM_ParseExt(p, qfalse);
+qboolean Int_Parse(char **p, int *i)
+{
+ char *token;
+ token = COM_ParseExt(p, qfalse);
- if (token && token[0] != 0) {
- *i = atoi(token);
- return qtrue;
- } else {
- return qfalse;
- }
+ if (token && token[0] != 0)
+ {
+ *i = atoi(token);
+ return qtrue;
+ }
+ else
+ return qfalse;
}
/*
@@ -419,25 +748,47 @@ qboolean Int_Parse(char **p, int *i) {
PC_Int_Parse
=================
*/
-qboolean PC_Int_Parse(int handle, int *i) {
- pc_token_t token;
- int negative = qfalse;
+qboolean PC_Int_Parse(int handle, int *i)
+{
+ pc_token_t token;
+ int negative = qfalse;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- if (token.string[0] == '-') {
if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- negative = qtrue;
- }
- if (token.type != TT_NUMBER) {
- PC_SourceError(handle, "expected integer but found %s\n", token.string);
- return qfalse;
- }
- *i = token.intvalue;
- if (negative)
- *i = - *i;
- return qtrue;
+ return qfalse;
+
+ if (token.string[0] == '(')
+ {
+ float f;
+
+ if (PC_Expression_Parse(handle, &f))
+ {
+ *i = (int)f;
+ return qtrue;
+ }
+ else
+ return qfalse;
+ }
+
+ if (token.string[0] == '-')
+ {
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
+
+ negative = qtrue;
+ }
+
+ if (token.type != TT_NUMBER)
+ {
+ PC_SourceError(handle, "expected integer but found %s\n", token.string);
+ return qfalse;
+ }
+
+ *i = token.intvalue;
+
+ if (negative)
+ *i = -*i;
+
+ return qtrue;
}
/*
@@ -445,17 +796,21 @@ qboolean PC_Int_Parse(int handle, int *i) {
Rect_Parse
=================
*/
-qboolean Rect_Parse(char **p, rectDef_t *r) {
- if (Float_Parse(p, &r->x)) {
- if (Float_Parse(p, &r->y)) {
- if (Float_Parse(p, &r->w)) {
- if (Float_Parse(p, &r->h)) {
- return qtrue;
+qboolean Rect_Parse(char **p, rectDef_t *r)
+{
+ if (Float_Parse(p, &r->x))
+ {
+ if (Float_Parse(p, &r->y))
+ {
+ if (Float_Parse(p, &r->w))
+ {
+ if (Float_Parse(p, &r->h))
+ return qtrue;
+ }
}
- }
}
- }
- return qfalse;
+
+ return qfalse;
}
/*
@@ -463,17 +818,21 @@ qboolean Rect_Parse(char **p, rectDef_t *r) {
PC_Rect_Parse
=================
*/
-qboolean PC_Rect_Parse(int handle, rectDef_t *r) {
- if (PC_Float_Parse(handle, &r->x)) {
- if (PC_Float_Parse(handle, &r->y)) {
- if (PC_Float_Parse(handle, &r->w)) {
- if (PC_Float_Parse(handle, &r->h)) {
- return qtrue;
+qboolean PC_Rect_Parse(int handle, rectDef_t *r)
+{
+ if (PC_Float_Parse(handle, &r->x))
+ {
+ if (PC_Float_Parse(handle, &r->y))
+ {
+ if (PC_Float_Parse(handle, &r->w))
+ {
+ if (PC_Float_Parse(handle, &r->h))
+ return qtrue;
+ }
}
- }
}
- }
- return qfalse;
+
+ return qfalse;
}
/*
@@ -481,15 +840,19 @@ qboolean PC_Rect_Parse(int handle, rectDef_t *r) {
String_Parse
=================
*/
-qboolean String_Parse(char **p, const char **out) {
- char *token;
+qboolean String_Parse(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;
+ token = COM_ParseExt(p, qfalse);
+
+ if (token && token[0] != 0)
+ {
+ *(out) = String_Alloc(token);
+ return qtrue;
+ }
+
+ return qfalse;
}
/*
@@ -497,13 +860,15 @@ qboolean String_Parse(char **p, const char **out) {
PC_String_Parse
=================
*/
-qboolean PC_String_Parse(int handle, const char **out) {
- pc_token_t token;
+qboolean PC_String_Parse(int handle, const char **out)
+{
+ pc_token_t token;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
+
+ *(out) = String_Alloc(token.string);
- *(out) = String_Alloc(token.string);
return qtrue;
}
@@ -512,37 +877,41 @@ qboolean PC_String_Parse(int handle, const char **out) {
PC_Script_Parse
=================
*/
-qboolean PC_Script_Parse(int handle, const char **out) {
- char script[1024];
- pc_token_t token;
-
- 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
+qboolean PC_Script_Parse(int handle, const char **out)
+{
+ char script[1024];
+ pc_token_t token;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- if (Q_stricmp(token.string, "{") != 0) {
- return qfalse;
- }
+ 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
- while ( 1 ) {
if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
+ return qfalse;
- if (Q_stricmp(token.string, "}") == 0) {
- *out = String_Alloc(script);
- return qtrue;
- }
+ if (Q_stricmp(token.string, "{") != 0)
+ return qfalse;
+
+ while (1)
+ {
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
- if (token.string[1] != '\0') {
- Q_strcat(script, 1024, va("\"%s\"", token.string));
- } else {
- Q_strcat(script, 1024, token.string);
+ if (Q_stricmp(token.string, "}") == 0)
+ {
+ *out = String_Alloc(script);
+ return qtrue;
+ }
+
+ if (token.string[1] != '\0')
+ Q_strcat(script, 1024, va("\"%s\"", token.string));
+ else
+ Q_strcat(script, 1024, token.string);
+
+ Q_strcat(script, 1024, " ");
}
- Q_strcat(script, 1024, " ");
- }
- return qfalse; // bk001105 - LCC missing return value
+
+ return qfalse;
}
// display, window, menu, item code
@@ -555,4123 +924,5279 @@ Init_Display
Initializes the display with a structure to all the drawing routines
==================
*/
-void Init_Display( displayContextDef_t *dc )
-{
- DC = dc;
-}
-
-
+void Init_Display(displayContextDef_t *dc) { DC = dc; }
// type and style painting
-void GradientBar_Paint( rectDef_t *rect, vec4_t color )
+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 );
+ // 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_Init
-Initializes a window structure ( windowDef_t ) with defaults
-
+Initializes a window structure ( Window ) 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;
-}
-
-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;
- }
- }
- }
- }
- }
-}
-
-
-
-void Window_Paint(Window *w, float fadeAmount, float fadeClamp, float fadeCycle) {
- //float bordersize = 0;
- vec4_t color;
- rectDef_t fillRect = w->rect;
-
-
- if (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);
- } else if (w->style == WINDOW_STYLE_TEAMCOLOR) {
- if (DC->getTeamColor) {
- DC->getTeamColor(&color);
- DC->fillRect(fillRect.x, fillRect.y, fillRect.w, fillRect.h, color);
- }
- } else if (w->style == WINDOW_STYLE_CINEMATIC) {
- if (w->cinematic == -1) {
- w->cinematic = DC->playCinematic(w->cinematicName, fillRect.x, fillRect.y, fillRect.w, fillRect.h);
- if (w->cinematic == -1) {
- w->cinematic = -2;
- }
- }
- if (w->cinematic >= 0) {
- DC->runCinematicFrame(w->cinematic);
- DC->drawCinematic(w->cinematic, fillRect.x, fillRect.y, fillRect.w, fillRect.h);
- }
- }
-
- 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);
- }
-
-}
-
-
-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;
+void Window_Init(Window *w)
+{
+ memset(w, 0, sizeof(Window));
+ w->borderSize = 1;
+ w->foreColor[0] = w->foreColor[1] = w->foreColor[2] = w->foreColor[3] = 1.0;
+ w->cinematic = -1;
}
-// FIXME: consolidate this with nearby stuff
-void Item_UpdatePosition(itemDef_t *item) {
- float x, y;
- menuDef_t *menu;
+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 (item == NULL || item->parent == NULL) {
- return;
- }
+ if (*flags & WINDOW_FADINGOUT)
+ {
+ *f -= fadeAmount;
- menu = item->parent;
+ if (bFlags && *f <= 0.0)
+ *flags &= ~(WINDOW_FADINGOUT | WINDOW_VISIBLE);
+ }
+ else
+ {
+ *f += fadeAmount;
- x = menu->window.rect.x;
- y = menu->window.rect.y;
+ if (*f >= clamp)
+ {
+ *f = clamp;
- if (menu->window.border != 0) {
- x += menu->window.borderSize;
- y += menu->window.borderSize;
- }
+ if (bFlags)
+ *flags &= ~WINDOW_FADINGIN;
+ }
+ }
+ }
+ }
+}
- Item_SetScreenCoords(item, x, y);
+static void Window_Paint(Window *w, float fadeAmount, float fadeClamp, float fadeCycle)
+{
+ vec4_t color;
+ rectDef_t fillRect = w->rect;
-}
+ if (DC->getCVarValue("ui_developer"))
+ {
+ 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);
+ }
-// menus
-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);
- }
-}
-
-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);
-}
-
-itemDef_t *Menu_ClearFocus(menuDef_t *menu) {
- int i;
- itemDef_t *ret = NULL;
-
- if (menu == NULL) {
- return NULL;
- }
+ if (w == NULL || (w->style == 0 && w->border == 0))
+ return;
- for (i = 0; i < menu->itemCount; i++) {
- if (menu->items[i]->window.flags & WINDOW_HASFOCUS) {
- ret = menu->items[i];
+ if (w->border != 0)
+ {
+ fillRect.x += w->borderSize;
+ fillRect.y += w->borderSize;
+ fillRect.w -= w->borderSize + 1;
+ fillRect.h -= w->borderSize + 1;
}
- menu->items[i]->window.flags &= ~WINDOW_HASFOCUS;
- if (menu->items[i]->leaveFocus) {
- Item_RunScript(menu->items[i], menu->items[i]->leaveFocus);
+
+ 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);
+ }
+ else if (w->style == WINDOW_STYLE_CINEMATIC)
+ {
+ if (w->cinematic == -1)
+ {
+ w->cinematic = DC->playCinematic(w->cinematicName, fillRect.x, fillRect.y, fillRect.w, fillRect.h);
- return ret;
+ if (w->cinematic == -1)
+ w->cinematic = -2;
+ }
+
+ if (w->cinematic >= 0)
+ {
+ DC->runCinematicFrame(w->cinematic);
+ DC->drawCinematic(w->cinematic, fillRect.x, fillRect.y, fillRect.w, fillRect.h);
+ }
+ }
}
-qboolean IsVisible(int flags) {
- return (flags & WINDOW_VISIBLE && !(flags & WINDOW_FADINGOUT));
+static void Border_Paint(Window *w)
+{
+ if (w == NULL || (w->style == 0 && w->border == 0))
+ return;
+
+ if (w->border == WINDOW_BORDER_FULL)
+ {
+ // full
+ 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);
+ }
}
-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) {
- return qtrue;
+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;
}
- }
- return qfalse;
+
+ 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;
}
-int Menu_ItemsMatchingGroup(menuDef_t *menu, 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)) {
- count++;
+// 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 = 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;
}
- }
- return count;
+
+ Item_SetScreenCoords(item, x, y);
}
-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++;
+// menus
+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;
}
- }
- return NULL;
+
+ for (i = 0; i < menu->itemCount; i++)
+ Item_SetScreenCoords(menu->items[i], x, y);
}
+static void Menu_AspectiseRect(int bias, Rectangle *rect)
+{
+ switch (bias)
+ {
+ case ALIGN_LEFT:
+ rect->x *= DC->aspectScale;
+ rect->w *= DC->aspectScale;
+ break;
+ case ALIGN_CENTER:
+ rect->x = (rect->x * DC->aspectScale) + (320.0f - (320.0f * DC->aspectScale));
+ rect->w *= DC->aspectScale;
+ break;
-void Script_SetColor(itemDef_t *item, 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;
- }
+ case ALIGN_RIGHT:
+ rect->x = 640.0f - ((640.0f - rect->x) * DC->aspectScale);
+ rect->w *= DC->aspectScale;
+ break;
- if (out) {
- for (i = 0; i < 4; i++) {
- if (!Float_Parse(args, &f)) {
- return;
- }
- (*out)[i] = f;
+ default:
+
+ case ASPECT_NONE:
+ break;
+ }
+}
+
+void Menu_AspectCompensate(menuDef_t *menu)
+{
+ int i;
+
+ if (menu->window.aspectBias != ASPECT_NONE)
+ {
+ Menu_AspectiseRect(menu->window.aspectBias, &menu->window.rect);
+
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ menu->items[i]->window.rectClient.x *= DC->aspectScale;
+ menu->items[i]->window.rectClient.w *= DC->aspectScale;
+ menu->items[i]->textalignx *= DC->aspectScale;
}
- }
- }
+ }
+ else
+ {
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ Menu_AspectiseRect(menu->items[i]->window.aspectBias, &menu->items[i]->window.rectClient);
+
+ if (menu->items[i]->window.aspectBias != ASPECT_NONE)
+ menu->items[i]->textalignx *= DC->aspectScale;
+ }
+ }
}
-void Script_SetAsset(itemDef_t *item, 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) {
+void Menu_PostParse(menuDef_t *menu)
+{
+ int i, j;
+
+ 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_AspectCompensate(menu);
+ Menu_UpdatePosition(menu);
+
+ // Push lists to the end of the array as they can potentially be drawn on top
+ // of other elements
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ itemDef_t *item = menu->items[i];
+
+ if (Item_IsListBox(item))
+ {
+ for (j = i; j < menu->itemCount - 1; j++)
+ menu->items[j] = menu->items[j + 1];
+
+ menu->items[j] = item;
+ }
}
- }
}
-void Script_SetBackground(itemDef_t *item, char **args) {
- const char *name;
- // expecting name to set asset to
- if (String_Parse(args, &name)) {
- item->window.background = DC->registerShaderNoMip(name);
- }
+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;
}
+qboolean IsVisible(int flags) { return (flags & WINDOW_VISIBLE && !(flags & WINDOW_FADINGOUT)); }
+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)
+ return qtrue;
+ }
+ return qfalse;
+}
-itemDef_t *Menu_FindItemByName(menuDef_t *menu, const char *p) {
- int i;
- if (menu == NULL || p == NULL) {
- return NULL;
- }
+int Menu_ItemsMatchingGroup(menuDef_t *menu, const char *name)
+{
+ int i;
+ int count = 0;
- for (i = 0; i < menu->itemCount; i++) {
- if (Q_stricmp(p, menu->items[i]->window.name) == 0) {
- return menu->items[i];
+ 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))
+ {
+ count++;
+ }
}
- }
- return NULL;
+ return count;
}
-void Script_SetTeamColor(itemDef_t *item, char **args) {
- if (DC->getTeamColor) {
+itemDef_t *Menu_GetMatchingItemByNumber(menuDef_t *menu, int index, const char *name)
+{
int i;
- vec4_t color;
- DC->getTeamColor(&color);
- for (i = 0; i < 4; i++) {
- item->window.backColor[i] = color[i];
- }
- }
-}
-
-void Script_SetItemColor(itemDef_t *item, 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(item->parent, itemname);
+ 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];
- if (!Color_Parse(args, &color)) {
- return;
+ count++;
+ }
}
- for (j = 0; j < count; j++) {
- item2 = Menu_GetMatchingItemByNumber(item->parent, j, itemname);
- if (item2 != NULL) {
+ return NULL;
+}
+
+void Script_SetColor(itemDef_t *item, 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 = &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 (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))
+ return;
- if (out) {
- for (i = 0; i < 4; i++) {
- (*out)[i] = color[i];
- }
+ (*out)[i] = f;
+ }
}
- }
}
- }
}
+void Script_SetAsset(itemDef_t *item, char **args)
+{
+ const char *name;
+ // expecting name to set asset to
-void Menu_ShowItemByName(menuDef_t *menu, const char *p, qboolean bShow) {
- 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 (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;
+ if (String_Parse(args, &name))
+ {
+ // check for a model
+ if (item->type == ITEM_TYPE_MODEL)
+ {
}
- }
}
- }
}
-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;
- }
+void Script_SetBackground(itemDef_t *item, char **args)
+{
+ const char *name;
+ // expecting name to set asset to
+
+ if (String_Parse(args, &name))
+ item->window.background = DC->registerShaderNoMip(name);
+}
+
+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;
}
-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];
+void Script_SetItemColor(itemDef_t *item, 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(item->parent, itemname);
+
+ if (!Color_Parse(args, &color))
+ return;
+
+ for (j = 0; j < count; j++)
+ {
+ item2 = Menu_GetMatchingItemByNumber(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];
+ }
+ }
+ }
}
- }
- return NULL;
}
-void Menus_ShowByName(const char *p) {
- menuDef_t *menu = Menus_FindByName(p);
- if (menu) {
- Menus_Activate(menu);
- }
+void Menu_ShowItemByName(menuDef_t *menu, const char *p, qboolean bShow)
+{
+ 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 (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;
+ }
+ }
+ }
+ }
}
-void Menus_OpenByName(const char *p) {
- Menus_ActivateByName(p);
+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;
+ }
+ }
+ }
}
-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);
- }
+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;
}
-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);
- }
+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);
+ }
}
-void Menus_CloseAll( void ) {
- int i;
- for (i = 0; i < menuCount; i++) {
- Menu_RunCloseScript(&Menus[i]);
- Menus[i].window.flags &= ~(WINDOW_HASFOCUS | WINDOW_VISIBLE);
- }
+static void Menus_Close(menuDef_t *menu)
+{
+ if (menu != NULL)
+ {
+ Menu_RunCloseScript(menu);
+ menu->window.flags &= ~(WINDOW_VISIBLE | WINDOW_HASFOCUS);
- g_editingField = qfalse;
- g_waitingForKey = qfalse;
+ if (openMenuCount > 0)
+ openMenuCount--;
+
+ if (openMenuCount > 0)
+ Menus_Activate(menuStack[openMenuCount - 1]);
+ }
}
+void Menus_CloseByName(const char *p) { Menus_Close(Menus_FindByName(p)); }
-void Script_Show(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- Menu_ShowItemByName(item->parent, name, qtrue);
- }
+void Menus_CloseAll(void)
+{
+ int i;
+
+ // Close any menus on the stack first
+ if (openMenuCount > 0)
+ {
+ for (i = openMenuCount; i > 0; i--)
+ Menus_Close(menuStack[i - 1]);
+
+ openMenuCount = 0;
+ }
+
+ // Close all other menus
+ for (i = 0; i < menuCount; i++)
+ Menus_Close(&Menus[i]);
+
+ g_editingField = qfalse;
+ g_waitingForKey = qfalse;
+ g_comboBoxItem = NULL;
}
-void Script_Hide(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- Menu_ShowItemByName(item->parent, name, qfalse);
- }
+void Script_Show(itemDef_t *item, char **args)
+{
+ const char *name;
+
+ if (String_Parse(args, &name))
+ Menu_ShowItemByName(item->parent, name, qtrue);
}
-void Script_FadeIn(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- Menu_FadeItemByName(item->parent, name, qfalse);
- }
+void Script_Hide(itemDef_t *item, char **args)
+{
+ const char *name;
+
+ if (String_Parse(args, &name))
+ Menu_ShowItemByName(item->parent, name, qfalse);
}
-void Script_FadeOut(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- Menu_FadeItemByName(item->parent, name, qtrue);
- }
+void Script_FadeIn(itemDef_t *item, char **args)
+{
+ const char *name;
+
+ if (String_Parse(args, &name))
+ Menu_FadeItemByName(item->parent, name, qfalse);
}
+void Script_FadeOut(itemDef_t *item, char **args)
+{
+ const char *name;
+ if (String_Parse(args, &name))
+ Menu_FadeItemByName(item->parent, name, qtrue);
+}
-void Script_Open(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- Menus_OpenByName(name);
- }
+void Script_Open(itemDef_t *item, char **args)
+{
+ const char *name;
+
+ if (String_Parse(args, &name))
+ Menus_ActivateByName(name);
}
-void Script_ConditionalOpen(itemDef_t *item, char **args) {
- const char *cvar;
- const char *name1;
- const char *name2;
- float val;
+void Script_ConditionalOpen(itemDef_t *item, char **args)
+{
+ const char *cvar;
+ const char *name1;
+ const char *name2;
+ float val;
+
+ if (String_Parse(args, &cvar) && String_Parse(args, &name1) && String_Parse(args, &name2))
+ {
+ val = DC->getCVarValue(cvar);
- if ( String_Parse(args, &cvar) && String_Parse(args, &name1) && String_Parse(args, &name2) ) {
- val = DC->getCVarValue( cvar );
- if ( val == 0.f ) {
- Menus_OpenByName(name2);
- } else {
- Menus_OpenByName(name1);
+ if (val == 0.0f)
+ Menus_ActivateByName(name2);
+ else
+ Menus_ActivateByName(name1);
}
- }
}
-void Script_Close(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- Menus_CloseByName(name);
- }
+void Script_Close(itemDef_t *item, char **args)
+{
+ const char *name;
+
+ (void)item;
+
+ if (String_Parse(args, &name))
+ Menus_CloseByName(name);
}
-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 = fabs(rectTo.x - rectFrom.x) / amt;
- item->window.rectEffects2.y = fabs(rectTo.y - rectFrom.y) / amt;
- item->window.rectEffects2.w = fabs(rectTo.w - rectFrom.w) / amt;
- item->window.rectEffects2.h = fabs(rectTo.h - rectFrom.h) / amt;
- Item_UpdatePosition(item);
+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 = fabs(rectTo.x - rectFrom.x) / amt;
+ item->window.rectEffects2.y = fabs(rectTo.y - rectFrom.y) / amt;
+ item->window.rectEffects2.w = fabs(rectTo.w - rectFrom.w) / amt;
+ item->window.rectEffects2.h = fabs(rectTo.h - rectFrom.h) / amt;
+ Item_UpdatePosition(item);
+ }
+ }
+}
+
+void Script_Transition(itemDef_t *item, 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))
+ {
+ Menu_TransitionItemByName(item->parent, name, rectFrom, rectTo, time, amt);
+ }
}
- }
}
+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);
-void Script_Transition(itemDef_t *item, char **args) {
- const char *name;
- rectDef_t rectFrom, rectTo;
- int time;
- float amt;
+ for (i = 0; i < count; i++)
+ {
+ item = Menu_GetMatchingItemByNumber(menu, i, p);
- if (String_Parse(args, &name)) {
- if ( Rect_Parse(args, &rectFrom) && Rect_Parse(args, &rectTo) && Int_Parse(args, &time) && Float_Parse(args, &amt)) {
- Menu_TransitionItemByName(item->parent, name, rectFrom, rectTo, time, amt);
+ 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);
+ }
}
- }
}
+void Script_Orbit(itemDef_t *item, char **args)
+{
+ const char *name;
+ float cx, cy, x, y;
+ int time;
-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);
+ 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))
+ {
+ Menu_OrbitItemByName(item->parent, name, x, y, cx, cy, time);
+ }
}
- }
}
+void Script_SetFocus(itemDef_t *item, char **args)
+{
+ const char *name;
+ itemDef_t *focusItem;
+
+ if (String_Parse(args, &name))
+ {
+ focusItem = Menu_FindItemByName(item->parent, name);
+
+ if (focusItem && !(focusItem->window.flags & WINDOW_DECORATION))
+ {
+ Menu_ClearFocus(item->parent);
+ focusItem->window.flags |= WINDOW_HASFOCUS;
-void Script_Orbit(itemDef_t *item, char **args) {
- const char *name;
- float cx, cy, x, y;
- int time;
+ if (focusItem->onFocus)
+ Item_RunScript(focusItem, focusItem->onFocus);
- 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) ) {
- Menu_OrbitItemByName(item->parent, name, x, y, cx, cy, time);
+ // Edit fields get activated too
+ if (Item_IsEditField(focusItem))
+ {
+ g_editingField = qtrue;
+ g_editItem = focusItem;
+ }
+
+ if (DC->Assets.itemFocusSound)
+ DC->startLocalSound(DC->Assets.itemFocusSound, CHAN_LOCAL_SOUND);
+ }
}
- }
}
+void Script_Reset(itemDef_t *item, char **args)
+{
+ const char *name;
+ itemDef_t *resetItem;
+ if (String_Parse(args, &name))
+ {
+ resetItem = Menu_FindItemByName(item->parent, name);
-void Script_SetFocus(itemDef_t *item, char **args) {
- const char *name;
- itemDef_t *focusItem;
-
- if (String_Parse(args, &name)) {
- focusItem = Menu_FindItemByName(item->parent, name);
- if (focusItem && !(focusItem->window.flags & WINDOW_DECORATION)) {
- Menu_ClearFocus(item->parent);
- focusItem->window.flags |= WINDOW_HASFOCUS;
- if (focusItem->onFocus) {
- Item_RunScript(focusItem, focusItem->onFocus);
- }
- if (focusItem->type == ITEM_TYPE_EDITFIELD || focusItem->type == ITEM_TYPE_SAYFIELD || focusItem->type == ITEM_TYPE_NUMERICFIELD) {
- focusItem->cursorPos = 0;
- g_editingField = qtrue;
- g_editItem = focusItem;
- if (focusItem->type == ITEM_TYPE_SAYFIELD) {
- DC->setOverstrikeMode(qfalse);
+ if (resetItem)
+ {
+ if (Item_IsListBox(resetItem))
+ {
+ resetItem->cursorPos = DC->feederInitialise(resetItem->feederID);
+ Item_ListBox_SetStartPos(resetItem, 0);
+ DC->feederSelection(resetItem->feederID, resetItem->cursorPos);
+ }
}
- }
- if (DC->Assets.itemFocusSound) {
- DC->startLocalSound( DC->Assets.itemFocusSound, CHAN_LOCAL_SOUND );
- }
}
- }
-}
-
-void Script_SetPlayerModel(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- DC->setCVar("team_model", name);
- }
-}
-
-void Script_SetPlayerHead(itemDef_t *item, char **args) {
- const char *name;
- if (String_Parse(args, &name)) {
- DC->setCVar("team_headmodel", name);
- }
}
-void Script_SetCvar(itemDef_t *item, char **args) {
- const char *cvar, *val;
- if (String_Parse(args, &cvar) && String_Parse(args, &val)) {
- DC->setCVar(cvar, val);
- }
+void Script_SetPlayerModel(itemDef_t *item, char **args)
+{
+ const char *name;
+ (void)item;
+
+ if (String_Parse(args, &name))
+ DC->setCVar("model", name);
}
-void Script_Exec(itemDef_t *item, char **args) {
- const char *val;
- if (String_Parse(args, &val)) {
- DC->executeText(EXEC_APPEND, va("%s ; ", val));
- }
+void Script_SetPlayerHead(itemDef_t *item, char **args)
+{
+ const char *name;
+
+ (void)item;
+
+ if (String_Parse(args, &name))
+ DC->setCVar("headmodel", name);
}
-void Script_Play(itemDef_t *item, char **args) {
- const char *val;
- if (String_Parse(args, &val)) {
- DC->startLocalSound(DC->registerSound(val, qfalse), CHAN_LOCAL_SOUND);
- }
+void Script_SetCvar(itemDef_t *item, char **args)
+{
+ const char *cvar, *val;
+
+ (void)item;
+
+ if (String_Parse(args, &cvar) && String_Parse(args, &val))
+ DC->setCVar(cvar, val);
}
-void Script_playLooped(itemDef_t *item, char **args) {
- const char *val;
- if (String_Parse(args, &val)) {
- DC->stopBackgroundTrack();
- DC->startBackgroundTrack(val, val);
- }
-}
-
-
-commandDef_t commandList[] =
+void Script_Exec(itemDef_t *item, char **args)
{
- {"fadein", &Script_FadeIn}, // group/name
- {"fadeout", &Script_FadeOut}, // group/name
- {"show", &Script_Show}, // group/name
- {"hide", &Script_Hide}, // group/name
- {"setcolor", &Script_SetColor}, // works on this
- {"open", &Script_Open}, // menu
- {"conditionalopen", &Script_ConditionalOpen}, // menu
- {"close", &Script_Close}, // menu
- {"setasset", &Script_SetAsset}, // works on this
- {"setbackground", &Script_SetBackground}, // works on this
- {"setitemcolor", &Script_SetItemColor}, // group/name
- {"setteamcolor", &Script_SetTeamColor}, // sets this background color to team color
- {"setfocus", &Script_SetFocus}, // sets this background color to team color
- {"setplayermodel", &Script_SetPlayerModel}, // sets this background color to team color
- {"setplayerhead", &Script_SetPlayerHead}, // sets this background color to team color
- {"transition", &Script_Transition}, // group/name
- {"setcvar", &Script_SetCvar}, // group/name
- {"exec", &Script_Exec}, // group/name
- {"play", &Script_Play}, // group/name
- {"playlooped", &Script_playLooped}, // group/name
- {"orbit", &Script_Orbit} // group/name
-};
+ const char *val;
-int scriptCommandCount = sizeof(commandList) / sizeof(commandDef_t);
-
-
-void Item_RunScript(itemDef_t *item, const char *s) {
- char script[1024], *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;
- }
+ (void)item;
- if (command[0] == ';' && command[1] == '\0') {
- continue;
- }
+ if (String_Parse(args, &val))
+ DC->executeText(EXEC_APPEND, va("%s\n", val));
+}
- 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);
- }
+void Script_Play(itemDef_t *item, char **args)
+{
+ const char *val;
+
+ (void)item;
+
+ if (String_Parse(args, &val))
+ DC->startLocalSound(DC->registerSound(val, qfalse), CHAN_LOCAL_SOUND);
+}
+
+void Script_playLooped(itemDef_t *item, char **args)
+{
+ const char *val;
+
+ (void)item;
+
+ if (String_Parse(args, &val))
+ {
+ DC->stopBackgroundTrack();
+ DC->startBackgroundTrack(val, val);
}
- }
}
+static ID_INLINE float UI_EmoticonHeight(fontInfo_t *font, float scale)
+{
+ return font->glyphs[(int)'['].height * scale * font->glyphScale;
+}
-qboolean Item_EnableShowViaCvar(itemDef_t *item, int flag) {
- char script[1024], *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;
- }
+static ID_INLINE float UI_EmoticonWidth(fontInfo_t *font, float scale)
+{
+ return UI_EmoticonHeight(font, scale) * DC->aspectScale;
+}
- if (val[0] == ';' && val[1] == '\0') {
- continue;
- }
+void UI_EscapeEmoticons(char *dest, const char *src, int destsize)
+{
+ int len;
+ qboolean escaped;
- // 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;
+ for (; *src && destsize > 1; src++, destsize--)
+ {
+ if (UI_Text_IsEmoticon(src, &escaped, &len, NULL, NULL) && !escaped)
+ {
+ *dest++ = '[';
+ destsize--;
}
- }
+ *dest++ = *src;
}
- return (item->cvarFlags & flag) ? qfalse : qtrue;
- }
- return qtrue;
+
+ *dest++ = '\0';
}
+qboolean UI_Text_IsEmoticon(const char *s, qboolean *escaped, int *length, qhandle_t *h, int *width)
+{
+ const char *p = s;
+ char emoticon[MAX_EMOTICON_NAME_LEN];
+ int i;
-// 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; // bk001206: = (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;
- }
+ if (*p != '[')
+ return qfalse;
+ p++;
- // bk001206 - this can be NULL.
- parent = (menuDef_t*)item->parent;
+ if (*p == '[')
+ {
+ *escaped = qtrue;
+ p++;
+ }
+ else
+ *escaped = qfalse;
- // items can be enabled and disabled based on cvars
- if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE)) {
- return qfalse;
- }
+ for (*length = 0; p[*length] != ']'; (*length)++)
+ {
+ if (!p[*length] || *length == MAX_EMOTICON_NAME_LEN - 1)
+ return qfalse;
- if (item->cvarFlags & (CVAR_SHOW | CVAR_HIDE) && !Item_EnableShowViaCvar(item, CVAR_SHOW)) {
- return qfalse;
- }
+ emoticon[*length] = p[*length];
+ }
+ emoticon[*length] = '\0';
- oldFocus = Menu_ClearFocus(item->parent);
+ for (i = 0; i < DC->Assets.emoticonCount; i++)
+ if (!Q_stricmp(DC->Assets.emoticons[i].name, emoticon))
+ break;
- 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;
-}
-
-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;
-}
-
-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;
- }
-}
-
-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);
- }
+ if (i == DC->Assets.emoticonCount)
+ return qfalse;
+
+ if (h)
+ *h = DC->Assets.emoticons[i].shader;
+ if (width)
+ *width = DC->Assets.emoticons[i].width;
+
+ (*length) += 2;
+
+ if (*escaped)
+ (*length)++;
+
+ return qtrue;
}
-float Item_Slider_ThumbPosition(itemDef_t *item) {
- float value, range, x;
- editFieldDef_t *editDef = item->typeData;
+static float UI_Parse_Indent(const char **text)
+{
+ char indentWidth[32];
+ char *indentWidthPtr;
+ const char *p = *text;
+ int numDigits;
+ float pixels;
- if (item->text) {
- x = item->textRect.x + item->textRect.w + 8;
- } else {
- x = item->window.rect.x;
- }
+ while (isdigit(*p) || *p == '.')
+ p++;
- 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;
-}
-
-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;
-}
-
-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;
+ if (*p != INDENT_MARKER)
+ return 0.0f;
+
+ numDigits = (p - *text);
+
+ if (numDigits > sizeof(indentWidth) - 1)
+ return 0.0f;
+
+ strncpy(indentWidth, *text, numDigits);
+
+ indentWidth[numDigits] = '\0';
+ indentWidthPtr = indentWidth;
+
+ if (!Float_Parse(&indentWidthPtr, &pixels))
+ return 0.0f;
+
+ (*text) += (numDigits + 1);
+
+ return pixels;
}
+static ID_INLINE fontInfo_t *UI_FontForScale(float scale)
+{
+ if (scale <= DC->smallFontScale)
+ return &DC->Assets.smallFont;
+ else if (scale >= DC->bigFontScale)
+ return &DC->Assets.bigFont;
+ else
+ return &DC->Assets.textFont;
+}
-void Item_ListBox_MouseEnter(itemDef_t *item, float x, float y)
+float UI_Char_Width(const char **text, float scale)
{
- 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;
- }
- }
- }
-}
-
-void Item_MouseEnter(itemDef_t *item, float x, float y) {
- rectDef_t r;
- if (item) {
- r = item->textRect;
- r.y -= r.h;
- // in the text rect?
+ glyphInfo_t *glyph;
+ fontInfo_t *font;
+ int emoticonLen;
+ qboolean emoticonEscaped;
+ int emoticonWidth;
- // items can be enabled and disabled based on cvars
- if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE)) {
- return;
- }
+ if (text && *text)
+ {
+ if (Q_IsColorString(*text))
+ {
+ *text += 2;
+ return 0.0f;
+ }
- if (item->cvarFlags & (CVAR_SHOW | CVAR_HIDE) && !Item_EnableShowViaCvar(item, CVAR_SHOW)) {
- return;
- }
+ if (**text == INDENT_MARKER)
+ {
+ (*text)++;
+ return 0.0f;
+ }
- 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;
- }
+ font = UI_FontForScale(scale);
- } 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 (UI_Text_IsEmoticon(*text, &emoticonEscaped, &emoticonLen, NULL, &emoticonWidth))
+ {
+ if (emoticonEscaped)
+ (*text)++;
+ else
+ {
+ *text += emoticonLen;
+ return emoticonWidth * UI_EmoticonWidth(font, scale);
+ }
+ }
+
+ (*text)++;
- if (item->type == ITEM_TYPE_LISTBOX) {
- Item_ListBox_MouseEnter(item, x, y);
- }
+ glyph = &font->glyphs[(int)**text];
+ return glyph->xSkip * DC->aspectScale * scale * font->glyphScale;
}
- }
+
+ return 0.0f;
}
-void Item_MouseLeave(itemDef_t *item) {
- if (item) {
- if (item->window.flags & WINDOW_MOUSEOVERTEXT) {
- Item_RunScript(item, item->mouseExitText);
- item->window.flags &= ~WINDOW_MOUSEOVERTEXT;
+float UI_Text_Width(const char *text, float scale)
+{
+ float out;
+ const char *s = text;
+ float indentWidth = 0.0f;
+
+ out = 0.0f;
+
+ if (text)
+ {
+ indentWidth = UI_Parse_Indent(&s);
+
+ while (*s)
+ out += UI_Char_Width(&s, scale);
}
- Item_RunScript(item, item->mouseExit);
- item->window.flags &= ~(WINDOW_LB_RIGHTARROW | WINDOW_LB_LEFTARROW);
- }
+
+ return out + indentWidth;
}
-itemDef_t *Menu_HitTest(menuDef_t *menu, float x, float y) {
- int i;
- for (i = 0; i < menu->itemCount; i++) {
- if (Rect_ContainsPoint(&menu->items[i]->window.rect, x, y)) {
- return menu->items[i];
+float UI_Text_Height(const char *text, float scale)
+{
+ float max;
+ glyphInfo_t *glyph;
+ float useScale;
+ const char *s = text;
+ fontInfo_t *font = UI_FontForScale(scale);
+
+ useScale = scale * font->glyphScale;
+ max = 0;
+
+ if (text)
+ {
+ while (s && *s)
+ {
+ if (Q_IsColorString(s))
+ {
+ s += 2;
+ continue;
+ }
+ else
+ {
+ glyph = &font->glyphs[(int)*s];
+
+ if (max < glyph->height)
+ max = glyph->height;
+
+ s++;
+ }
+ }
}
- }
- return NULL;
+
+ return max * useScale;
}
-void Item_SetMouseOver(itemDef_t *item, qboolean focus) {
- if (item) {
- if (focus) {
- item->window.flags |= WINDOW_MOUSEOVER;
- } else {
- item->window.flags &= ~WINDOW_MOUSEOVER;
- }
- }
+float UI_Text_EmWidth(float scale) { return UI_Text_Width("M", scale); }
+
+float UI_Text_EmHeight(float scale) { return UI_Text_Height("M", scale); }
+
+/*
+================
+UI_AdjustFrom640
+
+Adjusted for resolution and screen aspect ratio
+================
+*/
+void UI_AdjustFrom640(float *x, float *y, float *w, float *h)
+{
+ *x *= DC->xscale;
+ *y *= DC->yscale;
+ *w *= DC->xscale;
+ *h *= DC->yscale;
}
+/*
+================
+UI_SetClipRegion
+=================
+*/
+void UI_SetClipRegion(float x, float y, float w, float h)
+{
+ vec4_t clip;
+
+ UI_AdjustFrom640(&x, &y, &w, &h);
-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;
+ clip[0] = x;
+ clip[1] = y;
+ clip[2] = x + w;
+ clip[3] = y + h;
+
+ trap_R_SetClipRegion(clip);
}
-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;
+/*
+================
+UI_ClearClipRegion
+=================
+*/
+void UI_ClearClipRegion(void) { trap_R_SetClipRegion(NULL); }
- 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;
+static void UI_Text_PaintChar(float x, float y, float scale, glyphInfo_t *glyph, float size)
+{
+ float w, h;
+
+ w = glyph->imageWidth;
+ h = glyph->imageHeight;
+
+ if (size > 0.0f)
+ {
+ float half = size * 0.5f * scale;
+ x -= half;
+ y -= half;
+ w += size;
+ h += size;
+ }
+
+ w *= (DC->aspectScale * scale);
+ h *= scale;
+ y -= (glyph->top * scale);
+ UI_AdjustFrom640(&x, &y, &w, &h);
+
+ DC->drawStretchPic(x, y, w, h, glyph->s, glyph->t, glyph->s2, glyph->t2, glyph->glyph);
+}
+
+static void UI_Text_Paint_Generic(float x, float y, float scale, float gapAdjust, const char *text, vec4_t color,
+ int style, int limit, float *maxX, int cursorPos, char cursor)
+{
+ const char *s = text;
+ int len;
+ int count = 0;
+ vec4_t newColor;
+ fontInfo_t *font = UI_FontForScale(scale);
+ glyphInfo_t *glyph;
+ float useScale;
+ qhandle_t emoticonHandle = 0;
+ float emoticonH, emoticonW;
+ qboolean emoticonEscaped;
+ int emoticonLen = 0;
+ int emoticonWidth;
+ int cursorX = -1;
+
+ if (!text)
+ return;
+
+ useScale = scale * font->glyphScale;
+
+ emoticonH = UI_EmoticonHeight(font, scale);
+ emoticonW = UI_EmoticonWidth(font, scale);
+
+ len = strlen(text);
+ if (limit > 0 && len > limit)
+ len = limit;
+
+ DC->setColor(color);
+ memcpy(&newColor[0], &color[0], sizeof(vec4_t));
+
+ x += UI_Parse_Indent(&s);
+
+ while (s && *s && count < len)
+ {
+ const char *t = s;
+ float charWidth = UI_Char_Width(&t, scale);
+ glyph = &font->glyphs[(int)*s];
+
+ if (maxX && charWidth + x > *maxX)
+ {
+ *maxX = 0;
+ break;
}
- 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;
+
+ if (cursorPos < 0)
+ {
+ if (Q_IsColorString(s))
+ {
+ memcpy(newColor, g_color_table[ColorIndex(*(s + 1))], sizeof(newColor));
+ newColor[3] = color[3];
+ DC->setColor(newColor);
+ s += 2;
+ continue;
+ }
+
+ if (*s == INDENT_MARKER)
+ {
+ s++;
+ continue;
+ }
+
+ if (UI_Text_IsEmoticon(s, &emoticonEscaped, &emoticonLen, &emoticonHandle, &emoticonWidth))
+ {
+ if (emoticonEscaped)
+ s++;
+ else
+ {
+ float yadj = useScale * glyph->top;
+
+ DC->setColor(NULL);
+ DC->drawHandlePic(x, y - yadj, (emoticonW * emoticonWidth), emoticonH, emoticonHandle);
+ DC->setColor(newColor);
+ x += (emoticonW * emoticonWidth) + gapAdjust;
+ s += emoticonLen;
+ count += emoticonWidth;
+ continue;
+ }
+ }
}
- 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;
+
+ if (style == ITEM_TEXTSTYLE_SHADOWED || style == ITEM_TEXTSTYLE_SHADOWEDMORE)
+ {
+ int ofs;
+
+ if (style == ITEM_TEXTSTYLE_SHADOWED)
+ ofs = 1;
+ else
+ ofs = 2;
+ colorBlack[3] = newColor[3];
+ DC->setColor(colorBlack);
+ UI_Text_PaintChar(x + ofs, y + ofs, useScale, glyph, 0.0f);
+ DC->setColor(newColor);
+ colorBlack[3] = 1.0f;
}
- 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;
+ else if (style == ITEM_TEXTSTYLE_NEON)
+ {
+ vec4_t glow;
+
+ memcpy(&glow[0], &newColor[0], sizeof(vec4_t));
+ glow[3] *= 0.2f;
+
+ DC->setColor(glow);
+ UI_Text_PaintChar(x, y, useScale, glyph, 6.0f);
+ UI_Text_PaintChar(x, y, useScale, glyph, 4.0f);
+ DC->setColor(newColor);
+ UI_Text_PaintChar(x, y, useScale, glyph, 2.0f);
+
+ DC->setColor(colorWhite);
}
- 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;
- }
-
- // Scroll wheel
- if (key == K_MWHEELUP) {
- listPtr->startPos--;
- if (listPtr->startPos < 0) {
- listPtr->startPos = 0;
- }
- return qtrue;
+
+ UI_Text_PaintChar(x, y, useScale, glyph, 0.0f);
+
+ if (count == cursorPos)
+ cursorX = x;
+
+ x += (glyph->xSkip * DC->aspectScale * useScale) + gapAdjust;
+ s++;
+ count++;
}
- if (key == K_MWHEELDOWN) {
- listPtr->startPos++;
- if (listPtr->startPos > max) {
- listPtr->startPos = max;
- }
- return qtrue;
+
+ if (maxX)
+ *maxX = x;
+
+ // paint cursor
+ if (cursorPos >= 0)
+ {
+ if (cursorPos == len)
+ cursorX = x;
+
+ if (cursorX >= 0 && !((DC->realTime / BLINK_DIVISOR) & 1))
+ {
+ glyph = &font->glyphs[(int)cursor];
+ UI_Text_PaintChar(cursorX, y, useScale, glyph, 0.0f);
+ }
}
- // Invoke the doubleClick handler when enter is pressed
- if( key == K_ENTER )
+ DC->setColor(NULL);
+}
+
+void UI_Text_Paint_Limit(
+ float *maxX, float x, float y, float scale, vec4_t color, const char *text, float adjust, int limit)
+{
+ UI_Text_Paint_Generic(x, y, scale, adjust, text, color, ITEM_TEXTSTYLE_NORMAL, limit, maxX, -1, 0);
+}
+
+void UI_Text_Paint(float x, float y, float scale, vec4_t color, const char *text, float adjust, int limit, int style)
+{
+ UI_Text_Paint_Generic(x, y, scale, adjust, text, color, style, limit, NULL, -1, 0);
+}
+
+void UI_Text_PaintWithCursor(
+ float x, float y, float scale, vec4_t color, const char *text, int cursorPos, char cursor, int limit, int style)
+{
+ UI_Text_Paint_Generic(x, y, scale, 0.0, text, color, style, limit, NULL, cursorPos, cursor);
+}
+
+commandDef_t commandList[] = {
+ {"close", &Script_Close}, // menu
+ {"conditionalopen", &Script_ConditionalOpen}, // 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
+ {"orbit", &Script_Orbit}, // group/name
+ {"play", &Script_Play}, // group/name
+ {"playlooped", &Script_playLooped}, // group/name
+ {"reset", &Script_Reset}, // resets the state of the item argument
+ {"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
+ {"setplayerhead", &Script_SetPlayerHead}, // sets this background color to team color
+ {"setplayermodel", &Script_SetPlayerModel}, // sets this background color to team color
+ {"show", &Script_Show}, // group/name
+ {"transition", &Script_Transition}, // group/name
+};
+
+static size_t scriptCommandCount = ARRAY_LEN(commandList);
+
+// despite what lcc thinks, we do not get cmdcmp here
+static int commandComp(const void *a, const void *b) { return Q_stricmp((const char *)a, ((commandDef_t *)b)->name); }
+
+void Item_RunScript(itemDef_t *item, const char *s)
+{
+ char script[1024], *p;
+ commandDef_t *cmd;
+ memset(script, 0, sizeof(script));
+
+ if (item && s && s[0])
{
- if( listPtr->doubleClick )
- Item_RunScript( item, listPtr->doubleClick );
+ 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;
- return qtrue;
+ cmd = bsearch(command, commandList, scriptCommandCount, sizeof(commandDef_t), commandComp);
+ if (cmd)
+ cmd->handler(item, &p);
+ else
+ // not in our auto list, pass to handler
+ DC->runScript(&p);
+ }
}
+}
- if ( key == K_HOME || key == K_KP_HOME) {
- // home
- listPtr->startPos = 0;
- return qtrue;
+qboolean Item_EnableShowViaCvar(itemDef_t *item, int flag)
+{
+ char script[1024], *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;
}
- if ( key == K_END || key == K_KP_END) {
- // end
- listPtr->startPos = max;
- return qtrue;
+
+ return qtrue;
+}
+
+// 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;
+ // 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;
}
- 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;
+
+ parent = (menuDef_t *)item->parent;
+
+ // 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(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;
}
- if (listPtr->cursorPos >= listPtr->startPos + viewmax) {
- listPtr->startPos = listPtr->cursorPos - viewmax + 1;
+ else
+ {
+ if (oldFocus)
+ {
+ oldFocus->window.flags |= WINDOW_HASFOCUS;
+
+ if (oldFocus->onFocus)
+ Item_RunScript(oldFocus, oldFocus->onFocus);
+ }
}
- item->cursorPos = listPtr->cursorPos;
- DC->feederSelection(item->special, item->cursorPos);
- }
- else {
- listPtr->startPos -= viewmax;
- if (listPtr->startPos < 0) {
- listPtr->startPos = 0;
+ }
+ 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;
}
- 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;
+
+ return qtrue;
+}
+
+static float Item_ListBox_HeightForNumItems(itemDef_t *item, int numItems)
+{
+ listBoxDef_t *listPtr = item->typeData.list;
+
+ return (listPtr->elementHeight * numItems) + 2.0f;
+}
+
+static int Item_ListBox_NumItemsForItemHeight(itemDef_t *item)
+{
+ listBoxDef_t *listPtr = item->typeData.list;
+
+ if (item->type == ITEM_TYPE_COMBOBOX)
+ return listPtr->dropItems;
+ else
+ return ((item->window.rect.h - 2.0f) / listPtr->elementHeight);
+}
+
+int Item_ListBox_MaxScroll(itemDef_t *item)
+{
+ int total = DC->feederCount(item->feederID);
+ int max = total - Item_ListBox_NumItemsForItemHeight(item);
+
+ if (max < 0)
+ return 0;
+
+ return max;
+}
+
+static float oldComboBoxY;
+static float oldComboBoxH;
+
+static qboolean Item_ComboBox_MaybeCastToListBox(itemDef_t *item)
+{
+ listBoxDef_t *listPtr = item->typeData.list;
+ qboolean cast = g_comboBoxItem != NULL && (item->type == ITEM_TYPE_COMBOBOX);
+
+ if (cast)
+ {
+ oldComboBoxY = item->window.rect.y;
+ oldComboBoxH = item->window.rect.h;
+
+ item->window.rect.y += item->window.rect.h;
+ item->window.rect.h = Item_ListBox_HeightForNumItems(item, listPtr->dropItems);
+ item->type = ITEM_TYPE_LISTBOX;
+ }
+
+ return cast;
+}
+
+static void Item_ComboBox_MaybeUnCastFromListBox(itemDef_t *item, qboolean unCast)
+{
+ if (unCast)
+ {
+ item->window.rect.y = oldComboBoxY;
+ item->window.rect.h = oldComboBoxH;
+ item->type = ITEM_TYPE_COMBOBOX;
+ }
+}
+
+static void Item_ListBox_SetStartPos(itemDef_t *item, int startPos)
+{
+ listBoxDef_t *listPtr = item->typeData.list;
+ int total = DC->feederCount(item->feederID);
+ int max = Item_ListBox_MaxScroll(item);
+
+ if (startPos < 0)
+ listPtr->startPos = 0;
+ else if (startPos > max)
+ listPtr->startPos = max;
+ else
+ listPtr->startPos = startPos;
+
+ listPtr->endPos = listPtr->startPos + MIN((total - listPtr->startPos), Item_ListBox_NumItemsForItemHeight(item));
+}
+
+float Item_ListBox_ThumbPosition(itemDef_t *item)
+{
+ float max, pos, size;
+ float startPos = (float)item->typeData.list->startPos;
+
+ max = Item_ListBox_MaxScroll(item);
+ size = SCROLLBAR_SLIDER_HEIGHT(item);
+
+ if (max > 0.0f)
+ pos = (size - SCROLLBAR_ARROW_HEIGHT) / max;
+ else
+ pos = 0.0f;
+
+ pos *= startPos;
+
+ return SCROLLBAR_SLIDER_Y(item) + pos;
+}
+
+float Item_ListBox_ThumbDrawPosition(itemDef_t *item)
+{
+ if (itemCapture == item)
+ {
+ float min = SCROLLBAR_SLIDER_Y(item);
+ float max = min + SCROLLBAR_SLIDER_HEIGHT(item) - SCROLLBAR_ARROW_HEIGHT;
+ float halfThumbSize = SCROLLBAR_ARROW_HEIGHT / 2.0f;
+
+ if (DC->cursory >= min + halfThumbSize && DC->cursory <= max + halfThumbSize)
+ return DC->cursory - halfThumbSize;
+ }
+
+ return Item_ListBox_ThumbPosition(item);
+}
+
+float Item_Slider_ThumbPosition(itemDef_t *item)
+{
+ float value, range, x;
+ editFieldDef_t *editDef = item->typeData.edit;
+
+ if (item->text)
+ x = item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET;
+ 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 *= SLIDER_WIDTH;
+ x += value;
+
+ return x;
+}
+
+static float Item_Slider_VScale(itemDef_t *item)
+{
+ if (SLIDER_THUMB_HEIGHT > item->window.rect.h)
+ return item->window.rect.h / SLIDER_THUMB_HEIGHT;
+ else
+ return 1.0f;
+}
+
+int Item_Slider_OverSlider(itemDef_t *item, float x, float y)
+{
+ rectDef_t r;
+ float vScale = Item_Slider_VScale(item);
+
+ r.x = Item_Slider_ThumbPosition(item) - (SLIDER_THUMB_WIDTH / 2);
+ r.y = item->textRect.y - item->textRect.h + ((item->textRect.h - (SLIDER_THUMB_HEIGHT * vScale)) / 2.0f);
+ r.w = SLIDER_THUMB_WIDTH;
+ r.h = SLIDER_THUMB_HEIGHT * vScale;
+
+ if (Rect_ContainsPoint(&r, x, y))
+ return WINDOW_LB_THUMB;
+
+ return 0;
+}
+
+int Item_ListBox_OverLB(itemDef_t *item, float x, float y)
+{
+ rectDef_t r;
+ int thumbstart;
+
+ r.x = SCROLLBAR_SLIDER_X(item);
+ r.y = SCROLLBAR_Y(item);
+ r.w = SCROLLBAR_ARROW_WIDTH;
+ r.h = SCROLLBAR_ARROW_HEIGHT;
+
+ if (Rect_ContainsPoint(&r, x, y))
+ return WINDOW_LB_UPARROW;
+
+ r.y = SCROLLBAR_SLIDER_Y(item) + SCROLLBAR_SLIDER_HEIGHT(item);
+
+ if (Rect_ContainsPoint(&r, x, y))
+ return WINDOW_LB_DOWNARROW;
+
+ thumbstart = Item_ListBox_ThumbPosition(item);
+ r.y = thumbstart;
+
+ if (Rect_ContainsPoint(&r, x, y))
+ return WINDOW_LB_THUMB;
+
+ r.y = SCROLLBAR_SLIDER_Y(item);
+ r.h = thumbstart - r.y;
+
+ if (Rect_ContainsPoint(&r, x, y))
+ return WINDOW_LB_PGUP;
+
+ r.y = thumbstart + SCROLLBAR_ARROW_HEIGHT;
+ r.h = (SCROLLBAR_SLIDER_Y(item) + SCROLLBAR_SLIDER_HEIGHT(item)) - r.y;
+
+ if (Rect_ContainsPoint(&r, x, y))
+ return WINDOW_LB_PGDN;
+
+ return 0;
+}
+
+void Item_ListBox_MouseEnter(itemDef_t *item, float x, float y)
+{
+ rectDef_t r;
+ listBoxDef_t *listPtr = item->typeData.list;
+ int listBoxFlags = (WINDOW_LB_UPARROW | WINDOW_LB_DOWNARROW | WINDOW_LB_THUMB | WINDOW_LB_PGUP | WINDOW_LB_PGDN);
+ int total = DC->feederCount(item->feederID);
+
+ item->window.flags &= ~listBoxFlags;
+ item->window.flags |= Item_ListBox_OverLB(item, x, y);
+
+ if (!(item->window.flags & listBoxFlags))
+ {
+ r.x = SCROLLBAR_X(item);
+ r.y = SCROLLBAR_Y(item);
+ r.w = SCROLLBAR_W(item);
+ r.h = listPtr->elementHeight * MIN(Item_ListBox_NumItemsForItemHeight(item), total);
+
+ if (Rect_ContainsPoint(&r, x, y))
+ {
+ listPtr->cursorPos = (int)((y - r.y) / listPtr->elementHeight) + listPtr->startPos;
+
+ if (listPtr->cursorPos >= listPtr->endPos)
+ listPtr->cursorPos = listPtr->endPos - 1;
}
- if (listPtr->cursorPos >= count) {
- listPtr->cursorPos = count-1;
+ else
+ listPtr->cursorPos = -1;
+ }
+}
+
+void Item_MouseEnter(itemDef_t *item, float x, float y)
+{
+ rectDef_t r;
+
+ if (item)
+ {
+ r = item->textRect;
+ r.y -= r.h;
+ // 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;
+ }
}
- if (listPtr->cursorPos >= listPtr->startPos + viewmax) {
- listPtr->startPos = listPtr->cursorPos - viewmax + 1;
+ 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->cursorPos = listPtr->cursorPos;
- DC->feederSelection(item->special, item->cursorPos);
- }
- else {
- listPtr->startPos += viewmax;
- if (listPtr->startPos > max) {
- listPtr->startPos = max;
+ }
+}
+
+void Item_MouseLeave(itemDef_t *item)
+{
+ if (item)
+ {
+ if (item->window.flags & WINDOW_MOUSEOVERTEXT)
+ {
+ Item_RunScript(item, item->mouseExitText);
+ item->window.flags &= ~WINDOW_MOUSEOVERTEXT;
}
- }
- return qtrue;
+
+ Item_RunScript(item, item->mouseExit);
+ item->window.flags &= ~(WINDOW_LB_DOWNARROW | WINDOW_LB_UPARROW);
}
- }
- return qfalse;
}
-qboolean Item_YesNo_HandleKey(itemDef_t *item, int key) {
+itemDef_t *Menu_HitTest(menuDef_t *menu, float x, float y)
+{
+ int i;
- 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;
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ if (Rect_ContainsPoint(&menu->items[i]->window.rect, x, y))
+ return menu->items[i];
}
- }
- return qfalse;
+ return NULL;
+}
+void Item_SetMouseOver(itemDef_t *item, qboolean focus)
+{
+ if (item)
+ {
+ if (focus)
+ item->window.flags |= WINDOW_MOUSEOVER;
+ else
+ item->window.flags &= ~WINDOW_MOUSEOVER;
+ }
}
-int Item_Multi_CountSettings(itemDef_t *item) {
- multiDef_t *multiPtr = (multiDef_t*)item->typeData;
- if (multiPtr == NULL) {
- return 0;
- }
- return multiPtr->count;
-}
-
-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;
-}
-
-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) {
- 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 multiPtr->cvarList[i];
- }
- } else {
- if (multiPtr->cvarValue[i] == value) {
- return multiPtr->cvarList[i];
- }
- }
- }
- }
- return "";
-}
-
-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 (key == K_MOUSE1 || key == K_ENTER || key == K_MOUSE2 || key == K_MOUSE3) {
- 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 ));
- }
+qboolean Item_OwnerDraw_HandleKey(itemDef_t *item, int key)
+{
+ if (item && DC->ownerDrawHandleKey)
+ return DC->ownerDrawHandleKey(item->window.ownerDraw, key);
+
+ return qfalse;
+}
+
+qboolean Item_ListBox_HandleKey(itemDef_t *item, int key, qboolean down, qboolean force)
+{
+ listBoxDef_t *listPtr = item->typeData.list;
+ int count = DC->feederCount(item->feederID);
+ int viewmax;
+
+ if (force ||
+ (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory) && item->window.flags & WINDOW_HASFOCUS))
+ {
+ viewmax = Item_ListBox_NumItemsForItemHeight(item);
+
+ switch (key)
+ {
+ case K_MOUSE1:
+ case K_MOUSE2:
+ if (item->window.flags & WINDOW_LB_UPARROW)
+ Item_ListBox_SetStartPos(item, listPtr->startPos - 1);
+ else if (item->window.flags & WINDOW_LB_DOWNARROW)
+ Item_ListBox_SetStartPos(item, listPtr->startPos + 1);
+ else if (item->window.flags & WINDOW_LB_PGUP)
+ Item_ListBox_SetStartPos(item, listPtr->startPos - viewmax);
+ else if (item->window.flags & WINDOW_LB_PGDN)
+ Item_ListBox_SetStartPos(item, listPtr->startPos + viewmax);
+ else if (item->window.flags & WINDOW_LB_THUMB)
+ break; // Handled by capture function
+ else
+ {
+ // Select an item
+ qboolean runDoubleClick = qfalse;
+
+ // Mouse isn't over an item
+ if (listPtr->cursorPos < 0)
+ break;
+
+ if (item->cursorPos != listPtr->cursorPos)
+ {
+ item->cursorPos = listPtr->cursorPos;
+ DC->feederSelection(item->feederID, item->cursorPos);
+ }
+
+ runDoubleClick = DC->realTime < lastListBoxClickTime && listPtr->doubleClick;
+ lastListBoxClickTime = DC->realTime + DOUBLE_CLICK_DELAY;
+
+ // Made a selection, so close combobox
+ if (g_comboBoxItem != NULL)
+ {
+ if (listPtr->doubleClick)
+ runDoubleClick = qtrue;
+
+ g_comboBoxItem = NULL;
+ }
+
+ if (runDoubleClick)
+ Item_RunScript(item, listPtr->doubleClick);
+ }
+
+ break;
+
+ case K_MWHEELUP:
+ Item_ListBox_SetStartPos(item, listPtr->startPos - 1);
+ break;
+
+ case K_MWHEELDOWN:
+ Item_ListBox_SetStartPos(item, listPtr->startPos + 1);
+ break;
+
+ case K_ENTER:
+ // Invoke the doubleClick handler when enter is pressed
+ if (listPtr->doubleClick)
+ Item_RunScript(item, listPtr->doubleClick);
+
+ break;
+
+ case K_PGUP:
+ case K_KP_PGUP:
+ if (!listPtr->notselectable)
+ {
+ listPtr->cursorPos -= viewmax;
+
+ if (listPtr->cursorPos < 0)
+ listPtr->cursorPos = 0;
+
+ if (listPtr->cursorPos < listPtr->startPos)
+ Item_ListBox_SetStartPos(item, listPtr->cursorPos);
+
+ if (listPtr->cursorPos >= listPtr->startPos + viewmax)
+ Item_ListBox_SetStartPos(item, listPtr->cursorPos - viewmax + 1);
+
+ item->cursorPos = listPtr->cursorPos;
+ DC->feederSelection(item->feederID, item->cursorPos);
+ }
+ else
+ Item_ListBox_SetStartPos(item, listPtr->startPos - viewmax);
+
+ break;
+
+ case K_PGDN:
+ case K_KP_PGDN:
+ if (!listPtr->notselectable)
+ {
+ listPtr->cursorPos += viewmax;
+
+ if (listPtr->cursorPos < listPtr->startPos)
+ Item_ListBox_SetStartPos(item, listPtr->cursorPos);
+
+ if (listPtr->cursorPos >= count)
+ listPtr->cursorPos = count - 1;
+
+ if (listPtr->cursorPos >= listPtr->startPos + viewmax)
+ Item_ListBox_SetStartPos(item, listPtr->cursorPos - viewmax + 1);
+
+ item->cursorPos = listPtr->cursorPos;
+ DC->feederSelection(item->feederID, item->cursorPos);
+ }
+ else
+ Item_ListBox_SetStartPos(item, listPtr->startPos + viewmax);
+
+ break;
+
+ default:
+ // Not handled
+ return qfalse;
}
+
return qtrue;
- }
}
- }
- return qfalse;
+
+ return qfalse;
}
-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;
+qboolean Item_ComboBox_HandleKey(itemDef_t *item, int key, qboolean down, qboolean force)
+{
+ if (g_comboBoxItem != NULL)
+ {
+ qboolean result;
- if (item->cvar) {
+ qboolean cast = Item_ComboBox_MaybeCastToListBox(item);
+ result = Item_ListBox_HandleKey(item, key, down, force);
+ Item_ComboBox_MaybeUnCastFromListBox(item, cast);
- 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 (!result)
+ g_comboBoxItem = NULL;
+ return result;
+ }
+ else
+ {
+ if (force ||
+ (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory) && item->window.flags & WINDOW_HASFOCUS))
+ {
+ if (key == K_MOUSE1 || key == K_MOUSE2)
+ {
+ g_comboBoxItem = item;
- 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--;
- }
+ return qtrue;
+ }
}
- DC->setCVar(item->cvar, buff);
- return qtrue;
- }
+ }
+ return qfalse;
+}
- //
- // ignore any non printable chars
- //
- if ( key < 32 || !item->cvar) {
- return qtrue;
+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;
+}
+
+int Item_Multi_CountSettings(itemDef_t *item)
+{
+ if (item->typeData.multi == NULL)
+ return 0;
+
+ return item->typeData.multi->count;
+}
+
+int Item_Multi_FindCvarByValue(itemDef_t *item)
+{
+ char buff[1024];
+ float value = 0;
+ int i;
+ multiDef_t *multiPtr = item->typeData.multi;
+
+ if (multiPtr)
+ {
+ if (multiPtr->strDef)
+ DC->getCVarString(item->cvar, buff, sizeof(buff));
+ else
+ value = DC->getCVarValue(item->cvar);
- if (item->type == ITEM_TYPE_NUMERICFIELD) {
- if (key < '0' || key > '9') {
- return qfalse;
+ 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;
+ }
}
- }
+ }
- if (!DC->getOverstrikeMode()) {
- if (( len == MAX_EDITFIELD - 1 ) || (editPtr->maxChars && len >= editPtr->maxChars)) {
- return qtrue;
+ return 0;
+}
+
+const char *Item_Multi_Setting(itemDef_t *item)
+{
+ char buff[1024];
+ float value = 0;
+ int i;
+ multiDef_t *multiPtr = item->typeData.multi;
+
+ 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 multiPtr->cvarList[i];
+ }
+ else
+ {
+ if (multiPtr->cvarValue[i] == value)
+ return multiPtr->cvarList[i];
+ }
}
- memmove( &buff[item->cursorPos + 1], &buff[item->cursorPos], len + 1 - item->cursorPos );
- } else {
- if (editPtr->maxChars && item->cursorPos >= editPtr->maxChars) {
- return qtrue;
+ }
+
+ return "";
+}
+
+qboolean Item_Cycle_HandleKey(itemDef_t *item, int key)
+{
+ cycleDef_t *cyclePtr = item->typeData.cycle;
+ qboolean mouseOver = Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory);
+ int count = DC->feederCount(item->feederID);
+
+ if (cyclePtr)
+ {
+ if (item->window.flags & WINDOW_HASFOCUS)
+ {
+ if ((mouseOver && key == K_MOUSE1) || key == K_ENTER || key == K_RIGHTARROW || key == K_DOWNARROW)
+ {
+ if (count > 0)
+ cyclePtr->cursorPos = (cyclePtr->cursorPos + 1) % count;
+
+ DC->feederSelection(item->feederID, cyclePtr->cursorPos);
+
+ return qtrue;
+ }
+ else if ((mouseOver && key == K_MOUSE2) || key == K_LEFTARROW || key == K_UPARROW)
+ {
+ if (count > 0)
+ cyclePtr->cursorPos = (count + cyclePtr->cursorPos - 1) % count;
+
+ DC->feederSelection(item->feederID, cyclePtr->cursorPos);
+
+ return qtrue;
+ }
}
- }
+ }
+
+ return qfalse;
+}
- buff[item->cursorPos] = key;
+qboolean Item_Multi_HandleKey(itemDef_t *item, int key)
+{
+ qboolean mouseOver = Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory);
+ int max = Item_Multi_CountSettings(item);
+ qboolean changed = qfalse;
- DC->setCVar(item->cvar, buff);
+ if (item->typeData.multi)
+ {
+ if (item->window.flags & WINDOW_HASFOCUS && item->cvar && max > 0)
+ {
+ int current;
+
+ if ((mouseOver && key == K_MOUSE1) || key == K_ENTER || key == K_RIGHTARROW || key == K_DOWNARROW)
+ {
+ current = (Item_Multi_FindCvarByValue(item) + 1) % max;
+ changed = qtrue;
+ }
+ else if ((mouseOver && key == K_MOUSE2) || key == K_LEFTARROW || key == K_UPARROW)
+ {
+ current = (Item_Multi_FindCvarByValue(item) + max - 1) % max;
+ changed = qtrue;
+ }
- if (item->cursorPos < len + 1) {
- item->cursorPos++;
- if (editPtr->maxPaintChars && item->cursorPos > editPtr->maxPaintChars) {
- editPtr->paintOffset++;
+ if (changed)
+ {
+ if (item->typeData.multi->strDef)
+ DC->setCVar(item->cvar, item->typeData.multi->cvarStr[current]);
+ else
+ {
+ float value = item->typeData.multi->cvarValue[current];
+
+ if (((float)((int)value)) == value)
+ DC->setCVar(item->cvar, va("%i", (int)value));
+ else
+ DC->setCVar(item->cvar, va("%f", value));
+ }
+
+ return qtrue;
+ }
}
- }
+ }
+
+ return qfalse;
+}
- } else {
+#define MIN_FIELD_WIDTH 10
+#define EDIT_CURSOR_WIDTH 10
+
+static void Item_TextField_CalcPaintOffset(itemDef_t *item, char *buff)
+{
+ editFieldDef_t *editPtr = item->typeData.edit;
+
+ if (item->cursorPos < editPtr->paintOffset)
+ editPtr->paintOffset = item->cursorPos;
+ else
+ {
+ // If there is a maximum field width
- 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);
+ if (editPtr->maxFieldWidth > 0)
+ {
+ // If the cursor is at the end of the string, maximise the amount of the
+ // string that's visible
+
+ if (buff[item->cursorPos + 1] == '\0')
+ {
+ while (UI_Text_Width(&buff[editPtr->paintOffset], item->textscale) <=
+ (editPtr->maxFieldWidth - EDIT_CURSOR_WIDTH) &&
+ editPtr->paintOffset > 0)
+ editPtr->paintOffset--;
+ }
+
+ buff[item->cursorPos + 1] = '\0';
+
+ // Shift paintOffset so that the cursor is visible
+
+ while (UI_Text_Width(&buff[editPtr->paintOffset], item->textscale) >
+ (editPtr->maxFieldWidth - EDIT_CURSOR_WIDTH))
+ editPtr->paintOffset++;
}
- 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;
+qboolean Item_TextField_HandleKey(itemDef_t *item, int key)
+{
+ char buff[1024];
+ int len;
+ itemDef_t *newItem = NULL;
+ editFieldDef_t *editPtr = item->typeData.edit;
+ qboolean releaseFocus = qtrue;
+
+ if (item->cvar)
+ {
+ Com_Memset(buff, 0, sizeof(buff));
+ DC->getCVarString(item->cvar, buff, sizeof(buff));
+ len = strlen(buff);
+
+ if (len < item->cursorPos)
+ item->cursorPos = len;
+
+ 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--;
+ }
+
+ DC->setCVar(item->cvar, buff);
+ }
+ else if (key < 32 || !item->cvar)
+ {
+ // Ignore any non printable chars
+ releaseFocus = qfalse;
+ goto exit;
+ }
+ else if (item->type == ITEM_TYPE_NUMERICFIELD && (key < '0' || key > '9'))
+ {
+ // Ignore non-numeric characters
+ releaseFocus = qfalse;
+ goto exit;
+ }
+ else
+ {
+ if (!DC->getOverstrikeMode())
+ {
+ if ((len == MAX_EDITFIELD - 1) || (editPtr->maxChars && len >= editPtr->maxChars))
+ {
+ // Reached maximum field length
+ releaseFocus = qfalse;
+ goto exit;
+ }
+
+ memmove(&buff[item->cursorPos + 1], &buff[item->cursorPos], len + 1 - item->cursorPos);
+ }
+ else
+ {
+ // Reached maximum field length
+ if (editPtr->maxChars && item->cursorPos >= editPtr->maxChars)
+ {
+ releaseFocus = qfalse;
+ goto exit;
+ }
+ }
+
+ buff[item->cursorPos] = key;
+
+ DC->setCVar(item->cvar, buff);
+
+ if (item->cursorPos < len + 1)
+ item->cursorPos++;
+ }
}
- if (item->cursorPos < len) {
- item->cursorPos++;
+ else
+ {
+ switch (key)
+ {
+ case K_DEL:
+ case K_KP_DEL:
+ if (item->cursorPos < len)
+ {
+ memmove(buff + item->cursorPos, buff + item->cursorPos + 1, len - item->cursorPos);
+ DC->setCVar(item->cvar, buff);
+ }
+
+ break;
+
+ case K_RIGHTARROW:
+ case K_KP_RIGHTARROW:
+ if (item->cursorPos < len)
+ item->cursorPos++;
+
+ break;
+
+ case K_LEFTARROW:
+ case K_KP_LEFTARROW:
+ if (item->cursorPos > 0)
+ item->cursorPos--;
+
+ break;
+
+ case K_HOME:
+ case K_KP_HOME:
+ item->cursorPos = 0;
+
+ break;
+
+ case K_END:
+ case K_KP_END:
+ item->cursorPos = len;
+
+ break;
+
+ case K_INS:
+ case K_KP_INS:
+ DC->setOverstrikeMode(!DC->getOverstrikeMode());
+
+ break;
+
+ case K_TAB:
+ case K_DOWNARROW:
+ case K_KP_DOWNARROW:
+ case K_UPARROW:
+ case K_KP_UPARROW:
+ // Ignore these keys from the say field
+ if (item->type == ITEM_TYPE_SAYFIELD)
+ break;
+
+ newItem = Menu_SetNextCursorItem(item->parent);
+
+ if (newItem && Item_IsEditField(newItem))
+ {
+ g_editItem = newItem;
+ }
+ else
+ {
+ releaseFocus = qtrue;
+ goto exit;
+ }
+
+ break;
+
+ case K_MOUSE1:
+ case K_MOUSE2:
+ case K_MOUSE3:
+ case K_MOUSE4:
+ // Ignore these buttons from the say field
+ if (item->type == ITEM_TYPE_SAYFIELD)
+ break;
+ // FALLTHROUGH
+ case K_ENTER:
+ case K_KP_ENTER:
+ case K_ESCAPE:
+ releaseFocus = qtrue;
+ goto exit;
+
+ default:
+ break;
+ }
}
- return qtrue;
- }
- if ( key == K_LEFTARROW || key == K_KP_LEFTARROW )
- {
- if ( item->cursorPos > 0 ) {
- item->cursorPos--;
+ releaseFocus = qfalse;
+ }
+
+exit:
+ Item_TextField_CalcPaintOffset(item, buff);
+
+ return !releaseFocus;
+}
+
+static void _Scroll_ListBox_AutoFunc(scrollInfo_t *si)
+{
+ 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;
+ }
+}
+
+static void Scroll_ListBox_AutoFunc(void *p)
+{
+ scrollInfo_t *si = (scrollInfo_t *)p;
+
+ qboolean cast = Item_ComboBox_MaybeCastToListBox(si->item);
+ _Scroll_ListBox_AutoFunc(si);
+ Item_ComboBox_MaybeUnCastFromListBox(si->item, cast);
+}
+
+static void _Scroll_ListBox_ThumbFunc(scrollInfo_t *si)
+{
+ rectDef_t r;
+ int pos, max;
+
+ if (DC->cursory != si->yStart)
+ {
+ r.x = si->item->window.rect.x + si->item->window.rect.w - SCROLLBAR_ARROW_WIDTH - 1;
+ r.y = si->item->window.rect.y + SCROLLBAR_ARROW_HEIGHT + 1;
+ r.w = SCROLLBAR_ARROW_WIDTH;
+ r.h = si->item->window.rect.h - (SCROLLBAR_ARROW_HEIGHT * 2) - 2;
+ max = Item_ListBox_MaxScroll(si->item);
+ //
+ pos = (DC->cursory - r.y - SCROLLBAR_ARROW_HEIGHT / 2) * max / (r.h - SCROLLBAR_ARROW_HEIGHT);
+
+ if (pos < 0)
+ pos = 0;
+ else if (pos > max)
+ pos = max;
+
+ Item_ListBox_SetStartPos(si->item, 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;
+ }
+}
+
+static void Scroll_ListBox_ThumbFunc(void *p)
+{
+ scrollInfo_t *si = (scrollInfo_t *)p;
+
+ qboolean cast = Item_ComboBox_MaybeCastToListBox(si->item);
+ _Scroll_ListBox_ThumbFunc(si);
+ Item_ComboBox_MaybeUnCastFromListBox(si->item, cast);
+}
+
+static void Scroll_Slider_ThumbFunc(void *p)
+{
+ float x, value, cursorx;
+ scrollInfo_t *si = (scrollInfo_t *)p;
+
+ if (si->item->text)
+ x = si->item->textRect.x + si->item->textRect.w + ITEM_VALUE_OFFSET;
+ 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 *= si->item->typeData.edit->maxVal - si->item->typeData.edit->minVal;
+ value += si->item->typeData.edit->minVal;
+ DC->setCVar(si->item->cvar, va("%f", value));
+}
+
+void Item_StartCapture(itemDef_t *item, int key)
+{
+ int flags;
+
+ // Don't allow captureFunc to be overridden
+
+ if (captureFunc != voidFunction)
+ return;
+
+ switch (item->type)
+ {
+ case ITEM_TYPE_LISTBOX:
+ case ITEM_TYPE_COMBOBOX:
+ {
+ qboolean cast = Item_ComboBox_MaybeCastToListBox(item);
+ flags = Item_ListBox_OverLB(item, DC->cursorx, DC->cursory);
+ Item_ComboBox_MaybeUnCastFromListBox(item, cast);
+
+ if (flags & (WINDOW_LB_UPARROW | WINDOW_LB_DOWNARROW))
+ {
+ 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_UPARROW) ? qtrue : qfalse;
+ scrollInfo.item = item;
+ UI_InstallCaptureFunc(Scroll_ListBox_AutoFunc, &scrollInfo, 0);
+ itemCapture = item;
+ }
+ else if (flags & WINDOW_LB_THUMB)
+ {
+ scrollInfo.scrollKey = key;
+ scrollInfo.item = item;
+ scrollInfo.xStart = DC->cursorx;
+ scrollInfo.yStart = DC->cursory;
+ UI_InstallCaptureFunc(Scroll_ListBox_ThumbFunc, &scrollInfo, 0);
+ itemCapture = item;
+ }
+
+ break;
}
- if (item->cursorPos < editPtr->paintOffset) {
- editPtr->paintOffset--;
+
+ 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;
+ UI_InstallCaptureFunc(Scroll_Slider_ThumbFunc, &scrollInfo, 0);
+ itemCapture = item;
+ }
+
+ break;
}
- return qtrue;
- }
+ }
+}
- if ( key == K_HOME || key == K_KP_HOME) {// || ( tolower(key) == 'a' && trap_Key_IsDown( K_CTRL ) ) ) {
- item->cursorPos = 0;
- editPtr->paintOffset = 0;
- return qtrue;
- }
+void Item_StopCapture(itemDef_t *item) {}
+
+qboolean Item_Slider_HandleKey(itemDef_t *item, int key, qboolean down)
+{
+ float x, value, width;
- if ( key == K_END || key == K_KP_END) {// ( tolower(key) == 'e' && trap_Key_IsDown( K_CTRL ) ) ) {
- item->cursorPos = len;
- if(item->cursorPos > editPtr->maxPaintChars) {
- editPtr->paintOffset = len - editPtr->maxPaintChars;
+ if (item->window.flags & WINDOW_HASFOCUS && item->cvar &&
+ Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory))
+ {
+ if (item->typeData.edit && (key == K_ENTER || key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3))
+ {
+ rectDef_t testRect;
+ width = SLIDER_WIDTH;
+
+ if (item->text)
+ x = item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET;
+ else
+ x = item->window.rect.x;
+
+ testRect = item->window.rect;
+ value = (float)SLIDER_THUMB_WIDTH / 2;
+ testRect.x = x - value;
+ testRect.w = SLIDER_WIDTH + value;
+
+ if (Rect_ContainsPoint(&testRect, DC->cursorx, DC->cursory))
+ {
+ value = (float)(DC->cursorx - x) / width;
+ value *= (item->typeData.edit->maxVal - item->typeData.edit->minVal);
+ value += item->typeData.edit->minVal;
+ DC->setCVar(item->cvar, va("%f", value));
+ return qtrue;
+ }
}
- return qtrue;
- }
+ }
- if ( key == K_INS || key == K_KP_INS ) {
- DC->setOverstrikeMode(!DC->getOverstrikeMode());
- return qtrue;
- }
+ return qfalse;
+}
+
+qboolean Item_HandleKey(itemDef_t *item, int key, qboolean down)
+{
+ if (itemCapture)
+ {
+ Item_StopCapture(itemCapture);
+ itemCapture = NULL;
+ UI_RemoveCaptureFunc();
+ }
+ else
+ {
+ if (down && (key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3))
+ Item_StartCapture(item, key);
}
- if (key == K_TAB || key == K_DOWNARROW || key == K_KP_DOWNARROW) {
- if (item->type == ITEM_TYPE_SAYFIELD) {
- return qtrue;
- }
+ if (!down)
+ return qfalse;
+
+ // Edit fields are handled specially
+ if (Item_IsEditField(item))
+ return qfalse;
+
+ switch (item->type)
+ {
+ case ITEM_TYPE_BUTTON:
+ return qfalse;
+
+ case ITEM_TYPE_RADIOBUTTON:
+ return qfalse;
+
+ case ITEM_TYPE_CHECKBOX:
+ return qfalse;
+
+ case ITEM_TYPE_CYCLE:
+ return Item_Cycle_HandleKey(item, key);
- newItem = Menu_SetNextCursorItem(item->parent);
- if (newItem && (newItem->type == ITEM_TYPE_EDITFIELD || newItem->type == ITEM_TYPE_SAYFIELD || newItem->type == ITEM_TYPE_NUMERICFIELD)) {
- g_editItem = newItem;
- }
+ case ITEM_TYPE_LISTBOX:
+ return Item_ListBox_HandleKey(item, key, down, qfalse);
+
+ case ITEM_TYPE_COMBOBOX:
+ return Item_ComboBox_HandleKey(item, key, down, qfalse);
+
+ case ITEM_TYPE_YESNO:
+ return Item_YesNo_HandleKey(item, key);
+
+ case ITEM_TYPE_MULTI:
+ return Item_Multi_HandleKey(item, key);
+
+ case ITEM_TYPE_OWNERDRAW:
+ return Item_OwnerDraw_HandleKey(item, key);
+
+ case ITEM_TYPE_BIND:
+ return Item_Bind_HandleKey(item, key, down);
+
+ case ITEM_TYPE_SLIDER:
+ return Item_Slider_HandleKey(item, key, down);
+
+ default:
+ return qfalse;
}
+}
- if (key == K_UPARROW || key == K_KP_UPARROW) {
- if (item->type == ITEM_TYPE_SAYFIELD) {
- return qtrue;
- }
+void Item_Action(itemDef_t *item)
+{
+ if (item)
+ Item_RunScript(item, item->action);
+}
+
+itemDef_t *Menu_SetPrevCursorItem(menuDef_t *menu)
+{
+ qboolean wrapped = qfalse;
+ int oldCursor = menu->cursorItem;
- newItem = Menu_SetPrevCursorItem(item->parent);
- if (newItem && (newItem->type == ITEM_TYPE_EDITFIELD || newItem->type == ITEM_TYPE_SAYFIELD || newItem->type == ITEM_TYPE_NUMERICFIELD)) {
- g_editItem = newItem;
- }
+ if (menu->cursorItem < 0)
+ {
+ menu->cursorItem = menu->itemCount - 1;
+ wrapped = qtrue;
}
- if (key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3 || key == K_MOUSE4) {
- if (item->type == ITEM_TYPE_SAYFIELD) {
- return qtrue;
- }
- return qfalse;
+ 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];
+ }
}
- if ( key == K_ENTER || key == K_KP_ENTER || key == K_ESCAPE) {
- return qfalse;
+ menu->cursorItem = oldCursor;
+ return NULL;
+}
+
+itemDef_t *Menu_SetNextCursorItem(menuDef_t *menu)
+{
+ qboolean wrapped = qfalse;
+ int oldCursor = menu->cursorItem;
+
+ if (menu->cursorItem == -1)
+ {
+ menu->cursorItem = 0;
+ wrapped = qtrue;
}
- return qtrue;
- }
- return qfalse;
-
-}
-
-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;
- }
- }
-}
-
-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;
- }
- }
-}
-
-static void Scroll_Slider_ThumbFunc(void *p) {
- float x, value, cursorx;
- scrollInfo_t *si = (scrollInfo_t*)p;
- editFieldDef_t *editDef = 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));
-}
-
-void Item_StartCapture(itemDef_t *item, int key) {
- int flags;
- switch (item->type) {
- case ITEM_TYPE_EDITFIELD:
- case ITEM_TYPE_SAYFIELD:
- 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;
- }
- }
-}
-
-void Item_StopCapture(itemDef_t *item) {
-
-}
-
-qboolean Item_Slider_HandleKey(itemDef_t *item, int key, qboolean down) {
- float x, value, width, work;
-
- 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 = 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;
- testRect.w = (SLIDER_WIDTH + (float)SLIDER_THUMB_WIDTH / 2);
- 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;
- }
- }
- }
- }
- return qfalse;
-}
-
-
-qboolean Item_HandleKey(itemDef_t *item, int key, qboolean down) {
-
- if (itemCapture) {
- Item_StopCapture(itemCapture);
- itemCapture = NULL;
- captureFunc = voidFunction;
- captureData = NULL;
- } else {
- // bk001206 - parentheses
- if ( down && ( key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3 ) ) {
- Item_StartCapture(item, key);
+ 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];
+ }
}
- }
- 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_SAYFIELD:
- 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;
- //case ITEM_TYPE_IMAGE:
- // Item_Image_Paint(item);
- // break;
- default:
- return qfalse;
- break;
- }
-
- //return qfalse;
-}
-
-void Item_Action(itemDef_t *item) {
- if (item) {
- Item_RunScript(item, item->action);
- }
-}
-
-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;
-
-}
-
-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->cursorItem = oldCursor;
+ return NULL;
}
-static void Window_CloseCinematic(windowDef_t *window) {
- if (window->style == WINDOW_STYLE_CINEMATIC && window->cinematic >= 0) {
- DC->stopCinematic(window->cinematic);
- window->cinematic = -1;
- }
+static void Window_CloseCinematic(Window *window)
+{
+ if (window->style == WINDOW_STYLE_CINEMATIC && window->cinematic >= 0)
+ {
+ DC->stopCinematic(window->cinematic);
+ window->cinematic = -1;
+ }
}
-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);
- }
+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);
+ }
}
- }
}
-static void Display_CloseCinematics( void ) {
- int i;
- for (i = 0; i < menuCount; i++) {
- Menu_CloseCinematics(&Menus[i]);
- }
+static void Display_CloseCinematics(void)
+{
+ int i;
+
+ for (i = 0; i < menuCount; i++)
+ Menu_CloseCinematics(&Menus[i]);
}
-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);
- }
+void Menus_Activate(menuDef_t *menu)
+{
+ int i;
+ qboolean onTopOfMenuStack = qfalse;
- if (menu->soundName && *menu->soundName) {
-// DC->stopBackgroundTrack(); // you don't want to do this since it will reset s_rawend
- DC->startBackgroundTrack(menu->soundName, menu->soundName);
- }
+ if (openMenuCount > 0 && menuStack[openMenuCount - 1] == menu)
+ onTopOfMenuStack = qtrue;
- Display_CloseCinematics();
+ menu->window.flags |= (WINDOW_HASFOCUS | WINDOW_VISIBLE);
-}
+ // If being opened for the first time
+ if (!onTopOfMenuStack)
+ {
+ if (menu->onOpen)
+ {
+ itemDef_t item;
+ item.parent = menu;
+ Item_RunScript(&item, menu->onOpen);
+ }
+
+ if (menu->soundName && *menu->soundName)
+ DC->startBackgroundTrack(menu->soundName, menu->soundName);
+
+ Display_CloseCinematics();
-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++;
+ Menu_HandleMouseMove(menu, DC->cursorx, DC->cursory); // force the item under the cursor to focus
+
+ for (i = 0; i < menu->itemCount; i++) // reset selection in listboxes when opened
+ {
+ if (Item_IsListBox(menu->items[i]))
+ {
+ menu->items[i]->cursorPos = DC->feederInitialise(menu->items[i]->feederID);
+ Item_ListBox_SetStartPos(menu->items[i], 0);
+ DC->feederSelection(menu->items[i]->feederID, menu->items[i]->cursorPos);
+ }
+ else if (menu->items[i]->type == ITEM_TYPE_CYCLE)
+ {
+ menu->items[i]->typeData.cycle->cursorPos = DC->feederInitialise(menu->items[i]->feederID);
+ }
+ }
+
+ if (openMenuCount < MAX_OPEN_MENUS)
+ menuStack[openMenuCount++] = menu;
}
- }
- return count;
}
-void Menus_HandleOOBClick(menuDef_t *menu, int key, qboolean down) {
- if (menu) {
+qboolean Menus_ReplaceActive(menuDef_t *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);
+ menuDef_t *active;
+
+ if (openMenuCount < 1)
+ return qfalse;
+
+ active = menuStack[openMenuCount - 1];
+
+ if (!(active->window.flags & WINDOW_HASFOCUS) || !(active->window.flags & WINDOW_VISIBLE))
+ {
+ return qfalse;
}
- 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 (menu == active)
+ return qfalse;
+
+ if (menu->itemCount != active->itemCount)
+ {
+ Com_Printf(S_COLOR_YELLOW "WARNING: Menus_ReplaceActive: expecting %i menu items, found %i\n", menu->itemCount,
+ active->itemCount);
+ return qfalse;
}
- if (Display_VisibleMenuCount() == 0) {
- if (DC->Pause) {
- DC->Pause(qfalse);
- }
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ if (menu->items[i]->type != active->items[i]->type)
+ {
+ Com_Printf(S_COLOR_YELLOW "WARNING: Menus_ReplaceActive: type mismatch on item %i\n", i + 1);
+ return qfalse;
+ }
}
- Display_CloseCinematics();
- }
+
+ active->window.flags &= ~(WINDOW_FADINGOUT | WINDOW_VISIBLE);
+ menu->window.flags |= (WINDOW_HASFOCUS | WINDOW_VISIBLE);
+
+ menuStack[openMenuCount - 1] = menu;
+ if (menu->onOpen)
+ {
+ itemDef_t item;
+ item.parent = menu;
+ Item_RunScript(&item, menu->onOpen);
+ }
+
+ // set the cursor position on the new menu to match the active one
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ menu->items[i]->cursorPos = active->items[i]->cursorPos;
+ menu->items[i]->feederID = active->items[i]->feederID;
+ switch (Item_DataType(menu->items[i]))
+ {
+ case TYPE_LIST:
+ menu->items[i]->typeData.list->startPos = active->items[i]->typeData.list->startPos;
+ menu->items[i]->typeData.list->cursorPos = active->items[i]->typeData.list->cursorPos;
+ break;
+ case TYPE_COMBO:
+ menu->items[i]->typeData.cycle->cursorPos = active->items[i]->typeData.cycle->cursorPos;
+ break;
+ default:
+ break;
+ }
+ }
+ return qtrue;
}
-static rectDef_t *Item_CorrectedTextRect(itemDef_t *item) {
- static rectDef_t rect;
- memset(&rect, 0, sizeof(rectDef_t));
- if (item) {
- rect = item->textRect;
- if (rect.w) {
- rect.y -= rect.h;
+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 &rect;
+
+ return count;
}
-void Menu_HandleKey(menuDef_t *menu, int key, qboolean down) {
- int i;
- itemDef_t *item = NULL;
- qboolean inHandler = qfalse;
+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)
+ Menus_Close(menu);
- if (inHandler) {
- return;
- }
+ for (i = 0; i < menuCount; i++)
+ {
+ if (Menu_OverActiveItem(&Menus[i], DC->cursorx, DC->cursory))
+ {
+ Menus_Close(menu);
+ Menus_Activate(&Menus[i]);
+ Menu_HandleMouseMove(&Menus[i], DC->cursorx, DC->cursory);
+ Menu_HandleKey(&Menus[i], key, down);
+ }
+ }
- inHandler = qtrue;
- if (g_waitingForKey && down) {
- Item_Bind_HandleKey(g_bindItem, key, down);
- inHandler = qfalse;
- return;
- }
+ if (Display_VisibleMenuCount() == 0)
+ {
+ if (DC->Pause)
+ DC->Pause(qfalse);
+ }
- if (g_editingField && down) {
- if (!Item_TextField_HandleKey(g_editItem, key)) {
- g_editingField = qfalse;
- Item_RunScript(g_editItem, g_editItem->onTextEntry);
- g_editItem = NULL;
- inHandler = qfalse;
- return;
+ Display_CloseCinematics();
}
- }
+}
- if (menu == NULL) {
- inHandler = qfalse;
- return;
- }
+static rectDef_t *Item_CorrectedTextRect(itemDef_t *item)
+{
+ static rectDef_t rect;
+ memset(&rect, 0, sizeof(rectDef_t));
- // 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;
- // bk001206 - parentheses
- 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")) {
- 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 (item->type == ITEM_TYPE_SAYFIELD) {
- // do nothing
- } 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_SAYFIELD || item->type == ITEM_TYPE_NUMERICFIELD) {
- item->cursorPos = 0;
- g_editingField = qtrue;
- g_editItem = item;
- DC->setOverstrikeMode(qtrue);
- } else {
- Item_Action(item);
- }
- }
- break;
- }
- inHandler = qfalse;
-}
-
-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;
-}
-
-void Rect_ToWindowCoords(rectDef_t *rect, windowDef_t *window) {
- ToWindowCoords(&rect->x, &rect->y, window);
-}
-
-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 originalWidth = DC->textWidth(item->text, item->textscale, 0);
-
- 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, 0);
- }
-
- *width = DC->textWidth(textPtr, item->textscale, 0);
- *height = DC->textHeight(textPtr, item->textscale, 0);
- 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);
- }
-}
-
-void Item_TextColor(itemDef_t *item, vec4_t *newColor) {
- vec4_t 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);
-
- 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));*/
- //TA:
- memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
- } 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)
+ {
+ rect = item->textRect;
- if (item->enableCvar != NULL && *item->enableCvar && item->cvarTest != NULL && *item->cvarTest) {
- if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE)) {
- memcpy(newColor, &parent->disableColor, sizeof(vec4_t));
+ if (rect.w)
+ rect.y -= rect.h;
}
- }
+
+ return &rect;
}
-int Item_Text_AutoWrapped_Lines( itemDef_t *item )
+void Menu_HandleKey(menuDef_t *menu, int key, qboolean down)
{
- char text[ 1024 ];
- const char *p, *textPtr, *newLinePtr;
- char buff[ 1024 ];
- int len, textWidth, newLine;
- int lines = 0;
+ int i;
+ itemDef_t *item = NULL;
- textWidth = 0;
- newLinePtr = NULL;
+ if (g_waitingForKey && down)
+ {
+ Item_Bind_HandleKey(g_bindItem, key, down);
+ return;
+ }
- if( item->text == NULL )
- {
- if( item->cvar == NULL )
- return 0;
- else
+ if (g_editingField && down)
{
- DC->getCVarString( item->cvar, text, sizeof( text ) );
- textPtr = text;
+ if (!Item_TextField_HandleKey(g_editItem, key))
+ {
+ g_editingField = qfalse;
+ Item_RunScript(g_editItem, g_editItem->onTextEntry);
+ g_editItem = NULL;
+ return;
+ }
+ else
+ {
+ Item_RunScript(g_editItem, g_editItem->onCharEntry);
+ }
}
- }
- else
- textPtr = item->text;
- if( *textPtr == '\0' )
- return 0;
+ if (menu == NULL)
+ return;
- len = 0;
- buff[ 0 ] = '\0';
- newLine = 0;
- p = textPtr;
+ // 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;
- while( p )
- {
- textWidth = DC->textWidth( buff, item->textscale, 0 );
+ if (!inHandleKey && (key == K_MOUSE1 || key == K_MOUSE2 || key == K_MOUSE3))
+ {
+ inHandleKey = qtrue;
+ Menus_HandleOOBClick(menu, key, down);
+ inHandleKey = qfalse;
+ return;
+ }
+ }
- if( *p == ' ' || *p == '\t' || *p == '\n' || *p == '\0' )
+ if (g_comboBoxItem == NULL)
{
- newLine = len;
- newLinePtr = p + 1;
+ // get the item with focus
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ if (menu->items[i]->window.flags & WINDOW_HASFOCUS)
+ item = menu->items[i];
+ }
}
+ else
+ item = g_comboBoxItem;
- //TA: forceably split lines that are too long (where normal splitage has failed)
- if( textWidth > item->window.rect.w && newLine == 0 && *p != '\n' )
+ if (item != NULL)
{
- newLine = len;
- newLinePtr = p;
+ if (Item_HandleKey(item, key, down))
+ {
+ Item_Action(item);
+ return;
+ }
}
- if( ( newLine && textWidth > item->window.rect.w ) || *p == '\n' || *p == '\0' )
- {
- if( len )
- buff[ newLine ] = '\0';
+ if (!down)
+ return;
- if( !( *p == '\n' && !*( p + 1 ) ) )
- lines++;
+ // default handling
+ switch (key)
+ {
+ 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);
+ }
- if( *p == '\0' )
- break;
+ 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_IsEditField(item))
+ {
+ if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory))
+ {
+ char buffer[MAX_STRING_CHARS] = {0};
+
+ if (item->cvar)
+ DC->getCVarString(item->cvar, buffer, sizeof(buffer));
+
+ item->cursorPos = strlen(buffer);
+
+ Item_TextField_CalcPaintOffset(item, buffer);
+
+ g_editingField = qtrue;
+
+ g_editItem = item;
+ }
+ }
+ else
+ {
+ if (Rect_ContainsPoint(&item->window.rect, DC->cursorx, DC->cursory))
+ Item_Action(item);
+ }
+ }
- //
- p = newLinePtr;
- len = 0;
- newLine = 0;
+ 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_IsEditField(item))
+ {
+ char buffer[MAX_STRING_CHARS] = {0};
+
+ if (item->cvar)
+ DC->getCVarString(item->cvar, buffer, sizeof(buffer));
+
+ item->cursorPos = strlen(buffer);
+
+ Item_TextField_CalcPaintOffset(item, buffer);
+
+ g_editingField = qtrue;
+
+ g_editItem = item;
+ }
+ else
+ Item_Action(item);
+ }
- continue;
+ break;
}
+}
- buff[ len++ ] = *p++;
- buff[ len ] = '\0';
- }
+void ToWindowCoords(float *x, float *y, Window *window)
+{
+ if (window->border != 0)
+ {
+ *x += window->borderSize;
+ *y += window->borderSize;
+ }
- return lines;
+ *x += window->rect.x;
+ *y += window->rect.y;
}
-#define MAX_AUTOWRAP_CACHE 16
-#define MAX_AUTOWRAP_LINES 32
-#define MAX_AUTOWRAP_TEXT 512
+void Rect_ToWindowCoords(rectDef_t *rect, Window *window) { ToWindowCoords(&rect->x, &rect->y, window); }
-typedef struct
+void Item_SetTextExtents(itemDef_t *item, const char *text)
{
- //this is used purely for checking for cache hits
- char text[ MAX_AUTOWRAP_TEXT * MAX_AUTOWRAP_LINES ];
- rectDef_t rect;
- int textWidth, textHeight;
- char lines[ MAX_AUTOWRAP_LINES ][ MAX_AUTOWRAP_TEXT ];
- int lineOffsets[ MAX_AUTOWRAP_LINES ][ 2 ];
- int numLines;
-} autoWrapCache_t;
+ const char *textPtr = (text) ? text : item->text;
+ qboolean cvarContent;
-static int cacheIndex = 0;
-static autoWrapCache_t awc[ MAX_AUTOWRAP_CACHE ];
+ // It's hard to make a policy on what should be aligned statically and what
+ // should be aligned dynamically; there are reasonable cases for both. If
+ // it continues to be a problem then there should probably be an item keyword
+ // for it; but for the moment only adjusting the alignment of ITEM_TYPE_TEXT
+ // seems to suffice.
+ cvarContent = (item->cvar && item->textalignment != ALIGN_LEFT && item->type == ITEM_TYPE_TEXT);
-static int checkCache( const char *text, rectDef_t *rect, int width, int height )
-{
- int i;
+ if (textPtr == NULL)
+ return;
- for( i = 0; i < MAX_AUTOWRAP_CACHE; i++ )
- {
- if( Q_stricmp( text, awc[ i ].text ) )
- continue;
+ // as long as the item isn't dynamic content (ownerdraw or cvar), this
+ // keeps us from computing the widths and heights more than once
+ if (item->textRect.w == 0.0f || cvarContent ||
+ (item->type == ITEM_TYPE_OWNERDRAW && item->textalignment != ALIGN_LEFT))
+ {
+ float originalWidth = 0.0f;
+
+ if (item->textalignment == ALIGN_CENTER || item->textalignment == ALIGN_RIGHT)
+ {
+ if (cvarContent)
+ {
+ char buff[MAX_CVAR_VALUE_STRING];
+ DC->getCVarString(item->cvar, buff, sizeof(buff));
+ originalWidth = UI_Text_Width(item->text, item->textscale) + UI_Text_Width(buff, item->textscale);
+ }
+ else
+ originalWidth = UI_Text_Width(item->text, item->textscale);
+ }
- if( rect->x != awc[ i ].rect.x ||
- rect->y != awc[ i ].rect.y ||
- rect->w != awc[ i ].rect.w ||
- rect->h != awc[ i ].rect.h )
- continue;
+ item->textRect.w = UI_Text_Width(textPtr, item->textscale);
+ item->textRect.h = UI_Text_Height(textPtr, item->textscale);
- if( awc[ i ].textWidth != width || awc[ i ].textHeight != height )
- continue;
+ if (item->textvalignment == VALIGN_BOTTOM)
+ item->textRect.y = item->textaligny + item->window.rect.h;
+ else if (item->textvalignment == VALIGN_CENTER)
+ item->textRect.y = item->textaligny + ((item->textRect.h + item->window.rect.h) / 2.0f);
+ else if (item->textvalignment == VALIGN_TOP)
+ item->textRect.y = item->textaligny + item->textRect.h;
- //this is a match
- return i;
- }
+ if (item->textalignment == ALIGN_LEFT)
+ item->textRect.x = item->textalignx;
+ else if (item->textalignment == ALIGN_CENTER)
+ item->textRect.x = item->textalignx + ((item->window.rect.w - originalWidth) / 2.0f);
+ else if (item->textalignment == ALIGN_RIGHT)
+ item->textRect.x = item->textalignx + item->window.rect.w - originalWidth;
- //no match - autowrap isn't cached
- return -1;
+ ToWindowCoords(&item->textRect.x, &item->textRect.y, &item->window);
+ }
}
-void Item_Text_AutoWrapped_Paint( itemDef_t *item )
+void Item_TextColor(itemDef_t *item, vec4_t *newColor)
{
- char text[ 1024 ];
- const char *p, *textPtr, *newLinePtr;
- char buff[ 1024 ];
- char lastCMod[ 2 ] = { 0, 0 };
- qboolean forwardColor = qfalse;
- int width, height, len, textWidth, newLine, newLineWidth;
- int skipLines, totalLines, lineNum = 0;
- float y, totalY, diffY;
- vec4_t color;
- int cache, i;
+ vec4_t lowLight;
+ menuDef_t *parent = (menuDef_t *)item->parent;
- textWidth = 0;
- newLinePtr = NULL;
+ Fade(&item->window.flags, &item->window.foreColor[3], parent->fadeClamp, &item->window.nextTime, parent->fadeCycle,
+ qtrue, parent->fadeAmount);
- if( item->text == NULL )
- {
- if( item->cvar == NULL )
- return;
+ if (item->window.flags & WINDOW_HASFOCUS)
+ memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
+ 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
{
- DC->getCVarString( item->cvar, text, sizeof( text ) );
- textPtr = text;
+ memcpy(newColor, &item->window.foreColor, sizeof(vec4_t));
+ // items can be enabled and disabled based on cvars
}
- }
- else
- textPtr = item->text;
- if( *textPtr == '\0' )
- return;
-
- Item_TextColor( item, &color );
- Item_SetTextExtents( item, &width, &height, textPtr );
+ if (item->enableCvar != NULL && *item->enableCvar && item->cvarTest != NULL && *item->cvarTest)
+ {
+ if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE))
+ memcpy(newColor, &parent->disableColor, sizeof(vec4_t));
+ }
+}
- //check if this block is cached
- cache = checkCache( textPtr, &item->window.rect, width, height );
- if( cache >= 0 )
- {
- lineNum = awc[ cache ].numLines;
+static void SkipColorCodes(const char **text, char *lastColor)
+{
+ while (Q_IsColorString(*text))
+ {
+ lastColor[0] = (*text)[0];
+ lastColor[1] = (*text)[1];
+ (*text) += 2;
+ }
+}
- for( i = 0; i < lineNum; i++ )
+static void SkipWhiteSpace(const char **text, char *lastColor)
+{
+ while (**text)
{
- item->textRect.x = awc[ cache ].lineOffsets[ i ][ 0 ];
- item->textRect.y = awc[ cache ].lineOffsets[ i ][ 1 ];
+ SkipColorCodes(text, lastColor);
- DC->drawText( item->textRect.x, item->textRect.y, item->textscale, color,
- awc[ cache ].lines[ i ], 0, 0, item->textStyle );
+ if (**text != '\n' && isspace(**text))
+ (*text)++;
+ else
+ break;
}
- }
- else
- {
- y = item->textaligny;
- len = 0;
- buff[ 0 ] = '\0';
- newLine = 0;
- newLineWidth = 0;
- p = textPtr;
+}
- totalLines = Item_Text_AutoWrapped_Lines( item );
+const char *Item_Text_Wrap(const char *text, float scale, float width)
+{
+ static char out[8192] = "";
+ char *paint = out;
+ char c[3] = "";
+ const char *p;
+ const char *eos;
+ float indentWidth = 0.0f;
- totalY = totalLines * ( height + 5 );
- diffY = totalY - item->window.rect.h;
+ if (!text)
+ return NULL;
- if( diffY > 0.0f )
- skipLines = (int)( diffY / ( (float)height + 5.0f ) );
- else
- skipLines = 0;
-
- //set up a cache entry
- strcpy( awc[ cacheIndex ].text, textPtr );
- awc[ cacheIndex ].rect.x = item->window.rect.x;
- awc[ cacheIndex ].rect.y = item->window.rect.y;
- awc[ cacheIndex ].rect.w = item->window.rect.w;
- awc[ cacheIndex ].rect.h = item->window.rect.h;
- awc[ cacheIndex ].textWidth = width;
- awc[ cacheIndex ].textHeight = height;
-
- while( p )
- {
- textWidth = DC->textWidth( buff, item->textscale, 0 );
-
- if( *p == '^' )
- {
- lastCMod[ 0 ] = p[ 0 ];
- lastCMod[ 1 ] = p[ 1 ];
- }
-
- if( *p == ' ' || *p == '\t' || *p == '\n' || *p == '\0' )
- {
- newLine = len;
- newLinePtr = p+1;
- newLineWidth = textWidth;
-
- if( *p == '\n' ) //don't forward colours past deilberate \n's
- lastCMod[ 0 ] = lastCMod[ 1 ] = 0;
- else
- forwardColor = qtrue;
- }
+ p = text;
+ eos = p + strlen(p);
+
+ if ((eos - p) >= sizeof(out))
+ return NULL;
- //TA: forceably split lines that are too long (where normal splitage has failed)
- if( textWidth > item->window.rect.w && newLine == 0 && *p != '\n' )
- {
- newLine = len;
- newLinePtr = p;
- newLineWidth = textWidth;
+ *paint = '\0';
+
+ while (*p)
+ {
+ float textWidth = 0.0f;
+ const char *eol = p;
+ const char *q = p;
+ float testWidth = width - indentWidth;
- forwardColor = qtrue;
- }
+ SkipColorCodes(&q, c);
- if( ( newLine && textWidth > item->window.rect.w ) || *p == '\n' || *p == '\0' )
- {
- if( len )
+ while (q && textWidth < testWidth)
{
- 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;
+ qboolean previousCharIsSpace = qfalse;
+
+ // Remaining string is too short to wrap
+ if (q >= eos)
+ {
+ eol = eos;
+ break;
+ }
+
+ if (q > p && *q == INDENT_MARKER)
+ {
+ indentWidth = textWidth;
+ eol = p;
+ }
- item->textRect.y = y;
- ToWindowCoords( &item->textRect.x, &item->textRect.y, &item->window );
- //
- buff[ newLine ] = '\0';
+ // Some color escapes might still be present
+ SkipColorCodes(&q, c);
- if( !skipLines )
- {
- DC->drawText( item->textRect.x, item->textRect.y, item->textscale, color, buff, 0, 0, item->textStyle );
+ // Manual line break
+ if (*q == '\n')
+ {
+ eol = q + 1;
+ break;
+ }
- strcpy( awc[ cacheIndex ].lines[ lineNum ], buff );
- awc[ cacheIndex ].lineOffsets[ lineNum ][ 0 ] = item->textRect.x;
- awc[ cacheIndex ].lineOffsets[ lineNum ][ 1 ] = item->textRect.y;
+ if (!previousCharIsSpace && isspace(*q))
+ eol = q;
- lineNum++;
- }
+ textWidth += UI_Char_Width(&q, scale);
}
- if( *p == '\0' )
- break;
- //
- if( !skipLines )
- y += height + 5;
+ // No split has taken place, so just split mid-word
+ if (eol == p)
+ eol = q;
+
+ // Note that usage of strcat and strlen is deliberately being
+ // avoided here as it becomes surprisingly expensive on larger
+ // blocks of text
- if( skipLines )
- skipLines--;
-
- p = newLinePtr;
- len = 0;
- newLine = 0;
- newLineWidth = 0;
+ // Copy text
+ strncpy(paint, p, eol - p);
+ paint += (eol - p);
+ *paint = '\0';
- if( forwardColor && lastCMod[ 0 ] != 0 )
- {
- buff[ len++ ] = lastCMod[ 0 ];
- buff[ len++ ] = lastCMod[ 1 ];
- buff[ len ] = '\0';
+ p = eol;
- forwardColor = qfalse;
+ if (paint - out > 0 && *(paint - 1) == '\n')
+ {
+ // The line is deliberately broken, clear the color and
+ // any current indent
+ c[0] = '\0';
+ indentWidth = 0.0f;
}
+ else
+ {
+ // Add a \n if it's not there already
+ *paint++ = '\n';
+ *paint = '\0';
+
+ // Insert a pixel indent on the next line
+ if (indentWidth > 0.0f)
+ {
+ const char *indentMarkerText = va("%f%c", indentWidth, INDENT_MARKER);
+ int indentMarkerTextLength = strlen(indentMarkerText);
+
+ strncpy(paint, indentMarkerText, indentMarkerTextLength);
+ paint += indentMarkerTextLength;
+ *paint = '\0';
+ }
- continue;
- }
+ // Skip leading whitespace on next line and save the
+ // last color code
+ SkipWhiteSpace(&p, c);
+ }
- buff[ len++ ] = *p++;
- buff[ len ] = '\0';
+ if (c[0])
+ {
+ *paint++ = c[0];
+ *paint++ = c[1];
+ *paint = '\0';
+ }
}
- //mark the end of the lines list
- awc[ cacheIndex ].numLines = lineNum;
+ return out;
+}
+
+#define MAX_WRAP_CACHE 16
+#define MAX_WRAP_LINES 32
+#define MAX_WRAP_TEXT 512
+
+typedef struct {
+ char text[MAX_WRAP_TEXT * MAX_WRAP_LINES];
+ rectDef_t rect;
+ float scale;
+ char lines[MAX_WRAP_LINES][MAX_WRAP_TEXT];
+ float lineCoords[MAX_WRAP_LINES][2];
+ int numLines;
+} wrapCache_t;
+
+static wrapCache_t wrapCache[MAX_WRAP_CACHE];
+static qboolean cacheCreationFailed = qfalse;
+static int cacheWriteIndex = 0;
+static int cacheReadIndex = 0;
+static int cacheReadLineNum = 0;
+
+static void UI_CreateCacheEntry(const char *text, const rectDef_t *rect, float scale)
+{
+ wrapCache_t *cacheEntry = &wrapCache[cacheWriteIndex];
- //increment cacheIndex
- cacheIndex = ( cacheIndex + 1 ) % MAX_AUTOWRAP_CACHE;
- }
+ if (strlen(text) >= sizeof(cacheEntry->text))
+ {
+ cacheCreationFailed = qtrue;
+ return;
+ }
+
+ strcpy(cacheEntry->text, text);
+ cacheEntry->rect = *rect;
+ cacheEntry->scale = scale;
+ cacheEntry->numLines = 0;
}
-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;
+static void UI_AddCacheEntryLine(const char *text, float x, float y)
+{
+ wrapCache_t *cacheEntry = &wrapCache[cacheWriteIndex];
- // now paint the text and/or any optional images
- // default to left
+ if (cacheCreationFailed)
+ return;
- if (item->text == NULL) {
- if (item->cvar == NULL) {
- return;
+ if (cacheEntry->numLines >= MAX_WRAP_LINES || strlen(text) >= sizeof(cacheEntry->lines[0]))
+ {
+ cacheCreationFailed = qtrue;
+ return;
}
- else {
- DC->getCVarString(item->cvar, text, sizeof(text));
- textPtr = text;
+
+ strcpy(cacheEntry->lines[cacheEntry->numLines], text);
+ cacheEntry->lineCoords[cacheEntry->numLines][0] = x;
+ cacheEntry->lineCoords[cacheEntry->numLines][1] = y;
+ cacheEntry->numLines++;
+}
+
+static void UI_FinishCacheEntry(void)
+{
+ if (cacheCreationFailed)
+ {
+ wrapCache[cacheWriteIndex].text[0] = '\0';
+ wrapCache[cacheWriteIndex].numLines = 0;
+ cacheCreationFailed = qfalse;
}
- }
- else {
- textPtr = item->text;
- }
- if (*textPtr == '\0') {
- return;
- }
+ else
+ cacheWriteIndex = (cacheWriteIndex + 1) % MAX_WRAP_CACHE;
+}
- Item_TextColor(item, &color);
- Item_SetTextExtents(item, &width, &height, textPtr);
+static qboolean UI_CheckWrapCache(const char *text, const rectDef_t *rect, float scale)
+{
+ int i;
+
+ for (i = 0; i < MAX_WRAP_CACHE; i++)
+ {
+ wrapCache_t *cacheEntry = &wrapCache[i];
+
+ if (rect->x != cacheEntry->rect.x || rect->y != cacheEntry->rect.y || rect->w != cacheEntry->rect.w ||
+ rect->h != cacheEntry->rect.h)
+ continue;
+
+ if (strcmp(text, cacheEntry->text))
+ continue;
+
+ if (cacheEntry->scale != scale)
+ continue;
+
+ cacheReadIndex = i;
+ cacheReadLineNum = 0;
+
+ return qtrue;
+ }
- 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, 0, item->textStyle);
- y += height + 5;
- start += p - start + 1;
- p = strchr(p+1, '\r');
- }
- DC->drawText(x, y, item->textscale, color, start, 0, 0, item->textStyle);
+ return qfalse;
}
-void Item_Text_Paint(itemDef_t *item) {
- char text[1024];
- const char *textPtr;
- int height, width;
- vec4_t color;
+static qboolean UI_NextWrapLine(const char **text, float *x, float *y)
+{
+ wrapCache_t *cacheEntry = &wrapCache[cacheReadIndex];
+
+ if (cacheReadLineNum >= cacheEntry->numLines)
+ return qfalse;
+
+ *text = cacheEntry->lines[cacheReadLineNum];
+ *x = cacheEntry->lineCoords[cacheReadLineNum][0];
+ *y = cacheEntry->lineCoords[cacheReadLineNum][1];
- if (item->window.flags & WINDOW_WRAPPED) {
- Item_Text_Wrapped_Paint(item);
- return;
- }
- if (item->window.flags & WINDOW_AUTOWRAPPED) {
- Item_Text_AutoWrapped_Paint(item);
- return;
- }
+ cacheReadLineNum++;
+
+ return qtrue;
+}
+
+void Item_Text_Wrapped_Paint(itemDef_t *item)
+{
+ char text[1024];
+ const char *p, *textPtr;
+ float x, y, w, h;
+ vec4_t color;
+ qboolean useWrapCache = (qboolean)DC->getCVarValue("ui_textWrapCache");
- if (item->text == NULL) {
- if (item->cvar == NULL) {
- return;
+ if (item->text == NULL)
+ {
+ if (item->cvar == NULL)
+ return;
+ else
+ {
+ DC->getCVarString(item->cvar, text, sizeof(text));
+ textPtr = text;
+ }
}
- else {
- DC->getCVarString(item->cvar, text, sizeof(text));
- textPtr = text;
+ else
+ textPtr = item->text;
+
+ if (*textPtr == '\0')
+ return;
+
+ Item_TextColor(item, &color);
+
+ // Check if this block is cached
+ if (useWrapCache && UI_CheckWrapCache(textPtr, &item->window.rect, item->textscale))
+ {
+ while (UI_NextWrapLine(&p, &x, &y))
+ {
+ UI_Text_Paint(x, y, item->textscale, color, p, 0, 0, item->textStyle);
+ }
}
- }
- else {
- textPtr = item->text;
- }
+ else
+ {
+ char buff[1024];
+ float fontHeight = UI_Text_EmHeight(item->textscale);
+ const float lineSpacing = fontHeight * 0.4f;
+ float lineHeight = fontHeight + lineSpacing;
+ float textHeight;
+ int textLength;
+ int firstLine, paintLines, totalLines, lineNum;
+ float paintY;
+ int i;
+
+ if (useWrapCache)
+ UI_CreateCacheEntry(textPtr, &item->window.rect, item->textscale);
+
+ x = item->window.rect.x + item->textalignx;
+ y = item->window.rect.y + item->textaligny;
+ w = item->window.rect.w - (2.0f * item->textalignx);
+ h = item->window.rect.h - (2.0f * item->textaligny);
+
+ textPtr = Item_Text_Wrap(textPtr, item->textscale, w);
+ textLength = strlen(textPtr);
+
+ // Count lines
+ totalLines = 0;
+
+ for (i = 0; i < textLength; i++)
+ {
+ if (textPtr[i] == '\n')
+ totalLines++;
+ }
+ if (textLength && textPtr[textLength - 1] != '\n')
+ {
+ totalLines++; // count the last, non-newline-terminated line
+ textLength++; // a '\0' will mark the end of the last line
+ }
- // this needs to go here as it sets extents for cvar types as well
- Item_SetTextExtents(item, &width, &height, textPtr);
+ paintLines = (int)floor((h + lineSpacing) / lineHeight);
- if (*textPtr == '\0') {
- return;
- }
+ if (totalLines > paintLines)
+ firstLine = totalLines - paintLines;
+ else
+ {
+ firstLine = 0;
+ paintLines = totalLines;
+ }
+
+ textHeight = (paintLines * lineHeight) - lineSpacing;
+
+ switch (item->textvalignment)
+ {
+ default:
+
+ case VALIGN_BOTTOM:
+ paintY = y + (h - textHeight);
+ break;
+
+ case VALIGN_CENTER:
+ paintY = y + ((h - textHeight) / 2.0f);
+ break;
+
+ case VALIGN_TOP:
+ paintY = y;
+ break;
+ }
+
+ p = textPtr;
+ // skip the first few lines
+ for (lineNum = 0; lineNum < firstLine; lineNum++)
+ p = strchr(p, '\n') + 1;
- Item_TextColor(item, &color);
+ for (i = p - textPtr; i < textLength && lineNum < firstLine + paintLines; i++)
+ {
+ unsigned long lineLength = &textPtr[i] - p;
+
+ if (lineLength >= sizeof(buff))
+ break;
+
+ if (textPtr[i] == '\n' || textPtr[i] == '\0')
+ {
+ itemDef_t lineItem;
+
+ memset(&lineItem, 0, sizeof(itemDef_t));
+ strncpy(buff, p, lineLength);
+ buff[lineLength] = '\0';
+ p = &textPtr[i + 1];
+
+ lineItem.type = ITEM_TYPE_TEXT;
+ lineItem.textscale = item->textscale;
+ lineItem.textStyle = item->textStyle;
+ lineItem.text = buff;
+ lineItem.textalignment = item->textalignment;
+ lineItem.textvalignment = VALIGN_TOP;
+ lineItem.textalignx = 0.0f;
+ lineItem.textaligny = 0.0f;
+
+ lineItem.textRect.w = 0.0f;
+ lineItem.textRect.h = 0.0f;
+ lineItem.window.rect.x = x;
+ lineItem.window.rect.y = paintY + ((lineNum - firstLine) * lineHeight);
+ lineItem.window.rect.w = w;
+ lineItem.window.rect.h = lineHeight;
+ lineItem.window.border = item->window.border;
+ lineItem.window.borderSize = item->window.borderSize;
+
+ if (DC->getCVarValue("ui_developer"))
+ {
+ vec4_t color;
+ color[0] = color[2] = color[3] = 1.0f;
+ color[1] = 0.0f;
+ DC->drawRect(lineItem.window.rect.x, lineItem.window.rect.y, lineItem.window.rect.w,
+ lineItem.window.rect.h, 1, color);
+ }
+
+ Item_SetTextExtents(&lineItem, buff);
+ UI_Text_Paint(lineItem.textRect.x, lineItem.textRect.y, lineItem.textscale, color, buff, 0, 0,
+ lineItem.textStyle);
+
+ if (useWrapCache)
+ UI_AddCacheEntryLine(buff, lineItem.textRect.x, lineItem.textRect.y);
+
+ lineNum++;
+ }
+ }
+
+ if (useWrapCache)
+ UI_FinishCacheEntry();
+ }
+}
- //FIXME: this is a fucking mess
/*
- adjust = 0;
- if (item->textStyle == ITEM_TEXTSTYLE_OUTLINED || item->textStyle == ITEM_TEXTSTYLE_OUTLINESHADOWED) {
- adjust = 0.5;
- }
-
- if (item->textStyle == ITEM_TEXTSTYLE_SHADOWED || item->textStyle == ITEM_TEXTSTYLE_OUTLINESHADOWED) {
- Fade(&item->window.flags, &DC->Assets.shadowColor[3], DC->Assets.fadeClamp, &item->window.nextTime, DC->Assets.fadeCycle, qfalse);
- DC->drawText(item->textRect.x + DC->Assets.shadowX, item->textRect.y + DC->Assets.shadowY, item->textscale, DC->Assets.shadowColor, textPtr, adjust);
- }
+==============
+UI_DrawTextBlock
+==============
*/
+void UI_DrawTextBlock(rectDef_t *rect, float text_x, float text_y, vec4_t color, float scale, int textalign,
+ int textvalign, int textStyle, const char *text)
+{
+ static menuDef_t dummyParent;
+ static itemDef_t textItem;
+
+ textItem.text = text;
+
+ textItem.parent = &dummyParent;
+ memcpy(textItem.window.foreColor, color, sizeof(vec4_t));
+ textItem.window.flags = 0;
+
+ textItem.window.rect.x = rect->x;
+ textItem.window.rect.y = rect->y;
+ textItem.window.rect.w = rect->w;
+ textItem.window.rect.h = rect->h;
+ textItem.window.border = 0;
+ textItem.window.borderSize = 0.0f;
+ textItem.textRect.x = 0.0f;
+ textItem.textRect.y = 0.0f;
+ textItem.textRect.w = 0.0f;
+ textItem.textRect.h = 0.0f;
+ textItem.textalignment = textalign;
+ textItem.textvalignment = textvalign;
+ textItem.textalignx = text_x;
+ textItem.textaligny = text_y;
+ textItem.textscale = scale;
+ textItem.textStyle = textStyle;
+
+ // Utilise existing wrap code
+ Item_Text_Wrapped_Paint(&textItem);
+}
+
+void Item_Text_Paint(itemDef_t *item)
+{
+ char text[1024];
+ const char *textPtr;
+ vec4_t color;
+
+ if (item->window.flags & WINDOW_WRAPPED)
+ {
+ Item_Text_Wrapped_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 (item->textStyle == ITEM_TEXTSTYLE_OUTLINED || item->textStyle == ITEM_TEXTSTYLE_OUTLINESHADOWED) {
-// Fade(&item->window.flags, &item->window.outlineColor[3], DC->Assets.fadeClamp, &item->window.nextTime, DC->Assets.fadeCycle, qfalse);
-// /*
-// Text_Paint(item->textRect.x-1, item->textRect.y-1, item->textscale, item->window.foreColor, textPtr, adjust);
-// Text_Paint(item->textRect.x, item->textRect.y-1, item->textscale, item->window.foreColor, textPtr, adjust);
-// Text_Paint(item->textRect.x+1, item->textRect.y-1, item->textscale, item->window.foreColor, textPtr, adjust);
-// Text_Paint(item->textRect.x-1, item->textRect.y, item->textscale, item->window.foreColor, textPtr, adjust);
-// Text_Paint(item->textRect.x+1, item->textRect.y, item->textscale, item->window.foreColor, textPtr, adjust);
-// Text_Paint(item->textRect.x-1, item->textRect.y+1, item->textscale, item->window.foreColor, textPtr, adjust);
-// Text_Paint(item->textRect.x, item->textRect.y+1, item->textscale, item->window.foreColor, textPtr, adjust);
-// Text_Paint(item->textRect.x+1, item->textRect.y+1, item->textscale, item->window.foreColor, textPtr, adjust);
-// */
-// DC->drawText(item->textRect.x - 1, item->textRect.y + 1, item->textscale * 1.02, item->window.outlineColor, textPtr, adjust);
-// }
-
- DC->drawText(item->textRect.x, item->textRect.y, item->textscale, color, textPtr, 0, 0, item->textStyle);
-}
-
-
-
-//float trap_Cvar_VariableValue( const char *var_name );
-//void trap_Cvar_VariableStringBuffer( const char *var_name, char *buffer, int bufsize );
-
-void Item_TextField_Paint(itemDef_t *item) {
- char buff[1024];
- vec4_t newColor;
- 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));*/
- //TA:
- memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
- } else {
- memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
- }
-
- offset = (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->textStyle);
- } else {
- DC->drawText(item->textRect.x + item->textRect.w + offset, item->textRect.y, item->textscale, newColor, buff + editPtr->paintOffset, 0, editPtr->maxPaintChars, item->textStyle);
- }
-
-}
-
-void Item_YesNo_Paint(itemDef_t *item) {
- vec4_t newColor;
- 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));*/
- //TA:
- memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
- } 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, 0, item->textStyle);
- } else {
- DC->drawText(item->textRect.x, item->textRect.y, item->textscale, newColor, (value != 0) ? "Yes" : "No", 0, 0, item->textStyle);
- }
-}
-
-void Item_Multi_Paint(itemDef_t *item) {
- vec4_t newColor;
- 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));*/
- //TA:
- memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
- } else {
- memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
- }
-
- text = Item_Multi_Setting(item);
-
- if (item->text) {
+ // this needs to go here as it sets extents for cvar types as well
+ Item_SetTextExtents(item, textPtr);
+
+ if (*textPtr == '\0')
+ return;
+
+ Item_TextColor(item, &color);
+
+ UI_Text_Paint(item->textRect.x, item->textRect.y, item->textscale, color, textPtr, 0, 0, item->textStyle);
+}
+
+void Item_TextField_Paint(itemDef_t *item)
+{
+ char buff[1024];
+ vec4_t newColor;
+ menuDef_t *parent;
+ int offset = (item->text && *item->text) ? ITEM_VALUE_OFFSET : 0;
+ editFieldDef_t *editPtr = item->typeData.edit;
+ char cursor = DC->getOverstrikeMode() ? '|' : '_';
+ qboolean editing = (item->window.flags & WINDOW_HASFOCUS && g_editingField);
+ const int cursorWidth = editing ? EDIT_CURSOR_WIDTH : 0;
+
+ // FIXME: causes duplicate printing if item->text is not set (NULL)
Item_Text_Paint(item);
- DC->drawText(item->textRect.x + item->textRect.w + 8, item->textRect.y, item->textscale, newColor, text, 0, 0, item->textStyle);
- } else {
- DC->drawText(item->textRect.x, item->textRect.y, item->textscale, newColor, text, 0, 0, item->textStyle);
- }
+
+ buff[0] = '\0';
+
+ if (item->cvar)
+ DC->getCVarString(item->cvar, buff, sizeof(buff));
+
+ // maxFieldWidth hasn't been set, so use the item's rect
+ if (editPtr->maxFieldWidth == 0)
+ {
+ editPtr->maxFieldWidth =
+ item->window.rect.w - (item->textRect.w + offset + (item->textRect.x - item->window.rect.x));
+
+ if (editPtr->maxFieldWidth < MIN_FIELD_WIDTH)
+ editPtr->maxFieldWidth = MIN_FIELD_WIDTH;
+ }
+
+ if (!editing)
+ editPtr->paintOffset = 0;
+
+ // Shorten string to max viewable
+ while (UI_Text_Width(buff + editPtr->paintOffset, item->textscale) > (editPtr->maxFieldWidth - cursorWidth) &&
+ strlen(buff) > 0)
+ buff[strlen(buff) - 1] = '\0';
+
+ parent = (menuDef_t *)item->parent;
+
+ if (item->window.flags & WINDOW_HASFOCUS)
+ memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
+ else
+ memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
+
+ if (editing)
+ {
+ UI_Text_PaintWithCursor(item->textRect.x + item->textRect.w + offset, item->textRect.y, item->textscale,
+ newColor, buff + editPtr->paintOffset, item->cursorPos - editPtr->paintOffset, cursor,
+ editPtr->maxPaintChars, item->textStyle);
+ }
+ else
+ {
+ UI_Text_Paint(item->textRect.x + item->textRect.w + offset, item->textRect.y, item->textscale, newColor,
+ buff + editPtr->paintOffset, 0, editPtr->maxPaintChars, item->textStyle);
+ }
}
+void Item_YesNo_Paint(itemDef_t *item)
+{
+ vec4_t newColor;
+ float value;
+ int offset;
+ menuDef_t *parent = (menuDef_t *)item->parent;
-typedef struct {
- char *command;
- int id;
- int defaultbind1;
- int defaultbind2;
- int bind1;
- int bind2;
-} bind_t;
+ value = (item->cvar) ? DC->getCVarValue(item->cvar) : 0;
-typedef struct
-{
- char* name;
- float defaultvalue;
- float value;
-} configcvar_t;
-
-
-static bind_t g_bindings[] =
-{
- { "+scores", K_TAB, -1, -1, -1 },
- { "+button2", K_ENTER, -1, -1, -1 },
- { "+speed", K_SHIFT, -1, -1, -1 },
- { "boost", 'x', -1, -1, -1 }, //TA: human sprinting
- { "+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_MOUSE1, -1, -1, -1 },
- { "+button5", K_MOUSE2, -1, -1, -1 }, //TA: secondary attack
- { "reload", 'r', -1, -1, -1 }, //TA: reload
- { "buy ammo", 'b', -1, -1, -1 }, //TA: buy ammo
- { "itemact medkit", 'm', -1, -1, -1 }, //TA: use medkit
- { "+button7", 'q', -1, -1, -1 }, //TA: buildable use
- { "deconstruct", 'e', -1, -1, -1 }, //TA: buildable destroy
- { "weapprev", '[', -1, -1, -1 },
- { "weapnext", ']', -1, -1, -1 },
- { "+button3", K_MOUSE3, -1, -1, -1 },
- { "+button4", K_MOUSE4, -1, -1, -1 },
- { "vote yes", K_F1, -1, -1, -1 },
- { "vote no", K_F2, -1, -1, -1 },
- { "teamvote yes", K_F3, -1, -1, -1 },
- { "teamvote no", K_F4, -1, -1, -1 },
- { "scoresUp", K_KP_PGUP, -1, -1, -1 },
- { "scoresDown", K_KP_PGDN, -1, -1, -1 },
- // bk001205 - this one below was: '-1'
- { "messagemode", -1, -1, -1, -1 },
- { "messagemode2", -1, -1, -1, -1 },
- { "messagemode3", -1, -1, -1, -1 },
- { "messagemode4", -1, -1, -1, -1 }
-};
+ if (item->window.flags & WINDOW_HASFOCUS)
+ memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
+ else
+ memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
+ offset = (item->text && *item->text) ? ITEM_VALUE_OFFSET : 0;
-static const int g_bindCount = sizeof(g_bindings) / sizeof(bind_t);
+ if (item->text)
+ {
+ Item_Text_Paint(item);
+ UI_Text_Paint(item->textRect.x + item->textRect.w + offset, item->textRect.y, item->textscale, newColor,
+ (value != 0) ? "Yes" : "No", 0, 0, item->textStyle);
+ }
+ else
+ UI_Text_Paint(item->textRect.x, item->textRect.y, item->textscale, newColor, (value != 0) ? "Yes" : "No", 0, 0,
+ item->textStyle);
+}
+
+void Item_Multi_Paint(itemDef_t *item)
+{
+ vec4_t newColor;
+ const char *text = "";
+ menuDef_t *parent = (menuDef_t *)item->parent;
+
+ if (item->window.flags & WINDOW_HASFOCUS)
+ memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
+ else
+ memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
+
+ text = Item_Multi_Setting(item);
+
+ if (item->text)
+ {
+ Item_Text_Paint(item);
+ UI_Text_Paint(item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET, item->textRect.y, item->textscale,
+ newColor, text, 0, 0, item->textStyle);
+ }
+ else
+ UI_Text_Paint(item->textRect.x, item->textRect.y, item->textscale, newColor, text, 0, 0, item->textStyle);
+}
+
+void Item_Cycle_Paint(itemDef_t *item)
+{
+ vec4_t newColor;
+ const char *text = "";
+ menuDef_t *parent = (menuDef_t *)item->parent;
+
+ if (item->window.flags & WINDOW_HASFOCUS)
+ memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
+ else
+ memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
+
+ if (item->typeData.cycle)
+ text = DC->feederItemText(item->feederID, item->typeData.cycle->cursorPos, 0, NULL);
+
+ if (item->text)
+ {
+ Item_Text_Paint(item);
+ UI_Text_Paint(item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET, item->textRect.y, item->textscale,
+ newColor, text, 0, 0, item->textStyle);
+ }
+ else
+ UI_Text_Paint(item->textRect.x, item->textRect.y, item->textscale, newColor, text, 0, 0, item->textStyle);
+}
+
+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, -1}, {"+button2", K_ENTER, -1, -1, -1, -1},
+ {"+speed", K_SHIFT, -1, -1, -1, -1}, {"+button6", 'z', -1, -1, -1, -1}, // human dodging
+ {"+button8", 'x', -1, -1, -1, -1}, {"+forward", K_UPARROW, -1, -1, -1, -1}, {"+back", K_DOWNARROW, -1, -1, -1, -1},
+ {"+moveleft", ',', -1, -1, -1, -1}, {"+moveright", '.', -1, -1, -1, -1}, {"+moveup", K_SPACE, -1, -1, -1, -1},
+ {"+movedown", 'c', -1, -1, -1, -1}, {"+left", K_LEFTARROW, -1, -1, -1, -1},
+ {"+right", K_RIGHTARROW, -1, -1, -1, -1}, {"+strafe", K_ALT, -1, -1, -1, -1}, {"+lookup", K_PGDN, -1, -1, -1, -1},
+ {"+lookdown", K_DEL, -1, -1, -1, -1}, {"+mlook", '/', -1, -1, -1, -1}, {"centerview", K_END, -1, -1, -1, -1},
+ {"+zoom", -1, -1, -1, -1, -1}, {"weapon 1", '1', -1, -1, -1, -1}, {"weapon 2", '2', -1, -1, -1, -1},
+ {"weapon 3", '3', -1, -1, -1, -1}, {"weapon 4", '4', -1, -1, -1, -1}, {"weapon 5", '5', -1, -1, -1, -1},
+ {"weapon 6", '6', -1, -1, -1, -1}, {"weapon 7", '7', -1, -1, -1, -1}, {"weapon 8", '8', -1, -1, -1, -1},
+ {"weapon 9", '9', -1, -1, -1, -1}, {"weapon 10", '0', -1, -1, -1, -1}, {"weapon 11", -1, -1, -1, -1, -1},
+ {"weapon 12", -1, -1, -1, -1, -1}, {"weapon 13", -1, -1, -1, -1, -1}, {"+attack", K_MOUSE1, -1, -1, -1, -1},
+ {"+button5", K_MOUSE2, -1, -1, -1, -1}, // secondary attack
+ {"reload", 'r', -1, -1, -1, -1}, // reload
+ {"buy ammo", 'b', -1, -1, -1, -1}, // buy ammo
+ {"itemact medkit", 'm', -1, -1, -1, -1}, // use medkit
+ {"+button7", 'q', -1, -1, -1, -1}, // buildable use
+ {"deconstruct", 'e', -1, -1, -1, -1}, // buildable destroy
+ {"weapprev", '[', -1, -1, -1, -1}, {"weapnext", ']', -1, -1, -1, -1}, {"+button3", K_MOUSE3, -1, -1, -1, -1},
+ {"+button4", K_MOUSE4, -1, -1, -1, -1}, {"vote yes", K_F1, -1, -1, -1, -1}, {"vote no", K_F2, -1, -1, -1, -1},
+ {"teamvote yes", K_F3, -1, -1, -1, -1}, {"teamvote no", K_F4, -1, -1, -1, -1},
+ {"scoresUp", K_KP_PGUP, -1, -1, -1, -1}, {"scoresDown", K_KP_PGDN, -1, -1, -1, -1},
+ {"screenshotJPEG", -1, -1, -1, -1, -1}, {"messagemode", -1, -1, -1, -1, -1}, {"messagemode2", -1, -1, -1, -1, -1}};
+
+static const size_t g_bindCount = ARRAY_LEN(g_bindings);
/*
=================
Controls_GetKeyAssignment
=================
*/
-static void Controls_GetKeyAssignment (char *command, int *twokeys)
+static void Controls_GetKeyAssignment(char *command, int *twokeys)
{
- int count;
- int j;
- char b[256];
+ int count;
+ int j;
+ char b[256];
- twokeys[0] = twokeys[1] = -1;
- count = 0;
+ 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;
- }
+ 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
+
+Iterate each command, get its numeric binding
=================
*/
-void Controls_GetConfig( void )
+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 );
+ size_t i;
+ int twokeys[2];
- g_bindings[ i ].bind1 = twokeys[ 0 ];
- g_bindings[ i ].bind2 = twokeys[ 1 ];
- }
+ for (i = 0; i < g_bindCount; i++)
+ {
+ Controls_GetKeyAssignment(g_bindings[i].command, twokeys);
- //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" ) );
+ g_bindings[i].bind1 = twokeys[0];
+ g_bindings[i].bind2 = twokeys[1];
+ }
}
/*
=================
Controls_SetConfig
+
+Iterate each command, get its numeric binding
=================
*/
void Controls_SetConfig(qboolean restart)
{
- int i;
+ unsigned int i;
- // iterate each command, get its numeric binding
- for (i=0; i < g_bindCount; i++)
- {
+ (void)restart;
- if (g_bindings[i].bind1 != -1)
+ for (i = 0; i < g_bindCount; i++)
{
- 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 (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");
- //trap_Cmd_ExecuteText( EXEC_APPEND, "in_restart\n" );
+ DC->executeText(EXEC_APPEND, "in_restart\n");
}
/*
=================
Controls_SetDefaults
+
+Iterate each command, set its default binding
=================
*/
-void Controls_SetDefaults( void )
+void Controls_SetDefaults(void)
{
- int i;
-
- // iterate each command, set its default binding
- for (i=0; i < g_bindCount; i++)
- {
- g_bindings[i].bind1 = g_bindings[i].defaultbind1;
- g_bindings[i].bind2 = g_bindings[i].defaultbind2;
- }
-
- //s_controls.invertmouse.curvalue = Controls_GetCvarDefault( "m_pitch" ) < 0;
- //s_controls.smoothmouse.curvalue = Controls_GetCvarDefault( "m_filter" );
- //s_controls.alwaysrun.curvalue = Controls_GetCvarDefault( "cl_run" );
- //s_controls.autoswitch.curvalue = Controls_GetCvarDefault( "cg_autoswitch" );
- //s_controls.sensitivity.curvalue = Controls_GetCvarDefault( "sensitivity" );
- //s_controls.joyenable.curvalue = Controls_GetCvarDefault( "in_joystick" );
- //s_controls.joythreshold.curvalue = Controls_GetCvarDefault( "joy_threshold" );
- //s_controls.freelook.curvalue = Controls_GetCvarDefault( "cl_freelook" );
+ unsigned int i;
+ for (i = 0; i < g_bindCount; i++)
+ {
+ g_bindings[i].bind1 = g_bindings[i].defaultbind1;
+ g_bindings[i].bind2 = g_bindings[i].defaultbind2;
+ }
}
-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;
+int BindingIDFromName(const char *name)
+{
+ size_t i;
+ for (i = 0; i < g_bindCount; i++)
+ {
+ if (Q_stricmp(name, g_bindings[i].command) == 0)
+ return i;
}
- }
- return -1;
+
+ return -1;
}
char g_nameBind1[32];
char g_nameBind2[32];
-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, " or " );
- strcat( g_nameBind1, g_nameBind2 );
- }
- return;
- }
- }
- strcpy(g_nameBind1, "???");
-}
-
-void Item_Slider_Paint(itemDef_t *item) {
- vec4_t newColor;
- 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));*/
- //TA:
- memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
- } 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, 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 );
-
-}
-
-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));*/
- //TA:
- memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
- } else {
- memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
- }
-
- 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, 0, maxChars, item->textStyle);
- } else {
- DC->drawText(item->textRect.x, item->textRect.y, item->textscale, newColor, (value != 0) ? "FIXME" : "FIXME", 0, maxChars, item->textStyle);
- }
-}
+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;
-qboolean Display_KeyBindPending( void ) {
- return g_waitingForKey;
+ 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, " or ");
+ strcat(g_nameBind1, g_nameBind2);
+ }
+
+ return;
+ }
+ }
+
+ strcpy(g_nameBind1, "???");
}
-qboolean Item_Bind_HandleKey(itemDef_t *item, int key, qboolean down) {
- int id;
- int i;
+void Item_Slider_Paint(itemDef_t *item)
+{
+ vec4_t newColor;
+ float x, y;
+
+ menuDef_t *parent = (menuDef_t *)item->parent;
+ float vScale = Item_Slider_VScale(item);
+
+ if (item->window.flags & WINDOW_HASFOCUS)
+ memcpy(newColor, &parent->focusColor, sizeof(vec4_t));
+ else
+ memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
- 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;
+ if (item->text)
+ {
+ Item_Text_Paint(item);
+ x = item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET;
+ y = item->textRect.y - item->textRect.h + ((item->textRect.h - (SLIDER_HEIGHT * vScale)) / 2.0f);
}
- return qtrue;
- }
- else
- {
- if (!g_waitingForKey || g_bindItem == NULL) {
- return qtrue;
+ else
+ {
+ x = item->window.rect.x;
+ y = item->window.rect.y;
}
- if (key & K_CHAR_FLAG) {
- return qtrue;
+ DC->setColor(newColor);
+ DC->drawHandlePic(x, y, SLIDER_WIDTH, SLIDER_HEIGHT * vScale, DC->Assets.sliderBar);
+
+ y = item->textRect.y - item->textRect.h + ((item->textRect.h - (SLIDER_THUMB_HEIGHT * vScale)) / 2.0f);
+
+ x = Item_Slider_ThumbPosition(item);
+ DC->drawHandlePic(
+ x - (SLIDER_THUMB_WIDTH / 2), y, SLIDER_THUMB_WIDTH, SLIDER_THUMB_HEIGHT * vScale, DC->Assets.sliderThumb);
+}
+
+void Item_Bind_Paint(itemDef_t *item)
+{
+ vec4_t newColor, lowLight;
+ float value;
+ int maxChars = 0;
+ menuDef_t *parent = (menuDef_t *)item->parent;
+
+ if (item->typeData.edit)
+ maxChars = item->typeData.edit->maxPaintChars;
+
+ value = (item->cvar) ? DC->getCVarValue(item->cvar) : 0;
+
+ if (item->window.flags & WINDOW_HASFOCUS)
+ {
+ if (g_bindItem == item)
+ {
+ 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
+ memcpy(&newColor, &parent->focusColor, sizeof(vec4_t));
}
+ else
+ memcpy(&newColor, &item->window.foreColor, sizeof(vec4_t));
- switch (key)
+ if (item->text)
{
- case K_ESCAPE:
- g_waitingForKey = qfalse;
- return qtrue;
+ Item_Text_Paint(item);
- case K_BACKSPACE:
- id = BindingIDFromName(item->cvar);
- if (id != -1) {
- g_bindings[id].bind1 = -1;
- g_bindings[id].bind2 = -1;
+ if (g_bindItem == item && g_waitingForKey)
+ {
+ UI_Text_Paint(item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET, item->textRect.y, item->textscale,
+ newColor, "Press key", 0, maxChars, item->textStyle);
+ }
+ else
+ {
+ BindingFromName(item->cvar);
+ UI_Text_Paint(item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET, item->textRect.y, item->textscale,
+ newColor, g_nameBind1, 0, maxChars, item->textStyle);
+ }
+ }
+ else
+ UI_Text_Paint(item->textRect.x, item->textRect.y, item->textscale, newColor, (value != 0) ? "FIXME" : "FIXME",
+ 0, maxChars, item->textStyle);
+}
+
+qboolean Display_KeyBindPending(void) { return g_waitingForKey; }
+
+qboolean Item_Bind_HandleKey(itemDef_t *item, int key, qboolean down)
+{
+ int id;
+
+ 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;
}
- Controls_SetConfig(qtrue);
- g_waitingForKey = qfalse;
- g_bindItem = NULL;
- return qtrue;
- case '`':
return qtrue;
}
- }
+ else
+ {
+ if (!g_waitingForKey || g_bindItem == NULL)
+ return qtrue;
- if (key != -1)
- {
+ if (key & K_CHAR_FLAG)
+ return qtrue;
- for (i=0; i < g_bindCount; i++)
- {
+ switch (key)
+ {
+ case K_ESCAPE:
+ g_waitingForKey = qfalse;
+ return qtrue;
+
+ case K_BACKSPACE:
+ id = BindingIDFromName(item->cvar);
+
+ if (id != -1)
+ {
+ g_bindings[id].bind1 = -1;
+ g_bindings[id].bind2 = -1;
+ }
+
+ Controls_SetConfig(qtrue);
+ g_waitingForKey = qfalse;
+ g_bindItem = NULL;
+ return qtrue;
+
+ case '`':
+ return qtrue;
+ }
+ }
- if (g_bindings[i].bind2 == key) {
- g_bindings[i].bind2 = -1;
- }
+ if (key != -1)
+ {
+ unsigned int i;
+ for (i = 0; i < g_bindCount; i++)
+ {
+ if (g_bindings[i].bind2 == key)
+ g_bindings[i].bind2 = -1;
- if (g_bindings[i].bind1 == key)
- {
- g_bindings[i].bind1 = g_bindings[i].bind2;
- g_bindings[i].bind2 = -1;
- }
+ if (g_bindings[i].bind1 == key)
+ {
+ g_bindings[i].bind1 = g_bindings[i].bind2;
+ g_bindings[i].bind2 = -1;
+ }
+ }
}
- }
+ id = BindingIDFromName(item->cvar);
- 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 (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;
+ 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;
+ }
}
- else if (g_bindings[id].bind1 != key && g_bindings[id].bind2 == -1) {
- g_bindings[id].bind2 = key;
+
+ Controls_SetConfig(qtrue);
+ g_waitingForKey = qfalse;
+
+ return qtrue;
+}
+
+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 = item->typeData.model;
+ 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;
+
+ UI_AdjustFrom640(&x, &y, &w, &h);
+
+ 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 {
- DC->setBinding( g_bindings[id].bind1, "" );
- DC->setBinding( g_bindings[id].bind2, "" );
- g_bindings[id].bind1 = key;
- g_bindings[id].bind2 = -1;
+ else
+ origin[0] = item->textscale;
+
+ refdef.fov_x = (modelPtr->fov_x) ? modelPtr->fov_x : w;
+ refdef.fov_y = (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 );
+
+ 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;
+ }
}
- }
- Controls_SetConfig(qtrue);
- g_waitingForKey = qfalse;
+ 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);
- return qtrue;
+ DC->addRefEntityToScene(&ent);
+ DC->renderScene(&refdef);
}
+void Item_ListBoxRow_Paint(itemDef_t *item, int row, int renderPos, qboolean highlight, qboolean scrollbar)
+{
+ float x, y, w;
+ listBoxDef_t *listPtr = item->typeData.list;
+ menuDef_t *menu = (menuDef_t *)item->parent;
+ float one, two;
+ one = 1.0f * DC->aspectScale;
+ two = 2.0f * DC->aspectScale;
-void AdjustFrom640(float *x, float *y, float *w, float *h) {
- //*x = *x * DC->scale + DC->bias;
- *x *= DC->xscale;
- *y *= DC->yscale;
- *w *= DC->xscale;
- *h *= DC->yscale;
-}
+ x = SCROLLBAR_X(item);
+ y = SCROLLBAR_Y(item) + (listPtr->elementHeight * renderPos);
+ w = item->window.rect.w - (two * item->window.borderSize);
-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 (scrollbar)
+ w -= SCROLLBAR_ARROW_WIDTH;
- if (modelPtr == NULL) {
- return;
- }
+ if (listPtr->elementStyle == LISTBOX_IMAGE)
+ {
+ qhandle_t image = DC->feederItemImage(item->feederID, row);
- // 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;
+ UI_SetClipRegion(x, y, listPtr->elementWidth, listPtr->elementHeight);
- AdjustFrom640( &x, &y, &w, &h );
+ if (image)
+ DC->drawHandlePic(x + one, y + 1.0f, listPtr->elementWidth - two, listPtr->elementHeight - 2.0f, image);
- refdef.x = x;
- refdef.y = y;
- refdef.width = w;
- refdef.height = h;
+ if (highlight && row == item->cursorPos)
+ {
+ DC->drawRect(
+ x, y, listPtr->elementWidth, listPtr->elementHeight, item->window.borderSize, item->window.borderColor);
+ }
- DC->modelBounds( item->asset, mins, maxs );
+ UI_ClearClipRegion();
+ }
+ else
+ {
+ const float m = UI_Text_EmHeight(item->textscale);
+ char text[MAX_STRING_CHARS];
+ qhandle_t optionalImage;
- origin[2] = -0.5 * ( mins[2] + maxs[2] );
- origin[1] = 0.5 * ( mins[1] + maxs[1] );
+ if (listPtr->numColumns > 0)
+ {
+ int j;
+
+ for (j = 0; j < listPtr->numColumns; j++)
+ {
+ float columnPos;
+ float width, height, yOffset;
+
+ if (menu->window.aspectBias != ASPECT_NONE || item->window.aspectBias != ASPECT_NONE)
+ {
+ columnPos = (listPtr->columnInfo[j].pos + 4.0f) * DC->aspectScale;
+ width = listPtr->columnInfo[j].width * DC->aspectScale;
+ }
+ else
+ {
+ columnPos = (listPtr->columnInfo[j].pos + 4.0f);
+ width = listPtr->columnInfo[j].width;
+ }
+
+ height = listPtr->columnInfo[j].width;
+ yOffset = y + ((listPtr->elementHeight - height) / 2.0f);
+
+ Q_strncpyz(text, DC->feederItemText(item->feederID, row, j, &optionalImage), sizeof(text));
+
+ UI_SetClipRegion(x + columnPos, yOffset, width, height);
+
+ if (optionalImage >= 0)
+ DC->drawHandlePic(x + columnPos, yOffset, width, height, optionalImage);
+ else if (text[0])
+ {
+ float alignOffset = 0.0f, tw;
+
+ tw = UI_Text_Width(text, item->textscale);
+
+ switch (listPtr->columnInfo[j].align)
+ {
+ case ALIGN_LEFT:
+ alignOffset = 0.0f;
+ break;
+
+ case ALIGN_RIGHT:
+ alignOffset = width - tw;
+ break;
+
+ case ALIGN_CENTER:
+ alignOffset = (width / 2.0f) - (tw / 2.0f);
+ break;
+
+ default:
+ alignOffset = 0.0f;
+ }
+
+ UI_Text_Paint(x + columnPos + alignOffset, y + m + ((listPtr->elementHeight - m) / 2.0f),
+ item->textscale, item->window.foreColor, text, 0, 0, item->textStyle);
+ }
+
+ UI_ClearClipRegion();
+ }
+ }
+ else
+ {
+ float offset;
- // 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_y = (modelPtr->fov_y) ? modelPtr->fov_y : h;
+ if (menu->window.aspectBias != ASPECT_NONE || item->window.aspectBias != ASPECT_NONE)
+ offset = 4.0f * DC->aspectScale;
+ else
+ offset = 4.0f;
- //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 );
+ Q_strncpyz(text, DC->feederItemText(item->feederID, row, 0, &optionalImage), sizeof(text));
- DC->clearScene();
+ UI_SetClipRegion(x, y, w, listPtr->elementHeight);
- refdef.time = DC->realTime;
+ if (optionalImage >= 0)
+ DC->drawHandlePic(x + offset, y, listPtr->elementHeight, listPtr->elementHeight, optionalImage);
+ else if (text[0])
+ {
+ UI_Text_Paint(x + offset, y + m + ((listPtr->elementHeight - m) / 2.0f), item->textscale,
+ item->window.foreColor, text, 0, 0, item->textStyle);
+ }
+
+ UI_ClearClipRegion();
+ }
- // add the model
+ if (highlight && row == item->cursorPos)
+ DC->fillRect(x, y, w, listPtr->elementHeight, item->window.outlineColor);
+ }
+}
- memset( &ent, 0, sizeof(ent) );
+void Item_ListBox_Paint(itemDef_t *item)
+{
+ float size;
+ int i;
+ listBoxDef_t *listPtr = item->typeData.list;
+ int count = DC->feederCount(item->feederID);
+ qboolean scrollbar = !(listPtr->noscrollbar && count > Item_ListBox_NumItemsForItemHeight(item));
+ if (scrollbar)
+ {
+ float x = SCROLLBAR_SLIDER_X(item);
+ float y = SCROLLBAR_Y(item);
+ float thumbY = Item_ListBox_ThumbDrawPosition(item);
- //adjust = 5.0 * sin( (float)uis.realtime / 500 );
- //adjust = 360 % (int)((float)uis.realtime / 1000);
- //VectorSet( angles, 0, 0, 1 );
+ // Up arrow
+ DC->drawHandlePic(x, y, SCROLLBAR_ARROW_WIDTH, SCROLLBAR_ARROW_HEIGHT, DC->Assets.scrollBarArrowUp);
+ y = SCROLLBAR_SLIDER_Y(item);
- // 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;
+ // Scroll bar
+ size = SCROLLBAR_SLIDER_HEIGHT(item);
+ DC->drawHandlePic(x, y, SCROLLBAR_ARROW_WIDTH, size, DC->Assets.scrollBar);
+ y = SCROLLBAR_SLIDER_Y(item) + size;
+
+ // Down arrow
+ DC->drawHandlePic(x, y, SCROLLBAR_ARROW_WIDTH, SCROLLBAR_ARROW_HEIGHT, DC->Assets.scrollBarArrowDown);
+
+ // Thumb
+ DC->drawHandlePic(x, thumbY, SCROLLBAR_ARROW_WIDTH, SCROLLBAR_ARROW_HEIGHT, DC->Assets.scrollBarThumb);
}
- }
- 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 );
+ // Paint rows
+ for (i = listPtr->startPos; i < listPtr->endPos; i++)
+ Item_ListBoxRow_Paint(item, i, i - listPtr->startPos, qtrue, scrollbar);
+}
+
+void Item_Paint(itemDef_t *item);
+
+void Item_ComboBox_Paint(itemDef_t *item)
+{
+ float x, y, h;
- DC->addRefEntityToScene( &ent );
- DC->renderScene( &refdef );
+ x = SCROLLBAR_SLIDER_X(item);
+ y = SCROLLBAR_Y(item);
+ h = item->window.rect.h - 2.0f;
+ // Down arrow
+ DC->drawHandlePic(x, y, SCROLLBAR_ARROW_WIDTH, h, DC->Assets.scrollBarArrowDown);
+ Item_ListBoxRow_Paint(item, item->cursorPos, 0, qfalse, qtrue);
+ if (g_comboBoxItem != NULL)
+ {
+ qboolean cast = Item_ComboBox_MaybeCastToListBox(item);
+ Item_Paint(item);
+ Item_ComboBox_MaybeUnCastFromListBox(item, cast);
+ }
}
+void Item_ListBox_Update(itemDef_t *item)
+{
+ listBoxDef_t *listPtr = item->typeData.list;
+ int feederCount = DC->feederCount(item->feederID);
+
+ if (listPtr->lastFeederCount != feederCount)
+ {
+ if (listPtr->resetonfeederchange)
+ {
+ item->cursorPos = DC->feederInitialise(item->feederID);
+ Item_ListBox_SetStartPos(item, 0);
+ DC->feederSelection(item->feederID, item->cursorPos);
+ }
+ else
+ {
+ // Make sure endPos is up-to-date
+ Item_ListBox_SetStartPos(item, listPtr->startPos);
-void Item_Image_Paint(itemDef_t *item) {
- if (item == NULL) {
- return;
- }
- DC->drawHandlePic(item->window.rect.x+1, item->window.rect.y+1, item->window.rect.w-2, item->window.rect.h-2, item->asset);
+ // If the selection is off the end now, select the last element
+ if (item->cursorPos >= feederCount)
+ item->cursorPos = feederCount - 1;
+ }
+ }
+
+ listPtr->lastFeederCount = feederCount;
}
-void Item_ListBox_Paint(itemDef_t *item) {
- float x, y, size, thumb;
- int i, count;
- 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);
- }
-
- listPtr->endPos++;
- size -= listPtr->elementWidth;
- if (size < listPtr->elementWidth) {
- listPtr->drawPadding = size; //listPtr->elementWidth - size;
- break;
- }
- x += listPtr->elementWidth;
- // 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;
- 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) {
- //TA:
- int alignOffset = 0.0f, tw;
-
- tw = DC->textWidth( text, item->textscale, 0 );
-
- switch( listPtr->columnInfo[ j ].align )
- {
- case ITEM_ALIGN_LEFT:
- alignOffset = 0.0f;
- break;
-
- case ITEM_ALIGN_RIGHT:
- alignOffset = listPtr->columnInfo[ j ].width - tw;
- break;
-
- case ITEM_ALIGN_CENTER:
- alignOffset = ( listPtr->columnInfo[ j ].width / 2.0f ) - ( tw / 2.0f );
- break;
+void Item_OwnerDraw_Paint(itemDef_t *item)
+{
+ menuDef_t *parent;
+ const char *text;
- default:
- alignOffset = 0.0f;
- }
+ if (item == NULL)
+ return;
+
+ parent = (menuDef_t *)item->parent;
+
+ if (DC->ownerDrawItem)
+ {
+ vec4_t color, lowLight;
+ 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));
- DC->drawText( x + 4 + listPtr->columnInfo[j].pos + alignOffset, y + listPtr->elementHeight,
- item->textscale, item->window.foreColor, text, 0,
- listPtr->columnInfo[j].maxChars, item->textStyle );
+ 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;
+ }
}
- }
- } 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, 0, item->textStyle);
- }
}
- if (i == item->cursorPos) {
- DC->fillRect(x + 2, y + 2, item->window.rect.w - SCROLLBAR_SIZE - 4, listPtr->elementHeight, item->window.outlineColor);
- }
-
- listPtr->endPos++;
- size -= listPtr->elementHeight;
- if (size < listPtr->elementHeight) {
- listPtr->drawPadding = listPtr->elementHeight - size;
- break;
- }
- y += listPtr->elementHeight;
- // fit++;
- }
+ if (item->window.flags & WINDOW_HASFOCUS)
+ memcpy(color, &parent->focusColor, sizeof(vec4_t));
+ 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))
+ Com_Memcpy(color, parent->disableColor, sizeof(vec4_t));
+
+ if (DC->ownerDrawText && (text = DC->ownerDrawText(item->window.ownerDraw)))
+ {
+ if (item->text && *item->text)
+ {
+ Item_Text_Paint(item);
+
+ UI_Text_Paint(item->textRect.x + item->textRect.w + ITEM_VALUE_OFFSET, item->textRect.y,
+ item->textscale, color, text, 0, 0, item->textStyle);
+ }
+ else
+ {
+ item->text = text;
+ Item_Text_Paint(item);
+ item->text = NULL;
+ }
+ }
+ 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->textalignment, item->textvalignment, item->window.borderSize, item->textscale,
+ color, item->window.backColor, item->window.background, item->textStyle);
+ }
}
- }
+}
- //TA: FIXME: hacky fix to off-by-one bug
- listPtr->endPos--;
-}
-
-
-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));*/
- //TA:
- memcpy(color, &parent->focusColor, sizeof(vec4_t));
- } 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)); // bk001207 - FIXME: Com_Memcpy
- }
-
- 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 );
- } 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 );
- }
- } 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 );
- }
- }
-}
-
-
-void Item_Paint(itemDef_t *item) {
- 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_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 = 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 (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_SAYFIELD:
- 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_IMAGE:
- // Item_Image_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;
- }
-
-}
-
-void Menu_Init(menuDef_t *menu) {
- memset(menu, 0, sizeof(menuDef_t));
- menu->cursorItem = -1;
- menu->fadeAmount = DC->Assets.fadeAmount;
- menu->fadeClamp = DC->Assets.fadeClamp;
- menu->fadeCycle = DC->Assets.fadeCycle;
- Window_Init(&menu->window);
-}
-
-itemDef_t *Menu_GetFocusedItem(menuDef_t *menu) {
- int i;
- if (menu) {
- for (i = 0; i < menu->itemCount; i++) {
- if (menu->items[i]->window.flags & WINDOW_HASFOCUS) {
- return menu->items[i];
- }
- }
- }
- return NULL;
-}
-
-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;
-}
-
-void Menu_ScrollFeeder(menuDef_t *menu, int feeder, qboolean down) {
- if (menu) {
- int i;
- for (i = 0; i < menu->itemCount; i++) {
- if (menu->items[i]->special == feeder) {
- Item_ListBox_HandleKey(menu->items[i], (down) ? K_DOWNARROW : K_UPARROW, qtrue, qtrue);
+void Item_Update(itemDef_t *item)
+{
+ if (item == NULL)
return;
- }
- }
- }
+
+ if (Item_IsListBox(item))
+ Item_ListBox_Update(item);
}
+void Item_Paint(itemDef_t *item)
+{
+ vec4_t red;
+ menuDef_t *parent = (menuDef_t *)item->parent;
+ red[0] = red[3] = 1;
+ red[1] = red[2] = 0;
+ if (item == NULL)
+ return;
-void Menu_SetFeederSelection(menuDef_t *menu, int feeder, int index, const char *name) {
- if (menu == NULL) {
- if (name == NULL) {
- menu = Menu_GetFocused();
- } else {
- menu = Menus_FindByName(name);
+ 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 = 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 (menu) {
- int i;
- for (i = 0; i < menu->itemCount; i++) {
- if (menu->items[i]->special == feeder) {
- if (index == 0) {
- listBoxDef_t *listPtr = (listBoxDef_t*)menu->items[i]->typeData;
- listPtr->cursorPos = 0;
- listPtr->startPos = 0;
- }
- menu->items[i]->cursorPos = index;
- DC->feederSelection(menu->items[i]->special, menu->items[i]->cursorPos);
+ 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;
- }
+
+ Window_Paint(&item->window, parent->fadeAmount, parent->fadeClamp, parent->fadeCycle);
+
+ if (DC->getCVarValue("ui_developer"))
+ {
+ 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);
}
- }
+
+ 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_CYCLE:
+ Item_Cycle_Paint(item);
+ break;
+
+ case ITEM_TYPE_LISTBOX:
+ Item_ListBox_Paint(item);
+ break;
+
+ case ITEM_TYPE_COMBOBOX:
+ Item_ComboBox_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:
+ if (Item_IsEditField(item))
+ Item_TextField_Paint(item);
+
+ break;
+ }
+
+ Border_Paint(&item->window);
+}
+
+void Menu_Init(menuDef_t *menu)
+{
+ memset(menu, 0, sizeof(menuDef_t));
+ menu->cursorItem = -1;
+ menu->fadeAmount = DC->Assets.fadeAmount;
+ menu->fadeClamp = DC->Assets.fadeClamp;
+ menu->fadeCycle = DC->Assets.fadeCycle;
+ Window_Init(&menu->window);
+ menu->window.aspectBias = ALIGN_CENTER;
}
-qboolean Menus_AnyFullScreenVisible( void ) {
- int i;
- for (i = 0; i < menuCount; i++) {
- if (Menus[i].window.flags & WINDOW_VISIBLE && Menus[i].fullScreen) {
- return qtrue;
+itemDef_t *Menu_GetFocusedItem(menuDef_t *menu)
+{
+ int i;
+
+ if (menu)
+ {
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ if (menu->items[i]->window.flags & WINDOW_HASFOCUS)
+ return menu->items[i];
+ }
}
- }
- return qfalse;
+
+ return NULL;
}
-menuDef_t *Menus_ActivateByName(const char *p) {
- int i, j;
- menuDef_t *m = NULL;
- menuDef_t *focus = Menu_GetFocused();
+menuDef_t *Menu_GetFocused(void)
+{
+ int i;
- for (i = 0; i < menuCount; i++) {
- if (Q_stricmp(Menus[i].window.name, p) == 0) {
- m = &Menus[i];
- Menus_Activate(m);
- Menu_HandleMouseMove( m, DC->cursorx, DC->cursory ); //TA: force the item under the cursor to focus
+ for (i = 0; i < menuCount; i++)
+ {
+ if (Menus[i].window.flags & WINDOW_HASFOCUS && Menus[i].window.flags & WINDOW_VISIBLE)
+ return &Menus[i];
+ }
- for( j = 0; j < m->itemCount; j++ ) //TA: reset selection in listboxes when opened
- {
- if( m->items[ j ]->type == ITEM_TYPE_LISTBOX )
+ return NULL;
+}
+
+void Menu_ScrollFeeder(menuDef_t *menu, int feeder, qboolean down)
+{
+ if (menu)
+ {
+ int i;
+
+ for (i = 0; i < menu->itemCount; i++)
{
- listBoxDef_t *listPtr = (listBoxDef_t*)m->items[ j ]->typeData;
- m->items[ j ]->cursorPos = 0;
- listPtr->startPos = 0;
- DC->feederSelection( m->items[ j ]->special, 0 );
+ itemDef_t *item = menu->items[i];
+
+ if (item->feederID == feeder)
+ {
+ qboolean cast = Item_ComboBox_MaybeCastToListBox(item);
+ Item_ListBox_HandleKey(item, down ? K_DOWNARROW : K_UPARROW, qtrue, qtrue);
+ Item_ComboBox_MaybeUnCastFromListBox(item, cast);
+
+ return;
+ }
}
- }
+ }
+}
- if (openMenuCount < MAX_OPEN_MENUS && focus != NULL) {
- menuStack[openMenuCount++] = focus;
- }
- } else {
- Menus[i].window.flags &= ~WINDOW_HASFOCUS;
+void Menu_SetFeederSelection(menuDef_t *menu, int feeder, int index, const char *name)
+{
+ if (menu == NULL)
+ {
+ if (name == NULL)
+ menu = Menu_GetFocused();
+ else
+ menu = Menus_FindByName(name);
+ }
+
+ if (menu)
+ {
+ int i;
+
+ for (i = 0; i < menu->itemCount; i++)
+ {
+ if (menu->items[i]->feederID == feeder)
+ {
+ if (Item_IsListBox(menu->items[i]) && index == 0)
+ {
+ menu->items[i]->typeData.list->cursorPos = 0;
+ Item_ListBox_SetStartPos(menu->items[i], 0);
+ }
+
+ menu->items[i]->cursorPos = index;
+ DC->feederSelection(menu->items[i]->feederID, menu->items[i]->cursorPos);
+ return;
+ }
+ }
}
- }
- Display_CloseCinematics();
- return m;
}
+qboolean Menus_AnyFullScreenVisible(void)
+{
+ int i;
+
+ for (i = 0; i < menuCount; i++)
+ {
+ if (Menus[i].window.flags & WINDOW_VISIBLE && Menus[i].fullScreen)
+ return qtrue;
+ }
-void Item_Init(itemDef_t *item) {
- memset(item, 0, sizeof(itemDef_t));
- item->textscale = 0.55f;
- Window_Init(&item->window);
+ return qfalse;
}
-void Menu_HandleMouseMove(menuDef_t *menu, float x, float y) {
- int i, pass;
- qboolean focusSet = qfalse;
+menuDef_t *Menus_ActivateByName(const char *p)
+{
+ int i;
+ menuDef_t *m = NULL;
- itemDef_t *overItem;
- if (menu == NULL) {
- return;
- }
+ // Activate one menu
- if (!(menu->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED))) {
- return;
- }
+ for (i = 0; i < menuCount; i++)
+ {
+ if (Q_stricmp(Menus[i].window.name, p) == 0)
+ {
+ m = &Menus[i];
+ Menus_Activate(m);
+ break;
+ }
+ }
- if (itemCapture) {
- //Item_MouseMove(itemCapture, x, y);
- return;
- }
+ // Defocus the others
+ for (i = 0; i < menuCount; i++)
+ {
+ if (Q_stricmp(Menus[i].window.name, p) != 0)
+ Menus[i].window.flags &= ~WINDOW_HASFOCUS;
+ }
- if (g_waitingForKey || g_editingField) {
- return;
- }
+ return m;
+}
- // 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;
+menuDef_t *Menus_ReplaceActiveByName(const char *p)
+{
+ int i;
+ menuDef_t *m = NULL;
- if (!(menu->items[i]->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED))) {
- continue;
- }
+ // Activate one menu
- // 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;
- }
+ for (i = 0; i < menuCount; i++)
+ {
+ if (Q_stricmp(Menus[i].window.name, p) == 0)
+ {
+ m = &Menus[i];
+ if (!Menus_ReplaceActive(m))
+ return NULL;
+ break;
+ }
+ }
+ return m;
+}
- if (menu->items[i]->cvarFlags & (CVAR_SHOW | CVAR_HIDE) && !Item_EnableShowViaCvar(menu->items[i], CVAR_SHOW)) {
- continue;
- }
+void Item_Init(itemDef_t *item)
+{
+ memset(item, 0, sizeof(itemDef_t));
+ item->textscale = 0.55f;
+ Window_Init(&item->window);
+ item->window.aspectBias = ASPECT_NONE;
+}
+static qboolean Item_HandleMouseMove(itemDef_t *item, float x, float y, int pass, qboolean focusSet)
+{
+ if (Rect_ContainsPoint(&item->window.rect, x, y))
+ {
+ if (pass == 1)
+ {
+ if (item->type == ITEM_TYPE_TEXT && item->text)
+ {
+ if (!Rect_ContainsPoint(Item_CorrectedTextRect(item), x, y))
+ return qtrue;
+ }
+ // if we are over an item
+ if (IsVisible(item->window.flags))
+ {
+ // different one
+ Item_MouseEnter(item, x, y);
- 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 (!focusSet)
+ focusSet = Item_SetFocus(item, x, y);
}
- }
- // 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);
+ }
+
+ return qtrue;
+ }
+
+ return qfalse;
+}
+
+void Menu_HandleMouseMove(menuDef_t *menu, float x, float y)
+{
+ int i, pass;
+ qboolean focusSet = qfalse;
+ qboolean result;
+ qboolean cast;
+
+ 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;
+
+ if (g_comboBoxItem != NULL)
+ {
+ Item_SetFocus(g_comboBoxItem, x, y);
+ focusSet = qtrue;
+ }
+
+ // 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++)
+ {
+ itemDef_t *item = menu->items[i];
+
+ // turn off focus each item
+ // menu->items[i].window.flags &= ~WINDOW_HASFOCUS;
+
+ if (!(item->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED)))
+ continue;
+
+ // items can be enabled and disabled based on cvars
+ if (item->cvarFlags & (CVAR_ENABLE | CVAR_DISABLE) && !Item_EnableShowViaCvar(item, CVAR_ENABLE))
+ continue;
+
+ if (item->cvarFlags & (CVAR_SHOW | CVAR_HIDE) && !Item_EnableShowViaCvar(item, CVAR_SHOW))
+ continue;
+
+ cast = Item_ComboBox_MaybeCastToListBox(item);
+ result = Item_HandleMouseMove(item, x, y, pass, focusSet);
+ Item_ComboBox_MaybeUnCastFromListBox(item, cast);
+
+ if (!result && item->window.flags & WINDOW_MOUSEOVER)
+ {
+ Item_MouseLeave(item);
+ Item_SetMouseOver(item, qfalse);
}
- }
}
- } else if (menu->items[i]->window.flags & WINDOW_MOUSEOVER) {
- Item_MouseLeave(menu->items[i]);
- Item_SetMouseOver(menu->items[i], qfalse);
- }
}
- }
+}
+void Menu_Update(menuDef_t *menu)
+{
+ int i;
+
+ if (menu == NULL)
+ return;
+
+ for (i = 0; i < menu->itemCount; i++)
+ Item_Update(menu->items[i]);
}
-void Menu_Paint(menuDef_t *menu, qboolean forcePaint) {
- int i;
+void Menu_Paint(menuDef_t *menu, qboolean forcePaint)
+{
+ int i;
- if (menu == NULL) {
- return;
- }
+ if (menu == NULL)
+ return;
- if (!(menu->window.flags & WINDOW_VISIBLE) && !forcePaint) {
- return;
- }
+ if (!(menu->window.flags & WINDOW_VISIBLE) && !forcePaint)
+ return;
- if (menu->window.ownerDrawFlags && DC->ownerDrawVisible && !DC->ownerDrawVisible(menu->window.ownerDrawFlags)) {
- return;
- }
+ if (menu->window.ownerDrawFlags && DC->ownerDrawVisible && !DC->ownerDrawVisible(menu->window.ownerDrawFlags))
+ return;
- if (forcePaint) {
- menu->window.flags |= WINDOW_FORCED;
- }
+ if (forcePaint)
+ menu->window.flags |= WINDOW_FORCED;
- // draw the background if necessary
- if (menu->fullScreen) {
- // implies a background shader
- // FIXME: make sure we have a default shader if fullscreen is set with no background
- 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);
- }
+ // draw the background if necessary
+ if (menu->fullScreen)
+ {
+ // implies a background shader
+ // FIXME: make sure we have a default shader if fullscreen is set with no background
+ DC->drawHandlePic(0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, menu->window.background);
+ }
- // paint the background and or border
- Window_Paint(&menu->window, menu->fadeAmount, menu->fadeClamp, menu->fadeCycle );
+ // paint the background and or border
+ Window_Paint(&menu->window, menu->fadeAmount, menu->fadeClamp, menu->fadeCycle);
- for (i = 0; i < menu->itemCount; i++) {
- Item_Paint(menu->items[i]);
- }
+ Border_Paint(&menu->window);
- if (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);
- }
+ for (i = 0; i < menu->itemCount; i++)
+ Item_Paint(menu->items[i]);
+
+ if (DC->getCVarValue("ui_developer"))
+ {
+ 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_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_SAYFIELD || 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 || item->type == ITEM_TYPE_SAYFIELD) {
- 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));
- }
+ /*
+ ===============
+ Keyword Hash
+ ===============
+ */
+
+#define KEYWORDHASH_SIZE 512
+
+typedef struct keywordHash_s {
+ char *keyword;
+ qboolean (*func)(itemDef_t *item, int handle);
+ int param;
+ struct keywordHash_s *next;
+} keywordHash_t;
+
+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;
+}
+
+void KeywordHash_Add(keywordHash_t *table[], keywordHash_t *key)
+{
+ int hash;
+
+ hash = KeywordHash_Key(key->keyword);
+ /*
+ if(table[hash]) int collision = qtrue;
+ */
+ key->next = table[hash];
+ table[hash] = key;
+}
+
+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;
}
/*
===============
-Keyword Hash
+Item_DataType
+
+Give a numeric representation of which typeData union element this item uses
===============
*/
+itemDataType_t Item_DataType(itemDef_t *item)
+{
+ switch (item->type)
+ {
+ default:
+ case ITEM_TYPE_NONE:
+ return TYPE_NONE;
-#define KEYWORDHASH_SIZE 512
+ case ITEM_TYPE_LISTBOX:
+ case ITEM_TYPE_COMBOBOX:
+ return TYPE_LIST;
-typedef struct keywordHash_s
-{
- char *keyword;
- qboolean (*func)(itemDef_t *item, int handle);
- struct keywordHash_s *next;
-} keywordHash_t;
+ case ITEM_TYPE_CYCLE:
+ return TYPE_COMBO;
-int KeywordHash_Key(char *keyword) {
- int register hash, i;
+ case ITEM_TYPE_EDITFIELD:
+ case ITEM_TYPE_NUMERICFIELD:
+ case ITEM_TYPE_SAYFIELD:
+ case ITEM_TYPE_YESNO:
+ case ITEM_TYPE_BIND:
+ case ITEM_TYPE_SLIDER:
+ case ITEM_TYPE_TEXT:
+ return TYPE_EDIT;
- 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;
-}
+ case ITEM_TYPE_MULTI:
+ return TYPE_MULTI;
-void KeywordHash_Add(keywordHash_t *table[], keywordHash_t *key) {
- int hash;
+ case ITEM_TYPE_MODEL:
+ return TYPE_MODEL;
+ }
+}
- hash = KeywordHash_Key(key->keyword);
/*
- if (table[hash]) {
- int collision = qtrue;
- }
+===============
+Item_IsEditField
+===============
*/
- key->next = table[hash];
- table[hash] = key;
+static ID_INLINE qboolean Item_IsEditField(itemDef_t *item)
+{
+ switch (item->type)
+ {
+ case ITEM_TYPE_EDITFIELD:
+ case ITEM_TYPE_NUMERICFIELD:
+ case ITEM_TYPE_SAYFIELD:
+ return qtrue;
+
+ default:
+ return qfalse;
+ }
}
-keywordHash_t *KeywordHash_Find(keywordHash_t *table[], char *keyword)
+/*
+===============
+Item_IsListBox
+===============
+*/
+static ID_INLINE qboolean Item_IsListBox(itemDef_t *item)
{
- keywordHash_t *key;
- int hash;
+ switch (item->type)
+ {
+ case ITEM_TYPE_LISTBOX:
+ case ITEM_TYPE_COMBOBOX:
+ return qtrue;
- hash = KeywordHash_Key(keyword);
- for (key = table[hash]; key; key = key->next) {
- if (!Q_stricmp(key->keyword, keyword))
- return key;
- }
- return NULL;
+ default:
+ return qfalse;
+ }
}
/*
@@ -4681,787 +6206,827 @@ Item Keyword Parse functions
*/
// name <string>
-qboolean ItemParse_name( itemDef_t *item, int handle ) {
- if (!PC_String_Parse(handle, &item->window.name)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_name(itemDef_t *item, int handle)
+{
+ if (!PC_String_Parse(handle, &item->window.name))
+ return qfalse;
+
+ return qtrue;
}
// name <string>
-qboolean ItemParse_focusSound( itemDef_t *item, int handle ) {
- const char *temp;
- if (!PC_String_Parse(handle, &temp)) {
- return qfalse;
- }
- item->focusSound = DC->registerSound(temp, qfalse);
- return qtrue;
-}
+qboolean ItemParse_focusSound(itemDef_t *item, int handle)
+{
+ const char *temp;
+ if (!PC_String_Parse(handle, &temp))
+ return qfalse;
+
+ item->focusSound = DC->registerSound(temp, qfalse);
+ return qtrue;
+}
// text <string>
-qboolean ItemParse_text( itemDef_t *item, int handle ) {
- if (!PC_String_Parse(handle, &item->text)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_text(itemDef_t *item, int handle)
+{
+ if (!PC_String_Parse(handle, &item->text))
+ return qfalse;
+
+ return qtrue;
}
// group <string>
-qboolean ItemParse_group( itemDef_t *item, int handle ) {
- if (!PC_String_Parse(handle, &item->window.group)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_group(itemDef_t *item, int handle)
+{
+ if (!PC_String_Parse(handle, &item->window.group))
+ return qfalse;
+
+ return qtrue;
}
// asset_model <string>
-qboolean ItemParse_asset_model( itemDef_t *item, int handle ) {
- const char *temp;
- modelDef_t *modelPtr;
- Item_ValidateTypeData(item);
- modelPtr = (modelDef_t*)item->typeData;
+qboolean ItemParse_asset_model(itemDef_t *item, int handle)
+{
+ const char *temp;
- if (!PC_String_Parse(handle, &temp)) {
- return qfalse;
- }
- item->asset = DC->registerModel(temp);
- modelPtr->angle = rand() % 360;
- return qtrue;
+ if (!PC_String_Parse(handle, &temp))
+ return qfalse;
+
+ item->asset = DC->registerModel(temp);
+ item->typeData.model->angle = rand() % 360;
+ return qtrue;
}
// asset_shader <string>
-qboolean ItemParse_asset_shader( itemDef_t *item, int handle ) {
- const char *temp;
+qboolean ItemParse_asset_shader(itemDef_t *item, int handle)
+{
+ const char *temp;
- if (!PC_String_Parse(handle, &temp)) {
- return qfalse;
- }
- item->asset = DC->registerShaderNoMip(temp);
- return qtrue;
+ if (!PC_String_Parse(handle, &temp))
+ return qfalse;
+
+ item->asset = DC->registerShaderNoMip(temp);
+ return qtrue;
}
// model_origin <number> <number> <number>
-qboolean ItemParse_model_origin( itemDef_t *item, int handle ) {
- modelDef_t *modelPtr;
- Item_ValidateTypeData(item);
- modelPtr = (modelDef_t*)item->typeData;
-
- if (PC_Float_Parse(handle, &modelPtr->origin[0])) {
- if (PC_Float_Parse(handle, &modelPtr->origin[1])) {
- if (PC_Float_Parse(handle, &modelPtr->origin[2])) {
- return qtrue;
- }
- }
- }
- return qfalse;
+qboolean ItemParse_model_origin(itemDef_t *item, int handle)
+{
+ return (PC_Float_Parse(handle, &item->typeData.model->origin[0]) &&
+ PC_Float_Parse(handle, &item->typeData.model->origin[1]) &&
+ PC_Float_Parse(handle, &item->typeData.model->origin[2]));
}
// model_fovx <number>
-qboolean ItemParse_model_fovx( itemDef_t *item, int handle ) {
- modelDef_t *modelPtr;
- Item_ValidateTypeData(item);
- modelPtr = (modelDef_t*)item->typeData;
-
- if (!PC_Float_Parse(handle, &modelPtr->fov_x)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_model_fovx(itemDef_t *item, int handle)
+{
+ return PC_Float_Parse(handle, &item->typeData.model->fov_x);
}
// model_fovy <number>
-qboolean ItemParse_model_fovy( itemDef_t *item, int handle ) {
- modelDef_t *modelPtr;
- Item_ValidateTypeData(item);
- modelPtr = (modelDef_t*)item->typeData;
-
- if (!PC_Float_Parse(handle, &modelPtr->fov_y)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_model_fovy(itemDef_t *item, int handle)
+{
+ return PC_Float_Parse(handle, &item->typeData.model->fov_y);
}
// model_rotation <integer>
-qboolean ItemParse_model_rotation( itemDef_t *item, int handle ) {
- modelDef_t *modelPtr;
- Item_ValidateTypeData(item);
- modelPtr = (modelDef_t*)item->typeData;
-
- if (!PC_Int_Parse(handle, &modelPtr->rotationSpeed)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_model_rotation(itemDef_t *item, int handle)
+{
+ return PC_Int_Parse(handle, &item->typeData.model->rotationSpeed);
}
// model_angle <integer>
-qboolean ItemParse_model_angle( itemDef_t *item, int handle ) {
- modelDef_t *modelPtr;
- Item_ValidateTypeData(item);
- modelPtr = (modelDef_t*)item->typeData;
-
- if (!PC_Int_Parse(handle, &modelPtr->angle)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_model_angle(itemDef_t *item, int handle)
+{
+ return PC_Int_Parse(handle, &item->typeData.model->angle);
}
// rect <rectangle>
-qboolean ItemParse_rect( itemDef_t *item, int handle ) {
- if (!PC_Rect_Parse(handle, &item->window.rectClient)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_rect(itemDef_t *item, int handle)
+{
+ if (!PC_Rect_Parse(handle, &item->window.rectClient))
+ return qfalse;
+
+ return qtrue;
+}
+
+// aspectBias <bias>
+qboolean ItemParse_aspectBias(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->window.aspectBias))
+ return qfalse;
+
+ return qtrue;
}
// style <integer>
-qboolean ItemParse_style( itemDef_t *item, int handle ) {
- if (!PC_Int_Parse(handle, &item->window.style)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_style(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->window.style))
+ return qfalse;
+
+ return qtrue;
}
// decoration
-qboolean ItemParse_decoration( itemDef_t *item, int handle ) {
- item->window.flags |= WINDOW_DECORATION;
- return qtrue;
+qboolean ItemParse_decoration(itemDef_t *item, int handle)
+{
+ (void)handle;
+
+ item->window.flags |= WINDOW_DECORATION;
+ return qtrue;
}
// notselectable
-qboolean ItemParse_notselectable( itemDef_t *item, int handle ) {
- listBoxDef_t *listPtr;
- Item_ValidateTypeData(item);
- listPtr = (listBoxDef_t*)item->typeData;
- if (item->type == ITEM_TYPE_LISTBOX && listPtr) {
- listPtr->notselectable = qtrue;
- }
- return qtrue;
+qboolean ItemParse_notselectable(itemDef_t *item, int handle)
+{
+ (void)handle;
+
+ item->typeData.list->notselectable = qtrue;
+ return qtrue;
}
-// manually wrapped
-qboolean ItemParse_wrapped( itemDef_t *item, int handle ) {
- item->window.flags |= WINDOW_WRAPPED;
- return qtrue;
+// noscrollbar
+qboolean ItemParse_noscrollbar(itemDef_t *item, int handle)
+{
+ (void)handle;
+
+ item->typeData.list->noscrollbar = qtrue;
+ return qtrue;
}
-// auto wrapped
-qboolean ItemParse_autowrapped( itemDef_t *item, int handle ) {
- item->window.flags |= WINDOW_AUTOWRAPPED;
- return qtrue;
+// resetonfeederchange
+qboolean ItemParse_resetonfeederchange(itemDef_t *item, int handle)
+{
+ (void)handle;
+
+ item->typeData.list->resetonfeederchange = qtrue;
+ return qtrue;
}
+// auto wrapped
+qboolean ItemParse_wrapped(itemDef_t *item, int handle)
+{
+ (void)handle;
-// horizontalscroll
-qboolean ItemParse_horizontalscroll( itemDef_t *item, int handle ) {
- item->window.flags |= WINDOW_HORIZONTAL;
- return qtrue;
+ item->window.flags |= WINDOW_WRAPPED;
+ return qtrue;
}
// type <integer>
-qboolean ItemParse_type( itemDef_t *item, int handle ) {
- if (!PC_Int_Parse(handle, &item->type)) {
- return qfalse;
- }
- Item_ValidateTypeData(item);
- return qtrue;
+qboolean ItemParse_type(itemDef_t *item, int handle)
+{
+ if (item->type != ITEM_TYPE_NONE)
+ {
+ PC_SourceError(handle, "item already has a type");
+ return qfalse;
+ }
+
+ if (!PC_Int_Parse(handle, &item->type))
+ return qfalse;
+
+ if (item->type == ITEM_TYPE_NONE)
+ {
+ PC_SourceError(handle, "type must not be none");
+ return qfalse;
+ }
+
+ // allocate the relevant type data
+ switch (item->type)
+ {
+ case ITEM_TYPE_LISTBOX:
+ case ITEM_TYPE_COMBOBOX:
+ item->typeData.list = UI_Alloc(sizeof(listBoxDef_t));
+ memset(item->typeData.list, 0, sizeof(listBoxDef_t));
+ break;
+
+ case ITEM_TYPE_CYCLE:
+ item->typeData.cycle = UI_Alloc(sizeof(cycleDef_t));
+ memset(item->typeData.cycle, 0, sizeof(cycleDef_t));
+ break;
+
+ case ITEM_TYPE_EDITFIELD:
+ case ITEM_TYPE_SAYFIELD:
+ case ITEM_TYPE_NUMERICFIELD:
+ case ITEM_TYPE_YESNO:
+ case ITEM_TYPE_BIND:
+ case ITEM_TYPE_SLIDER:
+ case ITEM_TYPE_TEXT:
+ item->typeData.edit = UI_Alloc(sizeof(editFieldDef_t));
+ memset(item->typeData.edit, 0, sizeof(editFieldDef_t));
+
+ if (item->type == ITEM_TYPE_EDITFIELD || item->type == ITEM_TYPE_SAYFIELD)
+ item->typeData.edit->maxPaintChars = MAX_EDITFIELD;
+ break;
+
+ case ITEM_TYPE_MULTI:
+ item->typeData.multi = UI_Alloc(sizeof(multiDef_t));
+ memset(item->typeData.multi, 0, sizeof(multiDef_t));
+ break;
+
+ case ITEM_TYPE_MODEL:
+ item->typeData.model = UI_Alloc(sizeof(modelDef_t));
+ memset(item->typeData.model, 0, sizeof(modelDef_t));
+ break;
+
+ default:
+ break;
+ }
+
+ return qtrue;
}
// elementwidth, used for listbox image elements
-// uses textalignx for storage
-qboolean ItemParse_elementwidth( itemDef_t *item, int handle ) {
- listBoxDef_t *listPtr;
-
- Item_ValidateTypeData(item);
- listPtr = (listBoxDef_t*)item->typeData;
- if (!PC_Float_Parse(handle, &listPtr->elementWidth)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_elementwidth(itemDef_t *item, int handle)
+{
+ return PC_Float_Parse(handle, &item->typeData.list->elementWidth);
}
// elementheight, used for listbox image elements
-// uses textaligny for storage
-qboolean ItemParse_elementheight( itemDef_t *item, int handle ) {
- listBoxDef_t *listPtr;
+qboolean ItemParse_elementheight(itemDef_t *item, int handle)
+{
+ return PC_Float_Parse(handle, &item->typeData.list->elementHeight);
+}
- Item_ValidateTypeData(item);
- listPtr = (listBoxDef_t*)item->typeData;
- if (!PC_Float_Parse(handle, &listPtr->elementHeight)) {
- return qfalse;
- }
- return qtrue;
+// dropitems, number of items to drop from a combobox
+qboolean ItemParse_dropitems(itemDef_t *item, int handle)
+{
+ return PC_Int_Parse(handle, &item->typeData.list->dropItems);
}
-// feeder <float>
-qboolean ItemParse_feeder( itemDef_t *item, int handle ) {
- if (!PC_Float_Parse(handle, &item->special)) {
- return qfalse;
- }
- return qtrue;
+// feeder <int>
+qboolean ItemParse_feeder(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->feederID))
+ return qfalse;
+
+ return qtrue;
}
// elementtype, used to specify what type of elements a listbox contains
// uses textstyle for storage
-qboolean ItemParse_elementtype( itemDef_t *item, int handle ) {
- listBoxDef_t *listPtr;
-
- Item_ValidateTypeData(item);
- if (!item->typeData)
- return qfalse;
- listPtr = (listBoxDef_t*)item->typeData;
- if (!PC_Int_Parse(handle, &listPtr->elementStyle)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_elementtype(itemDef_t *item, int handle)
+{
+ return PC_Int_Parse(handle, &item->typeData.list->elementStyle);
}
// columns sets a number of columns and an x pos and width per..
-qboolean ItemParse_columns( itemDef_t *item, int handle ) {
- int num, i;
- listBoxDef_t *listPtr;
+qboolean ItemParse_columns(itemDef_t *item, int handle)
+{
+ int i;
- Item_ValidateTypeData(item);
- if (!item->typeData)
- return qfalse;
- listPtr = (listBoxDef_t*)item->typeData;
- if (PC_Int_Parse(handle, &num)) {
- if (num > MAX_LB_COLUMNS) {
- num = MAX_LB_COLUMNS;
- }
- listPtr->numColumns = num;
- for (i = 0; i < num; i++) {
- int pos, width, maxChars, align;
-
- if( PC_Int_Parse( handle, &pos ) &&
- PC_Int_Parse( handle, &width ) &&
- PC_Int_Parse( handle, &maxChars ) &&
- PC_Int_Parse( handle, &align ) )
- {
- listPtr->columnInfo[i].pos = pos;
- listPtr->columnInfo[i].width = width;
- listPtr->columnInfo[i].maxChars = maxChars;
- listPtr->columnInfo[i].align = align;
- } else {
+ if (!PC_Int_Parse(handle, &item->typeData.list->numColumns))
+ return qfalse;
+
+ if (item->typeData.list->numColumns > MAX_LB_COLUMNS)
+ {
+ PC_SourceError(handle, "exceeded maximum allowed columns (%d)", MAX_LB_COLUMNS);
return qfalse;
- }
}
- } else {
- return qfalse;
- }
- return qtrue;
+
+ for (i = 0; i < item->typeData.list->numColumns; i++)
+ {
+ int pos, width, align;
+
+ if (!PC_Int_Parse(handle, &pos) || !PC_Int_Parse(handle, &width) || !PC_Int_Parse(handle, &align))
+ return qfalse;
+
+ item->typeData.list->columnInfo[i].pos = pos;
+ item->typeData.list->columnInfo[i].width = width;
+ item->typeData.list->columnInfo[i].align = align;
+ }
+
+ return qtrue;
}
-qboolean ItemParse_border( itemDef_t *item, int handle ) {
- if (!PC_Int_Parse(handle, &item->window.border)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_border(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->window.border))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_bordersize( itemDef_t *item, int handle ) {
- if (!PC_Float_Parse(handle, &item->window.borderSize)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_bordersize(itemDef_t *item, int handle)
+{
+ if (!PC_Float_Parse(handle, &item->window.borderSize))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_visible( itemDef_t *item, int handle ) {
- int i;
+// FIXME: why does this require a parameter? visible MENU_FALSE does nothing
+qboolean ItemParse_visible(itemDef_t *item, int handle)
+{
+ int i;
- if (!PC_Int_Parse(handle, &i)) {
- return qfalse;
- }
- if (i) {
- item->window.flags |= WINDOW_VISIBLE;
- }
- return qtrue;
+ if (!PC_Int_Parse(handle, &i))
+ return qfalse;
+
+ if (i)
+ item->window.flags |= WINDOW_VISIBLE;
+
+ return qtrue;
}
-qboolean ItemParse_ownerdraw( itemDef_t *item, int handle ) {
- if (!PC_Int_Parse(handle, &item->window.ownerDraw)) {
- return qfalse;
- }
- item->type = ITEM_TYPE_OWNERDRAW;
- return qtrue;
+// ownerdraw <number>, implies ITEM_TYPE_OWNERDRAW
+qboolean ItemParse_ownerdraw(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->window.ownerDraw))
+ return qfalse;
+
+ if (item->type != ITEM_TYPE_NONE && item->type != ITEM_TYPE_OWNERDRAW)
+ {
+ PC_SourceError(handle, "ownerdraws cannot have an item type");
+ return qfalse;
+ }
+
+ item->type = ITEM_TYPE_OWNERDRAW;
+
+ return qtrue;
}
-qboolean ItemParse_align( itemDef_t *item, int handle ) {
- if (!PC_Int_Parse(handle, &item->alignment)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_align(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->alignment))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_textalign( itemDef_t *item, int handle ) {
- if (!PC_Int_Parse(handle, &item->textalignment)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_textalign(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->textalignment))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_textalignx( itemDef_t *item, int handle ) {
- if (!PC_Float_Parse(handle, &item->textalignx)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_textvalign(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->textvalignment))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_textaligny( itemDef_t *item, int handle ) {
- if (!PC_Float_Parse(handle, &item->textaligny)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_textalignx(itemDef_t *item, int handle)
+{
+ if (!PC_Float_Parse(handle, &item->textalignx))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_textscale( itemDef_t *item, int handle ) {
- if (!PC_Float_Parse(handle, &item->textscale)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_textaligny(itemDef_t *item, int handle)
+{
+ if (!PC_Float_Parse(handle, &item->textaligny))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_textstyle( itemDef_t *item, int handle ) {
- if (!PC_Int_Parse(handle, &item->textStyle)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_textscale(itemDef_t *item, int handle)
+{
+ if (!PC_Float_Parse(handle, &item->textscale))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_backcolor( itemDef_t *item, int handle ) {
- int i;
- float f;
+qboolean ItemParse_textstyle(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->textStyle))
+ return qfalse;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
- }
- item->window.backColor[i] = f;
- }
- return qtrue;
+ return qtrue;
}
-qboolean ItemParse_forecolor( itemDef_t *item, int handle ) {
- int i;
- float f;
+qboolean ItemParse_backcolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
+
+ item->window.backColor[i] = f;
}
- item->window.foreColor[i] = f;
- item->window.flags |= WINDOW_FORECOLORSET;
- }
- return qtrue;
+
+ return qtrue;
}
-qboolean ItemParse_bordercolor( itemDef_t *item, int handle ) {
- int i;
- float f;
+qboolean ItemParse_forecolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
+
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
+ item->window.foreColor[i] = f;
+ item->window.flags |= WINDOW_FORECOLORSET;
}
- item->window.borderColor[i] = f;
- }
- return qtrue;
-}
-qboolean ItemParse_outlinecolor( itemDef_t *item, int handle ) {
- if (!PC_Color_Parse(handle, &item->window.outlineColor)){
- return qfalse;
- }
- return qtrue;
+ return qtrue;
}
-qboolean ItemParse_background( itemDef_t *item, int handle ) {
- const char *temp;
+qboolean ItemParse_bordercolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
- if (!PC_String_Parse(handle, &temp)) {
- return qfalse;
- }
- item->window.background = DC->registerShaderNoMip(temp);
- return qtrue;
-}
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
-qboolean ItemParse_cinematic( itemDef_t *item, int handle ) {
- if (!PC_String_Parse(handle, &item->window.cinematicName)) {
- return qfalse;
- }
- return qtrue;
+ item->window.borderColor[i] = f;
+ }
+
+ return qtrue;
}
-qboolean ItemParse_doubleClick( itemDef_t *item, int handle ) {
- listBoxDef_t *listPtr;
+qboolean ItemParse_outlinecolor(itemDef_t *item, int handle)
+{
+ if (!PC_Color_Parse(handle, &item->window.outlineColor))
+ return qfalse;
- Item_ValidateTypeData(item);
- if (!item->typeData) {
- return qfalse;
- }
+ return qtrue;
+}
- listPtr = (listBoxDef_t*)item->typeData;
+qboolean ItemParse_background(itemDef_t *item, int handle)
+{
+ const char *temp;
- if (!PC_Script_Parse(handle, &listPtr->doubleClick)) {
- return qfalse;
- }
- return qtrue;
+ if (!PC_String_Parse(handle, &temp))
+ return qfalse;
+
+ item->window.background = DC->registerShaderNoMip(temp);
+ return qtrue;
}
-qboolean ItemParse_onFocus( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->onFocus)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_cinematic(itemDef_t *item, int handle)
+{
+ if (!PC_String_Parse(handle, &item->window.cinematicName))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_leaveFocus( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->leaveFocus)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_doubleClick(itemDef_t *item, int handle)
+{
+ return (item->typeData.list && PC_Script_Parse(handle, &item->typeData.list->doubleClick));
}
-qboolean ItemParse_mouseEnter( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->mouseEnter)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_onFocus(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->onFocus))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_mouseExit( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->mouseExit)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_leaveFocus(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->leaveFocus))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_mouseEnterText( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->mouseEnterText)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_mouseEnter(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->mouseEnter))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_mouseExitText( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->mouseExitText)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_mouseExit(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->mouseExit))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_onTextEntry( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->onTextEntry)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_mouseEnterText(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->mouseEnterText))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_action( itemDef_t *item, int handle ) {
- if (!PC_Script_Parse(handle, &item->action)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_mouseExitText(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->mouseExitText))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_special( itemDef_t *item, int handle ) {
- if (!PC_Float_Parse(handle, &item->special)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_onTextEntry(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->onTextEntry))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_cvarTest( itemDef_t *item, int handle ) {
- if (!PC_String_Parse(handle, &item->cvarTest)) {
- return qfalse;
- }
- return qtrue;
+qboolean ItemParse_onCharEntry(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->onCharEntry))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_cvar( itemDef_t *item, int handle ) {
- editFieldDef_t *editPtr;
+qboolean ItemParse_action(itemDef_t *item, int handle)
+{
+ if (!PC_Script_Parse(handle, &item->action))
+ return qfalse;
- Item_ValidateTypeData(item);
- if (!PC_String_Parse(handle, &item->cvar)) {
- return qfalse;
- }
- if (item->typeData) {
- editPtr = (editFieldDef_t*)item->typeData;
- editPtr->minVal = -1;
- editPtr->maxVal = -1;
- editPtr->defVal = -1;
- }
- return qtrue;
-}
-
-qboolean ItemParse_maxChars( itemDef_t *item, int handle ) {
- editFieldDef_t *editPtr;
- int maxChars;
-
- Item_ValidateTypeData(item);
- if (!item->typeData)
- return qfalse;
+ return qtrue;
+}
- if (!PC_Int_Parse(handle, &maxChars)) {
- return qfalse;
- }
- editPtr = (editFieldDef_t*)item->typeData;
- editPtr->maxChars = maxChars;
- return qtrue;
+qboolean ItemParse_cvarTest(itemDef_t *item, int handle)
+{
+ if (!PC_String_Parse(handle, &item->cvarTest))
+ return qfalse;
+
+ return qtrue;
}
-qboolean ItemParse_maxPaintChars( itemDef_t *item, int handle ) {
- editFieldDef_t *editPtr;
- int maxChars;
+qboolean ItemParse_cvar(itemDef_t *item, int handle)
+{
+ if (!PC_String_Parse(handle, &item->cvar))
+ return qfalse;
- Item_ValidateTypeData(item);
- if (!item->typeData)
- return qfalse;
+ if (Item_DataType(item) == TYPE_EDIT)
+ {
+ item->typeData.edit->minVal = -1;
+ item->typeData.edit->maxVal = -1;
+ item->typeData.edit->defVal = -1;
+ }
- if (!PC_Int_Parse(handle, &maxChars)) {
- return qfalse;
- }
- editPtr = (editFieldDef_t*)item->typeData;
- editPtr->maxPaintChars = maxChars;
- return qtrue;
+ return qtrue;
+}
+
+qboolean ItemParse_maxChars(itemDef_t *item, int handle)
+{
+ return PC_Int_Parse(handle, &item->typeData.edit->maxChars);
}
+qboolean ItemParse_maxPaintChars(itemDef_t *item, int handle)
+{
+ return PC_Int_Parse(handle, &item->typeData.edit->maxPaintChars);
+}
+qboolean ItemParse_maxFieldWidth(itemDef_t *item, int handle)
+{
+ if (!PC_Int_Parse(handle, &item->typeData.edit->maxFieldWidth))
+ return qfalse;
-qboolean ItemParse_cvarFloat( itemDef_t *item, int handle ) {
- editFieldDef_t *editPtr;
+ if (item->typeData.edit->maxFieldWidth < MIN_FIELD_WIDTH)
+ {
+ PC_SourceError(handle, "max field width must be at least %d", MIN_FIELD_WIDTH);
+ return qfalse;
+ }
- Item_ValidateTypeData(item);
- if (!item->typeData)
- return qfalse;
- editPtr = (editFieldDef_t*)item->typeData;
- if (PC_String_Parse(handle, &item->cvar) &&
- PC_Float_Parse(handle, &editPtr->defVal) &&
- PC_Float_Parse(handle, &editPtr->minVal) &&
- PC_Float_Parse(handle, &editPtr->maxVal)) {
return qtrue;
- }
- return qfalse;
}
-qboolean ItemParse_cvarStrList( itemDef_t *item, int handle ) {
- pc_token_t token;
- multiDef_t *multiPtr;
- int pass;
+qboolean ItemParse_cvarFloat(itemDef_t *item, int handle)
+{
+ return (PC_String_Parse(handle, &item->cvar) && PC_Float_Parse(handle, &item->typeData.edit->defVal) &&
+ PC_Float_Parse(handle, &item->typeData.edit->minVal) &&
+ PC_Float_Parse(handle, &item->typeData.edit->maxVal));
+}
- Item_ValidateTypeData(item);
- if (!item->typeData)
- return qfalse;
- multiPtr = (multiDef_t*)item->typeData;
- multiPtr->count = 0;
- multiPtr->strDef = qtrue;
+qboolean ItemParse_cvarStrList(itemDef_t *item, int handle)
+{
+ pc_token_t token;
+ multiDef_t *multiPtr;
+ int pass;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- if (*token.string != '{') {
- return qfalse;
- }
+ multiPtr = item->typeData.multi;
+ multiPtr->count = 0;
+ multiPtr->strDef = qtrue;
- pass = 0;
- while ( 1 ) {
- if (!trap_Parse_ReadToken(handle, &token)) {
- PC_SourceError(handle, "end of file inside menu item\n");
- return qfalse;
- }
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
- if (*token.string == '}') {
- return qtrue;
- }
+ if (*token.string != '{')
+ return qfalse;
- if (*token.string == ',' || *token.string == ';') {
- continue;
- }
+ pass = 0;
- if (pass == 0) {
- multiPtr->cvarList[multiPtr->count] = String_Alloc(token.string);
- pass = 1;
- } else {
- multiPtr->cvarStr[multiPtr->count] = String_Alloc(token.string);
- pass = 0;
- multiPtr->count++;
- if (multiPtr->count >= MAX_MULTI_CVARS) {
- return qfalse;
- }
+ while (1)
+ {
+ if (!trap_Parse_ReadToken(handle, &token))
+ {
+ PC_SourceError(handle, "end of file inside menu item\n");
+ return qfalse;
+ }
+
+ if (*token.string == '}')
+ return qtrue;
+
+ if (*token.string == ',' || *token.string == ';')
+ continue;
+
+ if (pass == 0)
+ {
+ multiPtr->cvarList[multiPtr->count] = String_Alloc(token.string);
+ pass = 1;
+ }
+ else
+ {
+ multiPtr->cvarStr[multiPtr->count] = String_Alloc(token.string);
+ pass = 0;
+ multiPtr->count++;
+
+ if (multiPtr->count >= MAX_MULTI_CVARS)
+ {
+ PC_SourceError(handle, "cvar string list may not exceed %d cvars", MAX_MULTI_CVARS);
+ return qfalse;
+ }
+ }
}
- }
- return qfalse; // bk001205 - LCC missing return value
+ return qfalse;
}
-qboolean ItemParse_cvarFloatList( itemDef_t *item, int handle ) {
- pc_token_t token;
- multiDef_t *multiPtr;
+qboolean ItemParse_cvarFloatList(itemDef_t *item, int handle)
+{
+ pc_token_t token;
+ multiDef_t *multiPtr;
- Item_ValidateTypeData(item);
- if (!item->typeData)
- return qfalse;
- multiPtr = (multiDef_t*)item->typeData;
- multiPtr->count = 0;
- multiPtr->strDef = qfalse;
+ multiPtr = item->typeData.multi;
+ multiPtr->count = 0;
+ multiPtr->strDef = qfalse;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- if (*token.string != '{') {
- return qfalse;
- }
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
- while ( 1 ) {
- if (!trap_Parse_ReadToken(handle, &token)) {
- PC_SourceError(handle, "end of file inside menu item\n");
- return qfalse;
- }
+ if (*token.string != '{')
+ return qfalse;
- if (*token.string == '}') {
- return qtrue;
- }
+ while (1)
+ {
+ if (!trap_Parse_ReadToken(handle, &token))
+ {
+ PC_SourceError(handle, "end of file inside menu item\n");
+ return qfalse;
+ }
- if (*token.string == ',' || *token.string == ';') {
- continue;
- }
+ if (*token.string == '}')
+ return qtrue;
- multiPtr->cvarList[multiPtr->count] = String_Alloc(token.string);
- if (!PC_Float_Parse(handle, &multiPtr->cvarValue[multiPtr->count])) {
- return qfalse;
- }
+ if (*token.string == ',' || *token.string == ';')
+ continue;
- multiPtr->count++;
- if (multiPtr->count >= MAX_MULTI_CVARS) {
- return qfalse;
+ multiPtr->cvarList[multiPtr->count] = String_Alloc(token.string);
+
+ if (!PC_Float_Parse(handle, &multiPtr->cvarValue[multiPtr->count]))
+ return qfalse;
+
+ multiPtr->count++;
+
+ if (multiPtr->count >= MAX_MULTI_CVARS)
+ {
+ PC_SourceError(handle, "cvar string list may not exceed %d cvars", MAX_MULTI_CVARS);
+ return qfalse;
+ }
}
- }
- return qfalse; // bk001205 - LCC missing return value
+ return qfalse;
}
+qboolean ItemParse_addColorRange(itemDef_t *item, int handle)
+{
+ colorRangeDef_t color;
+ if (PC_Float_Parse(handle, &color.low) && PC_Float_Parse(handle, &color.high) &&
+ PC_Color_Parse(handle, &color.color))
+ {
+ if (item->numColors < MAX_COLOR_RANGES)
+ {
+ memcpy(&item->colorRanges[item->numColors], &color, sizeof(color));
+ item->numColors++;
+ }
+ else
+ {
+ PC_SourceError(handle, "may not exceed %d color ranges", MAX_COLOR_RANGES);
+ return qfalse;
+ }
-qboolean ItemParse_addColorRange( itemDef_t *item, int handle ) {
- colorRangeDef_t color;
-
- if (PC_Float_Parse(handle, &color.low) &&
- PC_Float_Parse(handle, &color.high) &&
- PC_Color_Parse(handle, &color.color) ) {
- if (item->numColors < MAX_COLOR_RANGES) {
- memcpy(&item->colorRanges[item->numColors], &color, sizeof(color));
- item->numColors++;
+ return qtrue;
}
+
+ return qfalse;
+}
+
+qboolean ItemParse_ownerdrawFlag(itemDef_t *item, int handle)
+{
+ int i;
+
+ if (!PC_Int_Parse(handle, &i))
+ return qfalse;
+
+ item->window.ownerDrawFlags |= i;
return qtrue;
- }
- return qfalse;
}
-qboolean ItemParse_ownerdrawFlag( itemDef_t *item, int handle ) {
- int i;
- if (!PC_Int_Parse(handle, &i)) {
+qboolean ItemParse_enableCvar(itemDef_t *item, int handle)
+{
+ if (PC_Script_Parse(handle, &item->enableCvar))
+ {
+ item->cvarFlags = CVAR_ENABLE;
+ return qtrue;
+ }
+
return qfalse;
- }
- item->window.ownerDrawFlags |= i;
- return qtrue;
}
-qboolean ItemParse_enableCvar( itemDef_t *item, int handle ) {
- if (PC_Script_Parse(handle, &item->enableCvar)) {
- item->cvarFlags = CVAR_ENABLE;
- return qtrue;
- }
- return qfalse;
+qboolean ItemParse_disableCvar(itemDef_t *item, int handle)
+{
+ if (PC_Script_Parse(handle, &item->enableCvar))
+ {
+ item->cvarFlags = CVAR_DISABLE;
+ return qtrue;
+ }
+
+ return qfalse;
}
-qboolean ItemParse_disableCvar( itemDef_t *item, int handle ) {
- if (PC_Script_Parse(handle, &item->enableCvar)) {
- item->cvarFlags = CVAR_DISABLE;
- return qtrue;
- }
- return qfalse;
+qboolean ItemParse_showCvar(itemDef_t *item, int handle)
+{
+ if (PC_Script_Parse(handle, &item->enableCvar))
+ {
+ item->cvarFlags = CVAR_SHOW;
+ return qtrue;
+ }
+
+ return qfalse;
}
-qboolean ItemParse_showCvar( itemDef_t *item, int handle ) {
- if (PC_Script_Parse(handle, &item->enableCvar)) {
- item->cvarFlags = CVAR_SHOW;
- return qtrue;
- }
- return qfalse;
+qboolean ItemParse_hideCvar(itemDef_t *item, int handle)
+{
+ if (PC_Script_Parse(handle, &item->enableCvar))
+ {
+ item->cvarFlags = CVAR_HIDE;
+ return qtrue;
+ }
+
+ return qfalse;
}
-qboolean ItemParse_hideCvar( itemDef_t *item, int handle ) {
- if (PC_Script_Parse(handle, &item->enableCvar)) {
- item->cvarFlags = CVAR_HIDE;
- return qtrue;
- }
- return qfalse;
-}
-
-
-keywordHash_t itemParseKeywords[] = {
- {"name", ItemParse_name, NULL},
- {"text", ItemParse_text, NULL},
- {"group", ItemParse_group, NULL},
- {"asset_model", ItemParse_asset_model, NULL},
- {"asset_shader", ItemParse_asset_shader, NULL},
- {"model_origin", ItemParse_model_origin, NULL},
- {"model_fovx", ItemParse_model_fovx, NULL},
- {"model_fovy", ItemParse_model_fovy, NULL},
- {"model_rotation", ItemParse_model_rotation, NULL},
- {"model_angle", ItemParse_model_angle, NULL},
- {"rect", ItemParse_rect, NULL},
- {"style", ItemParse_style, NULL},
- {"decoration", ItemParse_decoration, NULL},
- {"notselectable", ItemParse_notselectable, NULL},
- {"wrapped", ItemParse_wrapped, NULL},
- {"autowrapped", ItemParse_autowrapped, NULL},
- {"horizontalscroll", ItemParse_horizontalscroll, NULL},
- {"type", ItemParse_type, NULL},
- {"elementwidth", ItemParse_elementwidth, NULL},
- {"elementheight", ItemParse_elementheight, NULL},
- {"feeder", ItemParse_feeder, NULL},
- {"elementtype", ItemParse_elementtype, NULL},
- {"columns", ItemParse_columns, NULL},
- {"border", ItemParse_border, NULL},
- {"bordersize", ItemParse_bordersize, NULL},
- {"visible", ItemParse_visible, NULL},
- {"ownerdraw", ItemParse_ownerdraw, NULL},
- {"align", ItemParse_align, NULL},
- {"textalign", ItemParse_textalign, NULL},
- {"textalignx", ItemParse_textalignx, NULL},
- {"textaligny", ItemParse_textaligny, NULL},
- {"textscale", ItemParse_textscale, NULL},
- {"textstyle", ItemParse_textstyle, NULL},
- {"backcolor", ItemParse_backcolor, NULL},
- {"forecolor", ItemParse_forecolor, NULL},
- {"bordercolor", ItemParse_bordercolor, NULL},
- {"outlinecolor", ItemParse_outlinecolor, NULL},
- {"background", ItemParse_background, NULL},
- {"onFocus", ItemParse_onFocus, NULL},
- {"leaveFocus", ItemParse_leaveFocus, NULL},
- {"mouseEnter", ItemParse_mouseEnter, NULL},
- {"mouseExit", ItemParse_mouseExit, NULL},
- {"mouseEnterText", ItemParse_mouseEnterText, NULL},
- {"mouseExitText", ItemParse_mouseExitText, NULL},
- {"onTextEntry", ItemParse_onTextEntry, NULL},
- {"action", ItemParse_action, NULL},
- {"special", ItemParse_special, NULL},
- {"cvar", ItemParse_cvar, NULL},
- {"maxChars", ItemParse_maxChars, NULL},
- {"maxPaintChars", ItemParse_maxPaintChars, NULL},
- {"focusSound", ItemParse_focusSound, NULL},
- {"cvarFloat", ItemParse_cvarFloat, NULL},
- {"cvarStrList", ItemParse_cvarStrList, NULL},
- {"cvarFloatList", ItemParse_cvarFloatList, NULL},
- {"addColorRange", ItemParse_addColorRange, NULL},
- {"ownerdrawFlag", ItemParse_ownerdrawFlag, NULL},
- {"enableCvar", ItemParse_enableCvar, NULL},
- {"cvarTest", ItemParse_cvarTest, NULL},
- {"disableCvar", ItemParse_disableCvar, NULL},
- {"showCvar", ItemParse_showCvar, NULL},
- {"hideCvar", ItemParse_hideCvar, NULL},
- {"cinematic", ItemParse_cinematic, NULL},
- {"doubleclick", ItemParse_doubleClick, NULL},
- {NULL, voidFunction2, NULL}
-};
+keywordHash_t itemParseKeywords[] = {{"name", ItemParse_name, TYPE_ANY, NULL}, {"type", ItemParse_type, TYPE_ANY, NULL},
+ {"text", ItemParse_text, TYPE_ANY, NULL}, {"group", ItemParse_group, TYPE_ANY, NULL},
+ {"asset_model", ItemParse_asset_model, TYPE_MODEL, NULL},
+ {"asset_shader", ItemParse_asset_shader, TYPE_ANY, NULL}, // ?
+ {"model_origin", ItemParse_model_origin, TYPE_MODEL, NULL}, {"model_fovx", ItemParse_model_fovx, TYPE_MODEL, NULL},
+ {"model_fovy", ItemParse_model_fovy, TYPE_MODEL, NULL},
+ {"model_rotation", ItemParse_model_rotation, TYPE_MODEL, NULL},
+ {"model_angle", ItemParse_model_angle, TYPE_MODEL, NULL}, {"rect", ItemParse_rect, TYPE_ANY, NULL},
+ {"aspectBias", ItemParse_aspectBias, TYPE_ANY, NULL}, {"style", ItemParse_style, TYPE_ANY, NULL},
+ {"decoration", ItemParse_decoration, TYPE_ANY, NULL}, {"notselectable", ItemParse_notselectable, TYPE_LIST, NULL},
+ {"noscrollbar", ItemParse_noscrollbar, TYPE_LIST, NULL},
+ {"resetonfeederchange", ItemParse_resetonfeederchange, TYPE_LIST, NULL},
+ {"wrapped", ItemParse_wrapped, TYPE_ANY, NULL}, {"elementwidth", ItemParse_elementwidth, TYPE_LIST, NULL},
+ {"elementheight", ItemParse_elementheight, TYPE_LIST, NULL}, {"dropitems", ItemParse_dropitems, TYPE_LIST, NULL},
+ {"feeder", ItemParse_feeder, TYPE_ANY, NULL}, {"elementtype", ItemParse_elementtype, TYPE_LIST, NULL},
+ {"columns", ItemParse_columns, TYPE_LIST, NULL}, {"border", ItemParse_border, TYPE_ANY, NULL},
+ {"bordersize", ItemParse_bordersize, TYPE_ANY, NULL}, {"visible", ItemParse_visible, TYPE_ANY, NULL},
+ {"ownerdraw", ItemParse_ownerdraw, TYPE_ANY, NULL}, {"align", ItemParse_align, TYPE_ANY, NULL},
+ {"textalign", ItemParse_textalign, TYPE_ANY, NULL}, {"textvalign", ItemParse_textvalign, TYPE_ANY, NULL},
+ {"textalignx", ItemParse_textalignx, TYPE_ANY, NULL}, {"textaligny", ItemParse_textaligny, TYPE_ANY, NULL},
+ {"textscale", ItemParse_textscale, TYPE_ANY, NULL}, {"textstyle", ItemParse_textstyle, TYPE_ANY, NULL},
+ {"backcolor", ItemParse_backcolor, TYPE_ANY, NULL}, {"forecolor", ItemParse_forecolor, TYPE_ANY, NULL},
+ {"bordercolor", ItemParse_bordercolor, TYPE_ANY, NULL}, {"outlinecolor", ItemParse_outlinecolor, TYPE_ANY, NULL},
+ {"background", ItemParse_background, TYPE_ANY, NULL}, {"onFocus", ItemParse_onFocus, TYPE_ANY, NULL},
+ {"leaveFocus", ItemParse_leaveFocus, TYPE_ANY, NULL}, {"mouseEnter", ItemParse_mouseEnter, TYPE_ANY, NULL},
+ {"mouseExit", ItemParse_mouseExit, TYPE_ANY, NULL}, {"mouseEnterText", ItemParse_mouseEnterText, TYPE_ANY, NULL},
+ {"mouseExitText", ItemParse_mouseExitText, TYPE_ANY, NULL}, {"onTextEntry", ItemParse_onTextEntry, TYPE_ANY, NULL},
+ {"onCharEntry", ItemParse_onCharEntry, TYPE_ANY, NULL}, {"action", ItemParse_action, TYPE_ANY, NULL},
+ {"cvar", ItemParse_cvar, TYPE_ANY, NULL}, {"maxChars", ItemParse_maxChars, TYPE_EDIT, NULL},
+ {"maxPaintChars", ItemParse_maxPaintChars, TYPE_EDIT, NULL},
+ {"maxFieldWidth", ItemParse_maxFieldWidth, TYPE_EDIT, NULL}, {"focusSound", ItemParse_focusSound, TYPE_ANY, NULL},
+ {"cvarFloat", ItemParse_cvarFloat, TYPE_EDIT, NULL}, {"cvarStrList", ItemParse_cvarStrList, TYPE_MULTI, NULL},
+ {"cvarFloatList", ItemParse_cvarFloatList, TYPE_MULTI, NULL},
+ {"addColorRange", ItemParse_addColorRange, TYPE_ANY, NULL},
+ {"ownerdrawFlag", ItemParse_ownerdrawFlag, TYPE_ANY, NULL}, // hm.
+ {"enableCvar", ItemParse_enableCvar, TYPE_ANY, NULL}, {"cvarTest", ItemParse_cvarTest, TYPE_ANY, NULL},
+ {"disableCvar", ItemParse_disableCvar, TYPE_ANY, NULL}, {"showCvar", ItemParse_showCvar, TYPE_ANY, NULL},
+ {"hideCvar", ItemParse_hideCvar, TYPE_ANY, NULL}, {"cinematic", ItemParse_cinematic, TYPE_ANY, NULL},
+ {"doubleclick", ItemParse_doubleClick, TYPE_LIST, NULL}, {NULL, voidFunction2, 0, NULL}};
keywordHash_t *itemParseKeywordHash[KEYWORDHASH_SIZE];
@@ -5470,14 +7035,14 @@ keywordHash_t *itemParseKeywordHash[KEYWORDHASH_SIZE];
Item_SetupKeywordHash
===============
*/
-void Item_SetupKeywordHash( void )
+void Item_SetupKeywordHash(void)
{
- int i;
+ int i;
- memset( itemParseKeywordHash, 0, sizeof( itemParseKeywordHash ) );
+ memset(itemParseKeywordHash, 0, sizeof(itemParseKeywordHash));
- for( i = 0; itemParseKeywords[ i ].keyword; i++ )
- KeywordHash_Add( itemParseKeywordHash, &itemParseKeywords[ i ] );
+ for (i = 0; itemParseKeywords[i].keyword; i++)
+ KeywordHash_Add(itemParseKeywordHash, &itemParseKeywords[i]);
}
/*
@@ -5485,56 +7050,85 @@ void Item_SetupKeywordHash( void )
Item_Parse
===============
*/
-qboolean Item_Parse(int handle, itemDef_t *item) {
- pc_token_t token;
- keywordHash_t *key;
+qboolean Item_Parse(int handle, itemDef_t *item)
+{
+ pc_token_t token;
+ keywordHash_t *key;
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- if (*token.string != '{') {
- return qfalse;
- }
- while ( 1 ) {
- if (!trap_Parse_ReadToken(handle, &token)) {
- PC_SourceError(handle, "end of file inside menu item\n");
- return qfalse;
- }
+ if (*token.string != '{')
+ return qfalse;
- if (*token.string == '}') {
- return qtrue;
- }
+ while (1)
+ {
+ if (!trap_Parse_ReadToken(handle, &token))
+ {
+ PC_SourceError(handle, "end of file inside menu item\n");
+ return qfalse;
+ }
- key = KeywordHash_Find(itemParseKeywordHash, token.string);
- if (!key) {
- PC_SourceError(handle, "unknown menu item keyword %s", token.string);
- continue;
- }
- if ( !key->func(item, handle) ) {
- PC_SourceError(handle, "couldn't parse menu item keyword %s", token.string);
- return qfalse;
+ if (*token.string == '}')
+ return qtrue;
+
+ key = KeywordHash_Find(itemParseKeywordHash, token.string);
+
+ if (!key)
+ {
+ PC_SourceError(handle, "unknown menu item keyword %s", token.string);
+ continue;
+ }
+
+ // do type-checks
+ if (key->param != TYPE_ANY)
+ {
+ itemDataType_t test = Item_DataType(item);
+
+ if (test != key->param)
+ {
+ if (test == TYPE_NONE)
+ PC_SourceError(handle,
+ "menu item keyword %s requires "
+ "type specification",
+ token.string);
+ else
+ PC_SourceError(handle,
+ "menu item keyword %s is incompatible with "
+ "specified item type",
+ token.string);
+ continue;
+ }
+ }
+
+ if (!key->func(item, handle))
+ {
+ PC_SourceError(handle, "couldn't parse menu item keyword %s", token.string);
+ return qfalse;
+ }
}
- }
- return qfalse; // bk001205 - LCC missing return value
-}
+ 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;
- }
- }
+void Item_InitControls(itemDef_t *item)
+{
+ if (item == NULL)
+ return;
+
+ if (Item_IsListBox(item))
+ {
+ item->cursorPos = 0;
+
+ if (item->typeData.list)
+ {
+ item->typeData.list->cursorPos = 0;
+ Item_ListBox_SetStartPos(item, 0);
+ item->typeData.list->cursorPos = 0;
+ }
+ }
}
/*
@@ -5543,327 +7137,376 @@ Menu Keyword Parse functions
===============
*/
-qboolean MenuParse_font( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_String_Parse(handle, &menu->font)) {
- return qfalse;
- }
- if (!DC->Assets.fontRegistered) {
- DC->registerFont(menu->font, 48, &DC->Assets.textFont);
- DC->Assets.fontRegistered = qtrue;
- }
- return qtrue;
+qboolean MenuParse_font(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_String_Parse(handle, &menu->font))
+ return qfalse;
+
+ if (!DC->Assets.fontRegistered)
+ {
+ DC->registerFont(menu->font, 48, &DC->Assets.textFont);
+ DC->Assets.fontRegistered = qtrue;
+ }
+
+ return qtrue;
}
-qboolean MenuParse_name( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_String_Parse(handle, &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;
+qboolean MenuParse_name(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_String_Parse(handle, &menu->window.name))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_fullscreen( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Int_Parse(handle, (int*) &menu->fullScreen)) { // bk001206 - cast qboolean
- return qfalse;
- }
- return qtrue;
+qboolean MenuParse_fullscreen(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Int_Parse(handle, (int *)&menu->fullScreen))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_rect( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Rect_Parse(handle, &menu->window.rect)) {
- return qfalse;
- }
- return qtrue;
+qboolean MenuParse_rect(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Rect_Parse(handle, &menu->window.rect))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_style( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Int_Parse(handle, &menu->window.style)) {
- return qfalse;
- }
- return qtrue;
+qboolean MenuParse_aspectBias(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Int_Parse(handle, &menu->window.aspectBias))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_visible( itemDef_t *item, int handle ) {
- int i;
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_style(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_Int_Parse(handle, &i)) {
- return qfalse;
- }
- if (i) {
- menu->window.flags |= WINDOW_VISIBLE;
- }
- return qtrue;
+ if (!PC_Int_Parse(handle, &menu->window.style))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_onOpen( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Script_Parse(handle, &menu->onOpen)) {
- return qfalse;
- }
- return qtrue;
+qboolean MenuParse_visible(itemDef_t *item, int handle)
+{
+ int i;
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Int_Parse(handle, &i))
+ return qfalse;
+
+ if (i)
+ menu->window.flags |= WINDOW_VISIBLE;
+
+ return qtrue;
}
-qboolean MenuParse_onClose( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Script_Parse(handle, &menu->onClose)) {
- return qfalse;
- }
- return qtrue;
+qboolean MenuParse_onOpen(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Script_Parse(handle, &menu->onOpen))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_onESC( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Script_Parse(handle, &menu->onESC)) {
- return qfalse;
- }
- return qtrue;
+qboolean MenuParse_onClose(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Script_Parse(handle, &menu->onClose))
+ return qfalse;
+
+ return qtrue;
}
+qboolean MenuParse_onESC(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+ if (!PC_Script_Parse(handle, &menu->onESC))
+ return qfalse;
-qboolean MenuParse_border( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Int_Parse(handle, &menu->window.border)) {
- return qfalse;
- }
- return qtrue;
+ return qtrue;
}
-qboolean MenuParse_borderSize( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Float_Parse(handle, &menu->window.borderSize)) {
- return qfalse;
- }
- return qtrue;
+qboolean MenuParse_border(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Int_Parse(handle, &menu->window.border))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_backcolor( itemDef_t *item, int handle ) {
- int i;
- float f;
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_borderSize(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Float_Parse(handle, &menu->window.borderSize))
+ return qfalse;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
- }
- menu->window.backColor[i] = f;
- }
- return qtrue;
+ return qtrue;
}
-qboolean MenuParse_forecolor( itemDef_t *item, int handle ) {
- int i;
- float f;
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_backcolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
+ menuDef_t *menu = (menuDef_t *)item;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
+
+ menu->window.backColor[i] = f;
}
- menu->window.foreColor[i] = f;
- menu->window.flags |= WINDOW_FORECOLORSET;
- }
- return qtrue;
+
+ return qtrue;
}
-qboolean MenuParse_bordercolor( itemDef_t *item, int handle ) {
- int i;
- float f;
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_forecolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
+ menuDef_t *menu = (menuDef_t *)item;
+
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
+ menu->window.foreColor[i] = f;
+ menu->window.flags |= WINDOW_FORECOLORSET;
}
- menu->window.borderColor[i] = f;
- }
- return qtrue;
+
+ return qtrue;
}
-qboolean MenuParse_focuscolor( itemDef_t *item, int handle ) {
- int i;
- float f;
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_bordercolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
+ menuDef_t *menu = (menuDef_t *)item;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
+
+ menu->window.borderColor[i] = f;
}
- menu->focusColor[i] = f;
- }
- return qtrue;
+
+ return qtrue;
}
-qboolean MenuParse_disablecolor( itemDef_t *item, int handle ) {
- int i;
- float f;
- menuDef_t *menu = (menuDef_t*)item;
- for (i = 0; i < 4; i++) {
- if (!PC_Float_Parse(handle, &f)) {
- return qfalse;
+qboolean MenuParse_focuscolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
+ menuDef_t *menu = (menuDef_t *)item;
+
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
+
+ menu->focusColor[i] = f;
}
- menu->disableColor[i] = f;
- }
- return qtrue;
+
+ return qtrue;
}
+qboolean MenuParse_disablecolor(itemDef_t *item, int handle)
+{
+ int i;
+ float f;
+ menuDef_t *menu = (menuDef_t *)item;
+
+ for (i = 0; i < 4; i++)
+ {
+ if (!PC_Float_Parse(handle, &f))
+ return qfalse;
-qboolean MenuParse_outlinecolor( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (!PC_Color_Parse(handle, &menu->window.outlineColor)){
- return qfalse;
- }
- return qtrue;
+ menu->disableColor[i] = f;
+ }
+
+ return qtrue;
}
-qboolean MenuParse_background( itemDef_t *item, int handle ) {
- const char *buff;
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_outlinecolor(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_String_Parse(handle, &buff)) {
- return qfalse;
- }
- menu->window.background = DC->registerShaderNoMip(buff);
- return qtrue;
+ if (!PC_Color_Parse(handle, &menu->window.outlineColor))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_cinematic( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_background(itemDef_t *item, int handle)
+{
+ const char *buff;
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_String_Parse(handle, &menu->window.cinematicName)) {
- return qfalse;
- }
- return qtrue;
+ if (!PC_String_Parse(handle, &buff))
+ return qfalse;
+
+ menu->window.background = DC->registerShaderNoMip(buff);
+ return qtrue;
}
-qboolean MenuParse_ownerdrawFlag( itemDef_t *item, int handle ) {
- int i;
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_cinematic(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_Int_Parse(handle, &i)) {
- return qfalse;
- }
- menu->window.ownerDrawFlags |= i;
- return qtrue;
+ if (!PC_String_Parse(handle, &menu->window.cinematicName))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_ownerdraw( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_ownerdrawFlag(itemDef_t *item, int handle)
+{
+ int i;
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_Int_Parse(handle, &menu->window.ownerDraw)) {
- return qfalse;
- }
- return qtrue;
+ if (!PC_Int_Parse(handle, &i))
+ return qfalse;
+
+ menu->window.ownerDrawFlags |= i;
+ return qtrue;
}
+qboolean MenuParse_ownerdraw(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (!PC_Int_Parse(handle, &menu->window.ownerDraw))
+ return qfalse;
+
+ return qtrue;
+}
// decoration
-qboolean MenuParse_popup( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- menu->window.flags |= WINDOW_POPUP;
- return qtrue;
+qboolean MenuParse_popup(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+ menu->window.flags |= WINDOW_POPUP;
+
+ (void)handle;
+
+ return qtrue;
}
+qboolean MenuParse_outOfBounds(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+ menu->window.flags |= WINDOW_OOB_CLICK;
-qboolean MenuParse_outOfBounds( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
+ (void)handle;
- menu->window.flags |= WINDOW_OOB_CLICK;
- return qtrue;
+ return qtrue;
}
-qboolean MenuParse_soundLoop( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_soundLoop(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_String_Parse(handle, &menu->soundName)) {
- return qfalse;
- }
- return qtrue;
+ if (!PC_String_Parse(handle, &menu->soundName))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_fadeClamp( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_fadeClamp(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_Float_Parse(handle, &menu->fadeClamp)) {
- return qfalse;
- }
- return qtrue;
+ if (!PC_Float_Parse(handle, &menu->fadeClamp))
+ return qfalse;
+
+ return qtrue;
}
-qboolean MenuParse_fadeAmount( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
+qboolean MenuParse_fadeAmount(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
- if (!PC_Float_Parse(handle, &menu->fadeAmount)) {
- return qfalse;
- }
- return qtrue;
+ if (!PC_Float_Parse(handle, &menu->fadeAmount))
+ return qfalse;
+
+ return qtrue;
}
+qboolean MenuParse_fadeCycle(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
-qboolean MenuParse_fadeCycle( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
+ if (!PC_Int_Parse(handle, &menu->fadeCycle))
+ return qfalse;
- if (!PC_Int_Parse(handle, &menu->fadeCycle)) {
- return qfalse;
- }
- return qtrue;
-}
-
-
-qboolean MenuParse_itemDef( itemDef_t *item, int handle ) {
- menuDef_t *menu = (menuDef_t*)item;
- if (menu->itemCount < MAX_MENUITEMS) {
- menu->items[menu->itemCount] = UI_Alloc(sizeof(itemDef_t));
- Item_Init(menu->items[menu->itemCount]);
- if (!Item_Parse(handle, menu->items[menu->itemCount])) {
- return qfalse;
- }
- Item_InitControls(menu->items[menu->itemCount]);
- menu->items[menu->itemCount++]->parent = menu;
- }
- return qtrue;
-}
-
-keywordHash_t menuParseKeywords[] = {
- {"font", MenuParse_font, NULL},
- {"name", MenuParse_name, NULL},
- {"fullscreen", MenuParse_fullscreen, NULL},
- {"rect", MenuParse_rect, NULL},
- {"style", MenuParse_style, NULL},
- {"visible", MenuParse_visible, NULL},
- {"onOpen", MenuParse_onOpen, NULL},
- {"onClose", MenuParse_onClose, NULL},
- {"onESC", MenuParse_onESC, NULL},
- {"border", MenuParse_border, NULL},
- {"borderSize", MenuParse_borderSize, NULL},
- {"backcolor", MenuParse_backcolor, NULL},
- {"forecolor", MenuParse_forecolor, NULL},
- {"bordercolor", MenuParse_bordercolor, NULL},
- {"focuscolor", MenuParse_focuscolor, NULL},
- {"disablecolor", MenuParse_disablecolor, NULL},
- {"outlinecolor", MenuParse_outlinecolor, NULL},
- {"background", MenuParse_background, NULL},
- {"ownerdraw", MenuParse_ownerdraw, NULL},
- {"ownerdrawFlag", MenuParse_ownerdrawFlag, NULL},
- {"outOfBoundsClick", MenuParse_outOfBounds, NULL},
- {"soundLoop", MenuParse_soundLoop, NULL},
- {"itemDef", MenuParse_itemDef, NULL},
- {"cinematic", MenuParse_cinematic, NULL},
- {"popup", MenuParse_popup, NULL},
- {"fadeClamp", MenuParse_fadeClamp, NULL},
- {"fadeCycle", MenuParse_fadeCycle, NULL},
- {"fadeAmount", MenuParse_fadeAmount, NULL},
- {NULL, voidFunction2, NULL}
-};
+ return qtrue;
+}
+
+qboolean MenuParse_itemDef(itemDef_t *item, int handle)
+{
+ menuDef_t *menu = (menuDef_t *)item;
+
+ if (menu->itemCount < MAX_MENUITEMS)
+ {
+ menu->items[menu->itemCount] = UI_Alloc(sizeof(itemDef_t));
+ Item_Init(menu->items[menu->itemCount]);
+
+ if (!Item_Parse(handle, menu->items[menu->itemCount]))
+ return qfalse;
+
+ Item_InitControls(menu->items[menu->itemCount]);
+ menu->items[menu->itemCount++]->parent = menu;
+ }
+ else
+ {
+ PC_SourceError(handle, "itemDefs per menu may not exceed %d", MAX_MENUITEMS);
+ return qfalse;
+ }
+
+ return qtrue;
+}
+
+keywordHash_t menuParseKeywords[] = {{"font", MenuParse_font, 0, NULL}, {"name", MenuParse_name, 0, NULL},
+ {"fullscreen", MenuParse_fullscreen, 0, NULL}, {"rect", MenuParse_rect, 0, NULL},
+ {"aspectBias", MenuParse_aspectBias, 0, NULL}, {"style", MenuParse_style, 0, NULL},
+ {"visible", MenuParse_visible, 0, NULL}, {"onOpen", MenuParse_onOpen, 0, NULL},
+ {"onClose", MenuParse_onClose, 0, NULL}, {"onESC", MenuParse_onESC, 0, NULL}, {"border", MenuParse_border, 0, NULL},
+ {"borderSize", MenuParse_borderSize, 0, NULL}, {"backcolor", MenuParse_backcolor, 0, NULL},
+ {"forecolor", MenuParse_forecolor, 0, NULL}, {"bordercolor", MenuParse_bordercolor, 0, NULL},
+ {"focuscolor", MenuParse_focuscolor, 0, NULL}, {"disablecolor", MenuParse_disablecolor, 0, NULL},
+ {"outlinecolor", MenuParse_outlinecolor, 0, NULL}, {"background", MenuParse_background, 0, NULL},
+ {"ownerdraw", MenuParse_ownerdraw, 0, NULL}, {"ownerdrawFlag", MenuParse_ownerdrawFlag, 0, NULL},
+ {"outOfBoundsClick", MenuParse_outOfBounds, 0, NULL}, {"soundLoop", MenuParse_soundLoop, 0, NULL},
+ {"itemDef", MenuParse_itemDef, 0, NULL}, {"cinematic", MenuParse_cinematic, 0, NULL},
+ {"popup", MenuParse_popup, 0, NULL}, {"fadeClamp", MenuParse_fadeClamp, 0, NULL},
+ {"fadeCycle", MenuParse_fadeCycle, 0, NULL}, {"fadeAmount", MenuParse_fadeAmount, 0, NULL},
+ {NULL, voidFunction2, 0, NULL}};
keywordHash_t *menuParseKeywordHash[KEYWORDHASH_SIZE];
@@ -5872,14 +7515,14 @@ keywordHash_t *menuParseKeywordHash[KEYWORDHASH_SIZE];
Menu_SetupKeywordHash
===============
*/
-void Menu_SetupKeywordHash( void )
+void Menu_SetupKeywordHash(void)
{
- int i;
+ int i;
- memset( menuParseKeywordHash, 0, sizeof( menuParseKeywordHash ) );
+ memset(menuParseKeywordHash, 0, sizeof(menuParseKeywordHash));
- for(i = 0; menuParseKeywords[ i ].keyword; i++ )
- KeywordHash_Add( menuParseKeywordHash, &menuParseKeywords[ i ] );
+ for (i = 0; menuParseKeywords[i].keyword; i++)
+ KeywordHash_Add(menuParseKeywordHash, &menuParseKeywords[i]);
}
/*
@@ -5887,39 +7530,46 @@ void Menu_SetupKeywordHash( void )
Menu_Parse
===============
*/
-qboolean Menu_Parse(int handle, menuDef_t *menu) {
- pc_token_t token;
- keywordHash_t *key;
+qboolean Menu_Parse(int handle, menuDef_t *menu)
+{
+ pc_token_t token;
+ keywordHash_t *key;
- if (!trap_Parse_ReadToken(handle, &token))
- return qfalse;
- if (*token.string != '{') {
- return qfalse;
- }
+ if (!trap_Parse_ReadToken(handle, &token))
+ return qfalse;
+
+ if (*token.string != '{')
+ return qfalse;
- while ( 1 ) {
+ while (1)
+ {
+ memset(&token, 0, sizeof(pc_token_t));
- memset(&token, 0, sizeof(pc_token_t));
- if (!trap_Parse_ReadToken(handle, &token)) {
- PC_SourceError(handle, "end of file inside menu\n");
- return qfalse;
- }
+ if (!trap_Parse_ReadToken(handle, &token))
+ {
+ PC_SourceError(handle, "end of file inside menu\n");
+ return qfalse;
+ }
- if (*token.string == '}') {
- return qtrue;
- }
+ if (*token.string == '}')
+ return qtrue;
- key = KeywordHash_Find(menuParseKeywordHash, token.string);
- if (!key) {
- PC_SourceError(handle, "unknown menu keyword %s", token.string);
- continue;
- }
- if ( !key->func((itemDef_t*)menu, handle) ) {
- PC_SourceError(handle, "couldn't parse menu keyword %s", token.string);
- return qfalse;
+ key = KeywordHash_Find(menuParseKeywordHash, token.string);
+
+ if (!key)
+ {
+ PC_SourceError(handle, "unknown menu keyword %s", token.string);
+ continue;
+ }
+
+ if (!key->func((itemDef_t *)menu, handle))
+ {
+ PC_SourceError(handle, "couldn't parse menu keyword %s", token.string);
+ return qfalse;
+ }
}
- }
- return qfalse; // bk001205 - LCC missing return value
+
+ return qfalse;
}
/*
@@ -5927,189 +7577,211 @@ qboolean Menu_Parse(int handle, menuDef_t *menu) {
Menu_New
===============
*/
-void Menu_New(int handle) {
- menuDef_t *menu = &Menus[menuCount];
+void Menu_New(int handle)
+{
+ menuDef_t *menu = &Menus[menuCount];
+
+ if (menuCount < MAX_MENUS)
+ {
+ Menu_Init(menu);
- if (menuCount < MAX_MENUS) {
- Menu_Init(menu);
- if (Menu_Parse(handle, menu)) {
- Menu_PostParse(menu);
- menuCount++;
+ if (Menu_Parse(handle, menu))
+ {
+ Menu_PostParse(menu);
+ menuCount++;
+ }
}
- }
}
-int Menu_Count( void ) {
- return menuCount;
+int Menu_Count(void) { return menuCount; }
+
+void Menu_UpdateAll(void)
+{
+ int i;
+
+ for (i = 0; i < openMenuCount; i++)
+ Menu_Update(menuStack[i]);
}
-void Menu_PaintAll( void ) {
- int i;
+void Menu_PaintAll(void)
+{
+ int i;
- if( g_editingField || g_waitingForKey )
- DC->setCVar( "ui_hideCursor", "1" );
- else
- DC->setCVar( "ui_hideCursor", "0" );
+ if (g_editingField || g_waitingForKey)
+ DC->setCVar("ui_hideCursor", "1");
+ else
+ DC->setCVar("ui_hideCursor", "0");
- if (captureFunc) {
- captureFunc(captureData);
- }
+ if (captureFunc != voidFunction)
+ {
+ if (captureFuncExpiry > 0 && DC->realTime > captureFuncExpiry)
+ UI_RemoveCaptureFunc();
+ else
+ captureFunc(captureData);
+ }
- for (i = 0; i < Menu_Count(); i++) {
- Menu_Paint(&Menus[i], qfalse);
- }
+ for (i = 0; i < openMenuCount; i++)
+ Menu_Paint(menuStack[i], qfalse);
- if (debugMode) {
- vec4_t v = {1, 1, 1, 1};
- DC->drawText(5, 25, .5, v, va("fps: %f", DC->FPS), 0, 0, 0);
- }
+ if (DC->getCVarValue("ui_developer"))
+ {
+ vec4_t v = {1, 1, 1, 1};
+ UI_Text_Paint(5, 25, .5, v, va("fps: %f", DC->FPS), 0, 0, 0);
+ }
}
-void Menu_Reset( void )
-{
- menuCount = 0;
-}
+void Menu_Reset(void) { menuCount = 0; }
-displayContextDef_t *Display_GetContext( void ) {
- return DC;
-}
+displayContextDef_t *Display_GetContext(void) { return DC; }
-void *Display_CaptureItem(int x, int y) {
- int i;
+void *Display_CaptureItem(int x, int y)
+{
+ int i;
- for (i = 0; i < menuCount; i++) {
- // turn off focus each item
- // menu->items[i].window.flags &= ~WINDOW_HASFOCUS;
- if (Rect_ContainsPoint(&Menus[i].window.rect, x, y)) {
- return &Menus[i];
+ for (i = 0; i < menuCount; i++)
+ {
+ if (Rect_ContainsPoint(&Menus[i].window.rect, x, y))
+ return &Menus[i];
}
- }
- return NULL;
-}
+ return NULL;
+}
// FIXME:
-qboolean Display_MouseMove(void *p, int x, int y) {
- int i;
- menuDef_t *menu = 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;
+qboolean Display_MouseMove(void *p, float x, float y)
+{
+ int i;
+ menuDef_t *menu = p;
-}
+ if (menu == NULL)
+ {
+ menu = Menu_GetFocused();
+
+ if (menu)
+ {
+ if (menu->window.flags & WINDOW_POPUP)
+ {
+ Menu_HandleMouseMove(menu, x, y);
+ return qtrue;
+ }
+ }
-int Display_CursorType(int x, int y) {
- int i;
- for (i = 0; i < menuCount; i++) {
- rectDef_t r2;
- r2.x = Menus[i].window.rect.x - 3;
- r2.y = Menus[i].window.rect.y - 3;
- r2.w = r2.h = 7;
- if (Rect_ContainsPoint(&r2, x, y)) {
- return CURSOR_SIZER;
+ 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 CURSOR_ARROW;
+
+ return qtrue;
}
+int Display_CursorType(int x, int y)
+{
+ int i;
-void Display_HandleKey(int key, qboolean down, int x, int y) {
- menuDef_t *menu = Display_CaptureItem(x, y);
- if (menu == NULL) {
- menu = Menu_GetFocused();
- }
- if (menu) {
- Menu_HandleKey(menu, key, down );
- }
-}
+ for (i = 0; i < menuCount; i++)
+ {
+ rectDef_t r2;
+ r2.x = Menus[i].window.rect.x - 3;
+ r2.y = Menus[i].window.rect.y - 3;
+ r2.w = r2.h = 7;
-static void Window_CacheContents(windowDef_t *window) {
- if (window) {
- if (window->cinematicName) {
- int cin = DC->playCinematic(window->cinematicName, 0, 0, 0, 0);
- DC->stopCinematic(cin);
+ if (Rect_ContainsPoint(&r2, x, y))
+ return CURSOR_SIZER;
}
- }
+
+ return CURSOR_ARROW;
}
+void Display_HandleKey(int key, qboolean down, int x, int y)
+{
+ menuDef_t *menu = Display_CaptureItem(x, y);
-static void Item_CacheContents(itemDef_t *item) {
- if (item) {
- Window_CacheContents(&item->window);
- }
+ if (menu == NULL)
+ menu = Menu_GetFocused();
+ if (menu)
+ Menu_HandleKey(menu, key, down);
}
-static void Menu_CacheContents(menuDef_t *menu) {
- if (menu) {
- int i;
- Window_CacheContents(&menu->window);
- for (i = 0; i < menu->itemCount; i++) {
- Item_CacheContents(menu->items[i]);
- }
-
- if (menu->soundName && *menu->soundName) {
- DC->registerSound(menu->soundName, qfalse);
+static void Window_CacheContents(Window *window)
+{
+ if (window)
+ {
+ if (window->cinematicName)
+ {
+ int cin = DC->playCinematic(window->cinematicName, 0, 0, 0, 0);
+ DC->stopCinematic(cin);
+ }
}
- }
-
}
-void Display_CacheAll( void ) {
- int i;
- for (i = 0; i < menuCount; i++) {
- Menu_CacheContents(&Menus[i]);
- }
+static void Item_CacheContents(itemDef_t *item)
+{
+ if (item)
+ Window_CacheContents(&item->window);
}
+static void Menu_CacheContents(menuDef_t *menu)
+{
+ if (menu)
+ {
+ int i;
+ Window_CacheContents(&menu->window);
-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;
+ for (i = 0; i < menu->itemCount; i++)
+ Item_CacheContents(menu->items[i]);
- if (!(menu->items[i]->window.flags & (WINDOW_VISIBLE | WINDOW_FORCED))) {
- continue;
- }
+ if (menu->soundName && *menu->soundName)
+ DC->registerSound(menu->soundName, qfalse);
+ }
+}
- if (menu->items[i]->window.flags & WINDOW_DECORATION) {
- continue;
- }
+void Display_CacheAll(void)
+{
+ int i;
- 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;
+ for (i = 0; i < menuCount; i++)
+ Menu_CacheContents(&Menus[i]);
+}
+
+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++)
+ {
+ 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;
+ }
}
- } else {
- return qtrue;
- }
}
- }
-
}
- }
- return qfalse;
-}
+ return qfalse;
+}