mirror of
https://github.com/nzp-team/fteqw.git
synced 2025-01-22 00:11:58 +00:00
2201b920c8
added frag message filter, and dedicated frag tracker. added 'windowed consoles' for social-type stuff without depending upon csqc mods for it. added in_deviceids command which allows listing/renumbering device ids. slider widgets now support inverted ranges, so gamma selection isn't so weird. fix top/bottom colour selection bug. software banding feature is now part of the 'software' preset (now that it supports mipmaps). support for loading .maps, and editing their brushes etc (with appropriate qc mod). 'map mymap.map' to use. expect problems with missing wads and replacement textures overriding them and messing up texture scales. snd_inactive is now default. fix threading issue with wavs, no more error from 0-sample-but-otherwise-valid wavs. added -makeinstaller option to embed a manifest inside the exe (and icon). the resulting program will insist on installing the game if its run from outside a valid basedir. framegroup support for q1mdl. textures are now loaded on multiple worker threads, for reduced load times. moo har har. netgraph shows packet+byte rates too. added r_lightstylescale, pretty similar to contrast, but doesn't impose any framerate cost, but may have overbrighting issues. r_softwarebanding now works on q2bsp too. fixed crepuscular lights. gzip transfer encoding is performed while downloading, instead of inducing stalls. FINALLY fix ezquake download compat issue (dimman found the issue). git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@4851 fc73d0e0-1445-4013-8a0c-d673dee63da5
307 lines
9.5 KiB
C
307 lines
9.5 KiB
C
#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 ActiveApp;
|
||
qboolean mouseactive;
|
||
extern qboolean mouseusedforgui;
|
||
|
||
|
||
static void *GLVID_getsdlglfunction(char *functionname)
|
||
{
|
||
return NULL;
|
||
}
|
||
|
||
static void IN_JoystickButtonEvent(int joydevid, int button, int ispressed)
|
||
{
|
||
if (button >= 32+4)
|
||
return;
|
||
IN_KeyEvent(joydevid, ispressed, K_JOY1+button, 0);
|
||
}
|
||
|
||
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)
|
||
{
|
||
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,
|
||
/* 48*/ '0','1','2','3','4','5','6','7', '8','9',0,';',0,'=',0,0,
|
||
|
||
/* 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,
|
||
/*160*/ 0,0,0,'#',0,0,0,0, 0,0,0,0,0,'-',0,0,
|
||
/*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]);
|
||
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,
|
||
/* 48*/ ')','!','\"',0/*<2A>*/,'$','%','^','&', '*','(',0,':',0,'+',0,0,
|
||
|
||
/* 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,
|
||
/*160*/ 0,0,0,'~',0,0,0,0, 0,0,0,0,0,'_',0,0,
|
||
/*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]);
|
||
return tab[code];
|
||
}
|
||
static int DOM_KeyEvent(int devid, int down, int scan, int uni)
|
||
{
|
||
extern int shift_down;
|
||
// Con_Printf("Key %i %i:%c\n", scan, uni, (char)uni);
|
||
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);
|
||
//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.
|
||
// if (scan == K_BACKSPACE || scan == K_LCTRL || scan == K_LALT || scan == K_LSHIFT || scan == K_RCTRL || scan == K_RALT || scan == K_RSHIFT)
|
||
return true;
|
||
// return false;
|
||
}
|
||
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)
|
||
{
|
||
IN_KeyEvent(devid, true, K_MWHEELUP, 0);
|
||
button += 1;
|
||
}
|
||
while(button > 0)
|
||
{
|
||
IN_KeyEvent(devid, true, K_MWHEELDOWN, 0);
|
||
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;
|
||
|
||
IN_KeyEvent(devid, down, K_MOUSE1+button, 0);
|
||
}
|
||
}
|
||
vfsfile_t *FSWEB_OpenTempHandle(int f);
|
||
void DOM_LoadFile(char *loc, char *mime, int handle)
|
||
{
|
||
vfsfile_t *file = NULL;
|
||
Con_Printf("DOM_LoadFile: %s %i\n", loc, handle);
|
||
if (handle != -1)
|
||
file = FSWEB_OpenTempHandle(handle);
|
||
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;
|
||
}
|
||
}
|
||
//try and open it. generally downloading it from the server.
|
||
if (!Host_RunFile(loc, strlen(loc), file))
|
||
{
|
||
if (file)
|
||
VFS_CLOSE(file);
|
||
}
|
||
}
|
||
int VID_ShouldSwitchToFullscreen(void)
|
||
{ //if false, mouse grabs won't work and we'll be forced to touchscreen mode.
|
||
extern cvar_t vid_fullscreen;
|
||
return !!vid_fullscreen.value;
|
||
}
|
||
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,
|
||
DOM_KeyEvent,
|
||
DOM_LoadFile,
|
||
IN_JoystickButtonEvent,
|
||
IN_JoystickAxisEvent,
|
||
VID_ShouldSwitchToFullscreen
|
||
))
|
||
{
|
||
Con_Printf("Couldn't set up canvas\n");
|
||
return false;
|
||
}
|
||
|
||
ActiveApp = true;
|
||
|
||
GL_Init(GLVID_getsdlglfunction);
|
||
|
||
qglViewport (0, 0, vid.pixelwidth, vid.pixelheight);
|
||
|
||
VID_Resized(vid.pixelwidth, vid.pixelheight);
|
||
|
||
mouseactive = false;
|
||
|
||
return true;
|
||
}
|
||
|
||
void GLVID_DeInit (void)
|
||
{
|
||
ActiveApp = false;
|
||
|
||
emscriptenfte_setupcanvas(-1, -1, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
|
||
}
|
||
|
||
|
||
void GLVID_SwapBuffers (void)
|
||
{
|
||
//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
|
||
{
|
||
if ((key_dest == key_game||mouseusedforgui) && ActiveApp)
|
||
IN_ActivateMouse ();
|
||
else if (!(key_dest == key_game || mouseusedforgui) || !ActiveApp)
|
||
IN_DeactivateMouse ();
|
||
}
|
||
}
|
||
*/
|
||
}
|
||
|
||
qboolean GLVID_ApplyGammaRamps (unsigned short *ramps)
|
||
{
|
||
gammaworks = false;
|
||
return gammaworks;
|
||
}
|
||
|
||
void GLVID_SetCaption(char *text)
|
||
{
|
||
emscriptenfte_settitle(text);
|
||
}
|
||
|
||
void Sys_SendKeyEvents(void)
|
||
{
|
||
/*most callbacks happen outside our code, we don't need to poll for events - except for joysticks*/
|
||
emscriptenfte_polljoyevents();
|
||
}
|
||
/*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)
|
||
{
|
||
}
|
||
void INS_EnumerateDevices(void *ctx, void(*callback)(void *ctx, char *type, char *devicename, int *qdevid))
|
||
{
|
||
}
|
||
|