You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
470 lines
12 KiB
470 lines
12 KiB
/* |
|
sys_con.c - win32 dedicated and developer console |
|
Copyright (C) 2007 Uncle Mike |
|
|
|
This program is free software: you can redistribute it and/or modify |
|
it under the terms of the GNU General Public License as published by |
|
the Free Software Foundation, either version 3 of the License, or |
|
(at your option) any later version. |
|
|
|
This program is distributed in the hope that it will be useful, |
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
GNU General Public License for more details. |
|
*/ |
|
|
|
#include "common.h" |
|
#include "xash3d_mathlib.h" |
|
|
|
/* |
|
=============================================================================== |
|
|
|
WIN32 CONSOLE |
|
|
|
=============================================================================== |
|
*/ |
|
|
|
// console defines |
|
#define SUBMIT_ID 1 // "submit" button |
|
#define QUIT_ON_ESCAPE_ID 2 // escape event |
|
#define EDIT_ID 110 |
|
#define INPUT_ID 109 |
|
#define IDI_ICON1 101 |
|
|
|
#define SYSCONSOLE "XashConsole" |
|
#define COMMAND_HISTORY 64 // system console keep more commands than game console |
|
|
|
typedef struct |
|
{ |
|
char title[64]; |
|
HWND hWnd; |
|
HWND hwndBuffer; |
|
HWND hwndButtonSubmit; |
|
HBRUSH hbrEditBackground; |
|
HFONT hfBufferFont; |
|
HWND hwndInputLine; |
|
string consoleText; |
|
string returnedText; |
|
string historyLines[COMMAND_HISTORY]; |
|
int nextHistoryLine; |
|
int historyLine; |
|
int status; |
|
int windowWidth, windowHeight; |
|
WNDPROC SysInputLineWndProc; |
|
size_t outLen; |
|
|
|
// log stuff |
|
qboolean log_active; |
|
char log_path[MAX_SYSPATH]; |
|
} WinConData; |
|
|
|
static WinConData s_wcd; |
|
|
|
void Wcon_ShowConsole( qboolean show ) |
|
{ |
|
if( !s_wcd.hWnd || show == s_wcd.status ) |
|
return; |
|
|
|
s_wcd.status = show; |
|
if( show ) |
|
{ |
|
ShowWindow( s_wcd.hWnd, SW_SHOWNORMAL ); |
|
SendMessage( s_wcd.hwndBuffer, EM_LINESCROLL, 0, 0xffff ); |
|
} |
|
else ShowWindow( s_wcd.hWnd, SW_HIDE ); |
|
} |
|
|
|
void Wcon_DisableInput( void ) |
|
{ |
|
if( host.type != HOST_DEDICATED ) return; |
|
SendMessage( s_wcd.hwndButtonSubmit, WM_ENABLE, 0, 0 ); |
|
SendMessage( s_wcd.hwndInputLine, WM_ENABLE, 0, 0 ); |
|
} |
|
|
|
void Wcon_SetInputText( const char *inputText ) |
|
{ |
|
if( host.type != HOST_DEDICATED ) return; |
|
SetWindowText( s_wcd.hwndInputLine, inputText ); |
|
SendMessage( s_wcd.hwndInputLine, EM_SETSEL, Q_strlen( inputText ), -1 ); |
|
} |
|
|
|
static void Wcon_Clear_f( void ) |
|
{ |
|
if( host.type != HOST_DEDICATED ) return; |
|
SendMessage( s_wcd.hwndBuffer, EM_SETSEL, 0, -1 ); |
|
SendMessage( s_wcd.hwndBuffer, EM_REPLACESEL, FALSE, (LPARAM)"" ); |
|
UpdateWindow( s_wcd.hwndBuffer ); |
|
} |
|
|
|
static int Wcon_KeyEvent( int key, qboolean down ) |
|
{ |
|
char inputBuffer[1024]; |
|
|
|
if( !down ) |
|
return 0; |
|
|
|
switch( key ) |
|
{ |
|
case VK_TAB: |
|
GetWindowText( s_wcd.hwndInputLine, inputBuffer, sizeof( inputBuffer )); |
|
Cmd_AutoComplete( inputBuffer ); |
|
Wcon_SetInputText( inputBuffer ); |
|
return 1; |
|
case VK_DOWN: |
|
if( s_wcd.historyLine == s_wcd.nextHistoryLine ) |
|
return 0; |
|
s_wcd.historyLine++; |
|
Wcon_SetInputText( s_wcd.historyLines[s_wcd.historyLine % COMMAND_HISTORY] ); |
|
return 1; |
|
case VK_UP: |
|
if( s_wcd.nextHistoryLine - s_wcd.historyLine < COMMAND_HISTORY && s_wcd.historyLine > 0 ) |
|
s_wcd.historyLine--; |
|
Wcon_SetInputText( s_wcd.historyLines[s_wcd.historyLine % COMMAND_HISTORY] ); |
|
return 1; |
|
} |
|
return 0; |
|
} |
|
|
|
static long _stdcall Wcon_WndProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) |
|
{ |
|
switch( uMsg ) |
|
{ |
|
case WM_ACTIVATE: |
|
if( LOWORD( wParam ) != WA_INACTIVE ) |
|
SetFocus( s_wcd.hwndInputLine ); |
|
break; |
|
case WM_CLOSE: |
|
if( host.status == HOST_ERR_FATAL ) |
|
{ |
|
// send windows message |
|
PostQuitMessage( 0 ); |
|
} |
|
else Sys_Quit(); // otherwise |
|
return 0; |
|
case WM_CTLCOLORSTATIC: |
|
if((HWND)lParam == s_wcd.hwndBuffer ) |
|
{ |
|
SetBkColor((HDC)wParam, RGB( 0x90, 0x90, 0x90 )); |
|
SetTextColor((HDC)wParam, RGB( 0xff, 0xff, 0xff )); |
|
return (long)s_wcd.hbrEditBackground; |
|
} |
|
break; |
|
case WM_COMMAND: |
|
if( wParam == SUBMIT_ID ) |
|
{ |
|
SendMessage( s_wcd.hwndInputLine, WM_CHAR, 13, 0L ); |
|
SetFocus( s_wcd.hwndInputLine ); |
|
} |
|
break; |
|
case WM_HOTKEY: |
|
switch( LOWORD( wParam )) |
|
{ |
|
case QUIT_ON_ESCAPE_ID: |
|
PostQuitMessage( 0 ); |
|
break; |
|
} |
|
break; |
|
case WM_CREATE: |
|
s_wcd.hbrEditBackground = CreateSolidBrush( RGB( 0x90, 0x90, 0x90 )); |
|
break; |
|
} |
|
return DefWindowProc( hWnd, uMsg, wParam, lParam ); |
|
} |
|
|
|
long _stdcall Wcon_InputLineProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam ) |
|
{ |
|
char inputBuffer[1024]; |
|
|
|
switch( uMsg ) |
|
{ |
|
case WM_KILLFOCUS: |
|
if(( HWND )wParam == s_wcd.hWnd ) |
|
{ |
|
SetFocus( hWnd ); |
|
return 0; |
|
} |
|
break; |
|
case WM_SYSKEYDOWN: |
|
case WM_KEYDOWN: |
|
if( Wcon_KeyEvent( LOWORD( wParam ), true )) |
|
return 0; |
|
break; |
|
case WM_SYSKEYUP: |
|
case WM_KEYUP: |
|
if( Wcon_KeyEvent( LOWORD( wParam ), false )) |
|
return 0; |
|
break; |
|
case WM_CHAR: |
|
if( Wcon_KeyEvent( wParam, true )) |
|
return 0; |
|
if( wParam == 13 && host.status != HOST_ERR_FATAL ) |
|
{ |
|
GetWindowText( s_wcd.hwndInputLine, inputBuffer, sizeof( inputBuffer )); |
|
Q_strncat( s_wcd.consoleText, inputBuffer, sizeof( s_wcd.consoleText ) - Q_strlen( s_wcd.consoleText ) - 5 ); |
|
Q_strcat( s_wcd.consoleText, "\n" ); |
|
SetWindowText( s_wcd.hwndInputLine, "" ); |
|
Con_Printf( ">%s\n", inputBuffer ); |
|
|
|
// copy line to history buffer |
|
Q_strncpy( s_wcd.historyLines[s_wcd.nextHistoryLine % COMMAND_HISTORY], inputBuffer, MAX_STRING ); |
|
s_wcd.nextHistoryLine++; |
|
s_wcd.historyLine = s_wcd.nextHistoryLine; |
|
return 0; |
|
} |
|
break; |
|
} |
|
return CallWindowProc( s_wcd.SysInputLineWndProc, hWnd, uMsg, wParam, lParam ); |
|
} |
|
|
|
|
|
/* |
|
=============================================================================== |
|
|
|
WIN32 IO |
|
|
|
=============================================================================== |
|
*/ |
|
/* |
|
================ |
|
Con_WinPrint |
|
|
|
print into window console |
|
================ |
|
*/ |
|
void Wcon_WinPrint( const char *pMsg ) |
|
{ |
|
size_t len = Q_strlen( pMsg ); |
|
|
|
// replace selection instead of appending if we're overflowing |
|
s_wcd.outLen += len; |
|
if( s_wcd.outLen >= 0x7fff ) |
|
{ |
|
SendMessage( s_wcd.hwndBuffer, EM_SETSEL, 0, -1 ); |
|
s_wcd.outLen = len; |
|
} |
|
|
|
SendMessage( s_wcd.hwndBuffer, EM_REPLACESEL, 0, (LPARAM)pMsg ); |
|
|
|
// put this text into the windows console |
|
SendMessage( s_wcd.hwndBuffer, EM_LINESCROLL, 0, 0xffff ); |
|
SendMessage( s_wcd.hwndBuffer, EM_SCROLLCARET, 0, 0 ); |
|
} |
|
|
|
|
|
/* |
|
================ |
|
Con_CreateConsole |
|
|
|
create win32 console |
|
================ |
|
*/ |
|
void Wcon_CreateConsole( void ) |
|
{ |
|
HDC hDC; |
|
WNDCLASS wc; |
|
RECT rect; |
|
int nHeight; |
|
int swidth, sheight, fontsize; |
|
int DEDSTYLE = WS_POPUPWINDOW | WS_CAPTION; |
|
int CONSTYLE = WS_CHILD|WS_VISIBLE|WS_VSCROLL|WS_BORDER|WS_EX_CLIENTEDGE|ES_LEFT|ES_MULTILINE|ES_AUTOVSCROLL|ES_READONLY; |
|
string FontName; |
|
|
|
wc.style = 0; |
|
wc.lpfnWndProc = (WNDPROC)Wcon_WndProc; |
|
wc.cbClsExtra = 0; |
|
wc.cbWndExtra = 0; |
|
wc.hInstance = host.hInst; |
|
wc.hIcon = LoadIcon( host.hInst, MAKEINTRESOURCE( IDI_ICON1 )); |
|
wc.hCursor = LoadCursor( NULL, IDC_ARROW ); |
|
wc.hbrBackground = (void *)COLOR_3DSHADOW; |
|
wc.lpszClassName = SYSCONSOLE; |
|
wc.lpszMenuName = 0; |
|
|
|
if( Sys_CheckParm( "-log" )) |
|
s_wcd.log_active = true; |
|
|
|
if( host.type == HOST_NORMAL ) |
|
{ |
|
rect.left = 0; |
|
rect.right = 536; |
|
rect.top = 0; |
|
rect.bottom = 364; |
|
Q_strncpy( FontName, "Fixedsys", sizeof( FontName )); |
|
Q_strncpy( s_wcd.title, va( "Xash3D %s", XASH_VERSION ), sizeof( s_wcd.title )); |
|
Q_strncpy( s_wcd.log_path, "engine.log", sizeof( s_wcd.log_path )); |
|
fontsize = 8; |
|
} |
|
else // dedicated console |
|
{ |
|
rect.left = 0; |
|
rect.right = 640; |
|
rect.top = 0; |
|
rect.bottom = 392; |
|
Q_strncpy( FontName, "System", sizeof( FontName )); |
|
Q_strncpy( s_wcd.title, va( "XashDS %s", XASH_VERSION ), sizeof( s_wcd.title )); |
|
Q_strncpy( s_wcd.log_path, "dedicated.log", sizeof( s_wcd.log_path )); |
|
s_wcd.log_active = true; // always make log |
|
fontsize = 14; |
|
} |
|
|
|
if( !RegisterClass( &wc )) |
|
{ |
|
// print into log |
|
Con_Reportf( S_ERROR "Can't register window class '%s'\n", SYSCONSOLE ); |
|
return; |
|
} |
|
|
|
AdjustWindowRect( &rect, DEDSTYLE, FALSE ); |
|
|
|
hDC = GetDC( GetDesktopWindow() ); |
|
swidth = GetDeviceCaps( hDC, HORZRES ); |
|
sheight = GetDeviceCaps( hDC, VERTRES ); |
|
ReleaseDC( GetDesktopWindow(), hDC ); |
|
|
|
s_wcd.windowWidth = rect.right - rect.left; |
|
s_wcd.windowHeight = rect.bottom - rect.top; |
|
|
|
s_wcd.hWnd = CreateWindowEx( WS_EX_DLGMODALFRAME, SYSCONSOLE, s_wcd.title, DEDSTYLE, ( swidth - 600 ) / 2, ( sheight - 450 ) / 2 , rect.right - rect.left + 1, rect.bottom - rect.top + 1, NULL, NULL, host.hInst, NULL ); |
|
if( s_wcd.hWnd == NULL ) |
|
{ |
|
Con_Reportf( S_ERROR "Can't create window '%s'\n", s_wcd.title ); |
|
return; |
|
} |
|
|
|
// create fonts |
|
hDC = GetDC( s_wcd.hWnd ); |
|
nHeight = -MulDiv( fontsize, GetDeviceCaps( hDC, LOGPIXELSY ), 72 ); |
|
s_wcd.hfBufferFont = CreateFont( nHeight, 0, 0, 0, FW_LIGHT, 0, 0, 0, DEFAULT_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, FF_MODERN|FIXED_PITCH, FontName ); |
|
ReleaseDC( s_wcd.hWnd, hDC ); |
|
|
|
if( host.type == HOST_DEDICATED ) |
|
{ |
|
// create the input line |
|
s_wcd.hwndInputLine = CreateWindowEx( WS_EX_CLIENTEDGE, "edit", NULL, WS_CHILD|WS_VISIBLE|WS_BORDER|ES_LEFT|ES_AUTOHSCROLL, 0, 366, 550, 25, s_wcd.hWnd, (HMENU)INPUT_ID, host.hInst, NULL ); |
|
|
|
s_wcd.hwndButtonSubmit = CreateWindow( "button", NULL, BS_PUSHBUTTON|WS_VISIBLE|WS_CHILD|BS_DEFPUSHBUTTON, 552, 367, 87, 25, s_wcd.hWnd, (HMENU)SUBMIT_ID, host.hInst, NULL ); |
|
SendMessage( s_wcd.hwndButtonSubmit, WM_SETTEXT, 0, ( LPARAM ) "submit" ); |
|
} |
|
|
|
// create the scrollbuffer |
|
GetClientRect( s_wcd.hWnd, &rect ); |
|
|
|
s_wcd.hwndBuffer = CreateWindowEx( WS_EX_DLGMODALFRAME|WS_EX_CLIENTEDGE, "edit", NULL, CONSTYLE, 0, 0, rect.right - rect.left, Q_min(365, rect.bottom), s_wcd.hWnd, (HMENU)EDIT_ID, host.hInst, NULL ); |
|
SendMessage( s_wcd.hwndBuffer, WM_SETFONT, (WPARAM)s_wcd.hfBufferFont, 0 ); |
|
|
|
if( host.type == HOST_DEDICATED ) |
|
{ |
|
#ifdef XASH_64BIT |
|
s_wcd.SysInputLineWndProc = (WNDPROC)SetWindowLongPtr( s_wcd.hwndInputLine, GWLP_WNDPROC, (LONG_PTR)Wcon_InputLineProc ); |
|
#else |
|
s_wcd.SysInputLineWndProc = (WNDPROC)SetWindowLong( s_wcd.hwndInputLine, GWL_WNDPROC, (long)Wcon_InputLineProc ); |
|
#endif |
|
SendMessage( s_wcd.hwndInputLine, WM_SETFONT, ( WPARAM )s_wcd.hfBufferFont, 0 ); |
|
} |
|
|
|
// show console if needed |
|
if( host.con_showalways ) |
|
{ |
|
// make console visible |
|
ShowWindow( s_wcd.hWnd, SW_SHOWDEFAULT ); |
|
UpdateWindow( s_wcd.hWnd ); |
|
SetForegroundWindow( s_wcd.hWnd ); |
|
|
|
if( host.type != HOST_DEDICATED ) |
|
SetFocus( s_wcd.hWnd ); |
|
else SetFocus( s_wcd.hwndInputLine ); |
|
s_wcd.status = true; |
|
} |
|
else s_wcd.status = false; |
|
} |
|
|
|
/* |
|
================ |
|
Con_InitConsoleCommands |
|
|
|
register console commands (dedicated only) |
|
================ |
|
*/ |
|
void Wcon_InitConsoleCommands( void ) |
|
{ |
|
if( host.type != HOST_DEDICATED ) return; |
|
Cmd_AddCommand( "clear", Wcon_Clear_f, "clear console history" ); |
|
} |
|
|
|
/* |
|
================ |
|
Con_DestroyConsole |
|
|
|
destroy win32 console |
|
================ |
|
*/ |
|
void Wcon_DestroyConsole( void ) |
|
{ |
|
// last text message into console or log |
|
Con_Reportf( "Sys_FreeLibrary: Unloading xash.dll\n" ); |
|
|
|
Sys_CloseLog(); |
|
|
|
if( s_wcd.hWnd ) |
|
{ |
|
DeleteObject( s_wcd.hbrEditBackground ); |
|
DeleteObject( s_wcd.hfBufferFont ); |
|
|
|
if( host.type == HOST_DEDICATED ) |
|
{ |
|
ShowWindow( s_wcd.hwndButtonSubmit, SW_HIDE ); |
|
DestroyWindow( s_wcd.hwndButtonSubmit ); |
|
s_wcd.hwndButtonSubmit = 0; |
|
|
|
ShowWindow( s_wcd.hwndInputLine, SW_HIDE ); |
|
DestroyWindow( s_wcd.hwndInputLine ); |
|
s_wcd.hwndInputLine = 0; |
|
} |
|
|
|
ShowWindow( s_wcd.hwndBuffer, SW_HIDE ); |
|
DestroyWindow( s_wcd.hwndBuffer ); |
|
s_wcd.hwndBuffer = 0; |
|
|
|
ShowWindow( s_wcd.hWnd, SW_HIDE ); |
|
DestroyWindow( s_wcd.hWnd ); |
|
s_wcd.hWnd = 0; |
|
} |
|
|
|
UnregisterClass( SYSCONSOLE, host.hInst ); |
|
|
|
// place it here in case Sys_Crash working properly |
|
if( host.hMutex ) CloseHandle( host.hMutex ); |
|
} |
|
|
|
/* |
|
================ |
|
Con_Input |
|
|
|
returned input text |
|
================ |
|
*/ |
|
char *Wcon_Input( void ) |
|
{ |
|
if( s_wcd.consoleText[0] == 0 ) |
|
return NULL; |
|
|
|
Q_strncpy( s_wcd.returnedText, s_wcd.consoleText, sizeof( s_wcd.returnedText )); |
|
s_wcd.consoleText[0] = 0; |
|
|
|
return s_wcd.returnedText; |
|
} |
|
|
|
/* |
|
================ |
|
Con_SetFocus |
|
|
|
change focus to console hwnd |
|
================ |
|
*/ |
|
void Wcon_RegisterHotkeys( void ) |
|
{ |
|
SetFocus( s_wcd.hWnd ); |
|
|
|
// user can hit escape for quit |
|
RegisterHotKey( s_wcd.hWnd, QUIT_ON_ESCAPE_ID, 0, VK_ESCAPE ); |
|
}
|
|
|