2013-12-19 18:41:57 +00:00
# include "quakedef.h"
# include "glquake.h"
# include "web/ftejslib.h"
extern cvar_t vid_hardwaregamma ;
extern cvar_t gl_lateswap ;
extern int gammaworks ;
extern qboolean vid_isfullscreen ;
qboolean mouseactive ;
extern qboolean mouseusedforgui ;
static void * GLVID_getsdlglfunction ( char * functionname )
{
return NULL ;
}
2014-10-11 19:39:45 +00:00
static void IN_JoystickButtonEvent ( int joydevid , int button , int ispressed )
{
if ( button > = 32 + 4 )
return ;
IN_KeyEvent ( joydevid , ispressed , K_JOY1 + button , 0 ) ;
}
2013-12-19 18:41:57 +00:00
static void VID_Resized ( int width , int height )
{
extern cvar_t vid_conautoscale , vid_conwidth ;
vid . pixelwidth = width ;
vid . pixelheight = height ;
//Con_Printf("Resized: %i %i\n", vid.pixelwidth, vid.pixelheight);
Cvar_ForceCallback ( & vid_conautoscale ) ;
Cvar_ForceCallback ( & vid_conwidth ) ;
}
static unsigned int domkeytoquake ( unsigned int code )
{
2014-08-25 07:35:41 +00:00
unsigned char tab [ 256 ] =
2013-12-19 18:41:57 +00:00
{
/* 0*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , K_BACKSPACE , K_TAB , 0 , 0 , 0 , K_ENTER , 0 , 0 ,
/* 16*/ K_SHIFT , K_CTRL , K_ALT , K_PAUSE , K_CAPSLOCK , 0 , 0 , 0 , 0 , 0 , 0 , K_ESCAPE , 0 , 0 , 0 , 0 ,
/* 32*/ ' ' , K_PGUP , K_PGDN , K_END , K_HOME , K_LEFTARROW , K_UPARROW , K_RIGHTARROW , K_DOWNARROW , 0 , 0 , 0 , K_PRINTSCREEN , K_INS , K_DEL , 0 ,
2015-04-14 23:12:17 +00:00
/* 48*/ ' 0 ' , ' 1 ' , ' 2 ' , ' 3 ' , ' 4 ' , ' 5 ' , ' 6 ' , ' 7 ' , ' 8 ' , ' 9 ' , 0 , ' ; ' , 0 , ' = ' , 0 , 0 ,
2013-12-19 18:41:57 +00:00
/* 64*/ 0 , ' a ' , ' b ' , ' c ' , ' d ' , ' e ' , ' f ' , ' g ' , ' h ' , ' i ' , ' j ' , ' k ' , ' l ' , ' m ' , ' n ' , ' o ' ,
/* 80*/ ' p ' , ' q ' , ' r ' , ' s ' , ' t ' , ' u ' , ' v ' , ' w ' , ' x ' , ' y ' , ' z ' , K_LWIN , K_RWIN , K_APP , 0 , 0 ,
/* 96*/ K_KP_INS , K_KP_END , K_KP_DOWNARROW , K_KP_PGDN , K_KP_LEFTARROW , K_KP_5 , K_KP_RIGHTARROW , K_KP_HOME , K_KP_UPARROW , K_KP_PGDN , K_KP_STAR , K_KP_PLUS , 0 , K_KP_MINUS , K_KP_DEL , K_KP_SLASH ,
/*112*/ K_F1 , K_F2 , K_F3 , K_F4 , K_F5 , K_F6 , K_F7 , K_F8 , K_F9 , K_F10 , K_F11 , K_F12 , 0 , 0 , 0 , 0 ,
/*128*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
/*144*/ K_KP_NUMLOCK , K_SCRLCK , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
2014-08-27 08:41:31 +00:00
/*160*/ 0 , 0 , 0 , ' # ' , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , ' - ' , 0 , 0 ,
2013-12-19 18:41:57 +00:00
/*176*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , ' ; ' , ' = ' , ' , ' , ' - ' , ' . ' , ' / ' ,
/*192*/ ' ` ' , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
/*208*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , ' [ ' , ' \\ ' , ' ] ' , ' \' ' , ' ` ' ,
/*224*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
/*240*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
} ;
if ( ! code )
return 0 ;
if ( code > = sizeof ( tab ) / sizeof ( tab [ 0 ] ) )
{
Con_DPrintf ( " You just pressed key %u, but I don't know what its meant to be \n " , code ) ;
return 0 ;
}
if ( ! tab [ code ] )
Con_DPrintf ( " You just pressed key %u, but I don't know what its meant to be \n " , code ) ;
2014-08-25 07:35:41 +00:00
// Con_DPrintf("You just pressed dom key %u, which is quake key %u\n", code, tab[code]);
return tab [ code ] ;
}
static unsigned int domkeytoshift ( unsigned int code )
{
unsigned char tab [ 256 ] =
{
/* 0*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , K_BACKSPACE , K_TAB , 0 , 0 , 0 , K_ENTER , 0 , 0 ,
/* 16*/ K_SHIFT , K_CTRL , K_ALT , K_PAUSE , K_CAPSLOCK , 0 , 0 , 0 , 0 , 0 , 0 , K_ESCAPE , 0 , 0 , 0 , 0 ,
/* 32*/ ' ' , K_PGUP , K_PGDN , K_END , K_HOME , K_LEFTARROW , K_UPARROW , K_RIGHTARROW , K_DOWNARROW , 0 , 0 , 0 , K_PRINTSCREEN , K_INS , K_DEL , 0 ,
2015-04-14 23:12:17 +00:00
/* 48*/ ' ) ' , ' ! ' , ' \" ' , 0 /*<2A> */ , ' $ ' , ' % ' , ' ^ ' , ' & ' , ' * ' , ' ( ' , 0 , ' : ' , 0 , ' + ' , 0 , 0 ,
2014-08-25 07:35:41 +00:00
/* 64*/ 0 , ' A ' , ' B ' , ' C ' , ' D ' , ' E ' , ' F ' , ' G ' , ' H ' , ' I ' , ' J ' , ' K ' , ' L ' , ' M ' , ' N ' , ' O ' ,
/* 80*/ ' P ' , ' Q ' , ' R ' , ' S ' , ' T ' , ' U ' , ' V ' , ' W ' , ' X ' , ' Y ' , ' Z ' , K_LWIN , K_RWIN , K_APP , 0 , 0 ,
/* 96*/ K_KP_INS , K_KP_END , K_KP_DOWNARROW , K_KP_PGDN , K_KP_LEFTARROW , K_KP_5 , K_KP_RIGHTARROW , K_KP_HOME , K_KP_UPARROW , K_KP_PGDN , K_KP_STAR , K_KP_PLUS , 0 , K_KP_MINUS , K_KP_DEL , K_KP_SLASH ,
/*112*/ K_F1 , K_F2 , K_F3 , K_F4 , K_F5 , K_F6 , K_F7 , K_F8 , K_F9 , K_F10 , K_F11 , K_F12 , 0 , 0 , 0 , 0 ,
/*128*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
/*144*/ K_KP_NUMLOCK , K_SCRLCK , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
2014-08-27 08:41:31 +00:00
/*160*/ 0 , 0 , 0 , ' ~ ' , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , ' _ ' , 0 , 0 ,
2014-08-25 07:35:41 +00:00
/*176*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , ' : ' , ' + ' , ' < ' , ' _ ' , ' > ' , ' ? ' ,
/*192*/ ' ` ' , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
/*208*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , ' { ' , ' | ' , ' } ' , ' @ ' , ' ` ' ,
/*224*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
/*240*/ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 ,
} ;
if ( ! code )
return 0 ;
if ( code > = sizeof ( tab ) / sizeof ( tab [ 0 ] ) )
{
Con_DPrintf ( " You just pressed key %u, but I don't know what its meant to be \n " , code ) ;
return 0 ;
}
if ( ! tab [ code ] )
Con_DPrintf ( " You just pressed key %u, but I don't know what its meant to be \n " , code ) ;
// Con_DPrintf("You just pressed dom key %u, which is quake key %u\n", code, tab[code]);
2013-12-19 18:41:57 +00:00
return tab [ code ] ;
}
static int DOM_KeyEvent ( int devid , int down , int scan , int uni )
{
2014-08-25 07:35:41 +00:00
extern int shift_down ;
2015-08-28 03:13:45 +00:00
// Con_Printf("Key %s %i %i:%c\n", down?"down":"up", scan, uni, uni?(char)uni:' ');
2014-08-25 07:35:41 +00:00
if ( shift_down )
{
uni = domkeytoshift ( scan ) ;
scan = domkeytoquake ( scan ) ;
uni = ( uni > = 32 & & uni < = 127 ) ? uni : 0 ;
}
else
{
scan = domkeytoquake ( scan ) ;
uni = ( scan > = 32 & & scan < = 127 ) ? scan : 0 ;
}
IN_KeyEvent ( devid , down , scan , uni ) ;
2013-12-19 18:41:57 +00:00
//Chars which don't map to some printable ascii value get preventDefaulted.
//This is to stop fucking annoying fucking things like backspace randomly destroying the page and thus game.
//And it has to be conditional, or we don't get any unicode chars at all.
//The behaviour browsers seem to give is retardedly unhelpful, and just results in hacks to detect keys that appear to map to ascii...
//Preventing the browser from leaving the page etc should NOT mean I can no longer get ascii/unicode values, only that the browser stops trying to do something random due to the event.
//If you are the person that decreed that this is the holy way, then please castrate yourself now.
2014-08-25 07:35:41 +00:00
// if (scan == K_BACKSPACE || scan == K_LCTRL || scan == K_LALT || scan == K_LSHIFT || scan == K_RCTRL || scan == K_RALT || scan == K_RSHIFT)
2013-12-20 11:26:28 +00:00
return true ;
2014-08-25 07:35:41 +00:00
// return false;
2013-12-19 18:41:57 +00:00
}
static void DOM_ButtonEvent ( int devid , int down , int button )
{
if ( down = = 2 )
{
//fixme: the event is a float. we ignore that.
while ( button < 0 )
{
2014-09-02 06:01:03 +00:00
IN_KeyEvent ( devid , true , K_MWHEELUP , 0 ) ;
2013-12-19 18:41:57 +00:00
button + = 1 ;
}
while ( button > 0 )
{
2014-09-02 06:01:03 +00:00
IN_KeyEvent ( devid , true , K_MWHEELDOWN , 0 ) ;
2013-12-19 18:41:57 +00:00
button - = 1 ;
}
}
else
{
//swap buttons 2 and 3, so rmb is still +forward by default and not +mlook.
if ( button = = 2 )
button = 1 ;
else if ( button = = 1 )
button = 2 ;
2014-09-02 06:01:03 +00:00
IN_KeyEvent ( devid , down , K_MOUSE1 + button , 0 ) ;
2013-12-19 18:41:57 +00:00
}
}
2014-09-08 23:47:19 +00:00
vfsfile_t * FSWEB_OpenTempHandle ( int f ) ;
2015-04-14 23:12:17 +00:00
void DOM_LoadFile ( char * loc , char * mime , int handle )
2014-08-27 08:41:31 +00:00
{
2014-09-08 23:47:19 +00:00
vfsfile_t * file = NULL ;
Con_Printf ( " DOM_LoadFile: %s %i \n " , loc , handle ) ;
if ( handle ! = - 1 )
file = FSWEB_OpenTempHandle ( handle ) ;
2015-04-14 23:12:17 +00:00
else
{
char str [ 1024 ] ;
if ( ! strcmp ( mime , " joinurl " ) | | ! strcmp ( mime , " observeurl " ) | | ! strcmp ( mime , " connecturl " ) )
{
extern cvar_t spectator ;
if ( ! strcmp ( mime , " joinurl " ) )
Cvar_Set ( & spectator , " 0 " ) ;
if ( ! strcmp ( mime , " observeurl " ) )
Cvar_Set ( & spectator , " 1 " ) ;
Cbuf_AddText ( va ( " connect %s \n " , COM_QuotedString ( loc , str , sizeof ( str ) , false ) ) , RESTRICT_INSECURE ) ;
return ;
}
if ( ! strcmp ( mime , " demourl " ) )
{
Cbuf_AddText ( va ( " qtvplay %s \n " , COM_QuotedString ( loc , str , sizeof ( str ) , false ) ) , RESTRICT_INSECURE ) ;
return ;
}
}
2014-08-27 08:41:31 +00:00
//try and open it. generally downloading it from the server.
2014-09-08 23:47:19 +00:00
if ( ! Host_RunFile ( loc , strlen ( loc ) , file ) )
{
if ( file )
VFS_CLOSE ( file ) ;
}
2014-08-27 08:41:31 +00:00
}
2014-12-11 16:26:26 +00:00
int VID_ShouldSwitchToFullscreen ( void )
{ //if false, mouse grabs won't work and we'll be forced to touchscreen mode.
2015-08-20 03:17:47 +00:00
//we can only go fullscreen when the user clicks something.
//this means that the user will get pissed off at the fullscreen state changing when they first click on the menus after it loading up.
//this is confounded by escape bringing up the menu. <ESC>GRR IT CHANGED MODE!<options>WTF IT CHANGED AGAIN FUCKING PIECE OF SHIT!.
//annoying, but that's web browsers for you. the best thing we can do is to not regrab until they next click while actually back in the game.
2014-12-11 16:26:26 +00:00
extern cvar_t vid_fullscreen ;
2015-08-20 03:17:47 +00:00
return ! ! vid_fullscreen . value & & ( ! Key_Dest_Has ( kdm_console | kdm_cwindows | kdm_emenu ) | | ! Key_MouseShouldBeFree ( ) ) ;
2014-12-11 16:26:26 +00:00
}
2013-12-19 18:41:57 +00:00
qboolean GLVID_Init ( rendererstate_t * info , unsigned char * palette )
{
vid_isfullscreen = true ;
if ( ! emscriptenfte_setupcanvas (
info - > width ,
info - > height ,
VID_Resized ,
IN_MouseMove ,
DOM_ButtonEvent ,
2014-08-27 08:41:31 +00:00
DOM_KeyEvent ,
2014-10-11 19:39:45 +00:00
DOM_LoadFile ,
IN_JoystickButtonEvent ,
2014-12-11 16:26:26 +00:00
IN_JoystickAxisEvent ,
VID_ShouldSwitchToFullscreen
2013-12-19 18:41:57 +00:00
) )
{
Con_Printf ( " Couldn't set up canvas \n " ) ;
return false ;
}
2016-02-10 23:23:43 +00:00
vid . activeapp = true ;
2013-12-19 18:41:57 +00:00
GL_Init ( GLVID_getsdlglfunction ) ;
qglViewport ( 0 , 0 , vid . pixelwidth , vid . pixelheight ) ;
2014-08-27 08:41:31 +00:00
VID_Resized ( vid . pixelwidth , vid . pixelheight ) ;
2013-12-19 18:41:57 +00:00
mouseactive = false ;
return true ;
}
void GLVID_DeInit ( void )
{
2016-02-10 23:23:43 +00:00
vid . activeapp = false ;
2013-12-19 18:41:57 +00:00
2014-12-11 16:26:26 +00:00
emscriptenfte_setupcanvas ( - 1 , - 1 , NULL , NULL , NULL , NULL , NULL , NULL , NULL , NULL ) ;
2013-12-19 18:41:57 +00:00
}
2014-08-25 07:35:41 +00:00
void GLVID_SwapBuffers ( void )
2013-12-19 18:41:57 +00:00
{
//webgl doesn't support swapbuffers.
//you can't use it for loading screens.
//such things must result in waiting until the following frame.
//although there IS a swapped-buffers event, which we should probably use in preference to requestanimationframe or whatever the call is.
/*
if ( ! vid_isfullscreen )
{
if ( ! _windowed_mouse . value )
{
if ( mouseactive )
{
IN_DeactivateMouse ( ) ;
}
}
else
{
2016-02-10 23:23:43 +00:00
if ( ( key_dest = = key_game | | mouseusedforgui ) & & vid . activeapp )
2013-12-19 18:41:57 +00:00
IN_ActivateMouse ( ) ;
2016-02-10 23:23:43 +00:00
else if ( ! ( key_dest = = key_game | | mouseusedforgui ) | | ! vid . activeapp )
2013-12-19 18:41:57 +00:00
IN_DeactivateMouse ( ) ;
}
}
*/
}
qboolean GLVID_ApplyGammaRamps ( unsigned short * ramps )
{
gammaworks = false ;
return gammaworks ;
}
void GLVID_SetCaption ( char * text )
{
2014-10-11 19:39:45 +00:00
emscriptenfte_settitle ( text ) ;
2013-12-19 18:41:57 +00:00
}
void Sys_SendKeyEvents ( void )
{
2014-10-11 19:39:45 +00:00
/*most callbacks happen outside our code, we don't need to poll for events - except for joysticks*/
2015-08-20 03:17:47 +00:00
qboolean shouldbefree = Key_MouseShouldBeFree ( ) ;
emscriptenfte_updatepointerlock ( _windowed_mouse . ival & & ! shouldbefree , shouldbefree ) ;
2014-10-11 19:39:45 +00:00
emscriptenfte_polljoyevents ( ) ;
2013-12-19 18:41:57 +00:00
}
/*various stuff for joysticks, which we don't support in this port*/
void INS_Shutdown ( void )
{
}
void INS_ReInit ( void )
{
}
void INS_Move ( float * movements , int pnum )
{
}
void INS_Init ( void )
{
}
void INS_Accumulate ( void )
{
}
void INS_Commands ( void )
{
}
2016-02-15 06:01:17 +00:00
void INS_EnumerateDevices ( void * ctx , void ( * callback ) ( void * ctx , const char * type , const char * devicename , int * qdevid ) )
2015-04-14 23:12:17 +00:00
{
}
2013-12-19 18:41:57 +00:00