2014-03-15 16:59:03 +00:00
|
|
|
// SONIC ROBO BLAST 2
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// Copyright (C) 1998-2000 by DooM Legacy Team.
|
2023-03-31 12:53:31 +00:00
|
|
|
// Copyright (C) 1999-2023 by Sonic Team Junior.
|
2014-03-15 16:59:03 +00:00
|
|
|
//
|
|
|
|
// This program is free software distributed under the
|
|
|
|
// terms of the GNU General Public License, version 2.
|
|
|
|
// See the 'LICENSE' file for more details.
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/// \file screen.c
|
|
|
|
/// \brief Handles multiple resolutions, 8bpp/16bpp(highcolor) modes
|
|
|
|
|
|
|
|
#include "doomdef.h"
|
|
|
|
#include "doomstat.h"
|
|
|
|
#include "screen.h"
|
|
|
|
#include "console.h"
|
|
|
|
#include "am_map.h"
|
2022-04-30 21:33:23 +00:00
|
|
|
#include "i_time.h"
|
2014-03-15 16:59:03 +00:00
|
|
|
#include "i_system.h"
|
|
|
|
#include "i_video.h"
|
|
|
|
#include "r_local.h"
|
|
|
|
#include "r_sky.h"
|
|
|
|
#include "m_argv.h"
|
|
|
|
#include "m_misc.h"
|
|
|
|
#include "v_video.h"
|
|
|
|
#include "st_stuff.h"
|
|
|
|
#include "hu_stuff.h"
|
|
|
|
#include "z_zone.h"
|
|
|
|
#include "d_main.h"
|
|
|
|
#include "d_clisrv.h"
|
|
|
|
#include "f_finale.h"
|
2020-08-15 01:27:16 +00:00
|
|
|
#include "y_inter.h" // usebuffer
|
2017-04-17 00:09:47 +00:00
|
|
|
#include "i_sound.h" // closed captions
|
|
|
|
#include "s_sound.h" // ditto
|
|
|
|
#include "g_game.h" // ditto
|
2017-09-25 19:35:04 +00:00
|
|
|
#include "p_local.h" // P_AutoPause()
|
2022-03-27 03:48:08 +00:00
|
|
|
|
2021-03-28 17:22:04 +00:00
|
|
|
#ifdef HWRENDER
|
|
|
|
#include "hardware/hw_main.h"
|
|
|
|
#include "hardware/hw_light.h"
|
|
|
|
#include "hardware/hw_model.h"
|
|
|
|
#endif
|
2014-03-15 16:59:03 +00:00
|
|
|
|
2022-03-27 03:48:08 +00:00
|
|
|
// SRB2Kart
|
|
|
|
#include "r_fps.h" // R_GetFramerateCap
|
2014-03-15 16:59:03 +00:00
|
|
|
|
|
|
|
// --------------------------------------------
|
|
|
|
// assembly or c drawer routines for 8bpp/16bpp
|
|
|
|
// --------------------------------------------
|
2019-12-13 15:26:47 +00:00
|
|
|
void (*colfunc)(void);
|
|
|
|
void (*colfuncs[COLDRAWFUNC_MAX])(void);
|
|
|
|
|
|
|
|
void (*spanfunc)(void);
|
|
|
|
void (*spanfuncs[SPANDRAWFUNC_MAX])(void);
|
2019-12-13 16:11:56 +00:00
|
|
|
void (*spanfuncs_npo2[SPANDRAWFUNC_MAX])(void);
|
2014-03-15 16:59:03 +00:00
|
|
|
|
|
|
|
// ------------------
|
|
|
|
// global video state
|
|
|
|
// ------------------
|
|
|
|
viddef_t vid;
|
|
|
|
INT32 setmodeneeded; //video mode change needed if > 0 (the mode number to set + 1)
|
2019-09-09 22:34:51 +00:00
|
|
|
UINT8 setrenderneeded = 0;
|
2014-03-15 16:59:03 +00:00
|
|
|
|
|
|
|
static CV_PossibleValue_t scr_depth_cons_t[] = {{8, "8 bits"}, {16, "16 bits"}, {24, "24 bits"}, {32, "32 bits"}, {0, NULL}};
|
|
|
|
|
|
|
|
//added : 03-02-98: default screen mode, as loaded/saved in config
|
2020-10-07 06:04:23 +00:00
|
|
|
consvar_t cv_scr_width = CVAR_INIT ("scr_width", "1280", CV_SAVE, CV_Unsigned, NULL);
|
|
|
|
consvar_t cv_scr_height = CVAR_INIT ("scr_height", "800", CV_SAVE, CV_Unsigned, NULL);
|
2022-01-31 13:40:41 +00:00
|
|
|
consvar_t cv_scr_width_w = CVAR_INIT ("scr_width_w", "640", CV_SAVE, CV_Unsigned, NULL);
|
|
|
|
consvar_t cv_scr_height_w = CVAR_INIT ("scr_height_w", "400", CV_SAVE, CV_Unsigned, NULL);
|
2020-10-07 06:04:23 +00:00
|
|
|
consvar_t cv_scr_depth = CVAR_INIT ("scr_depth", "16 bits", CV_SAVE, scr_depth_cons_t, NULL);
|
2022-03-27 03:48:08 +00:00
|
|
|
|
2020-10-07 06:04:23 +00:00
|
|
|
consvar_t cv_renderview = CVAR_INIT ("renderview", "On", 0, CV_OnOff, NULL);
|
2020-12-15 18:41:03 +00:00
|
|
|
consvar_t cv_renderwalls = CVAR_INIT ("renderwalls", "On", 0, CV_OnOff, NULL);
|
2020-12-15 18:52:39 +00:00
|
|
|
consvar_t cv_renderfloors = CVAR_INIT ("renderfloors", "On", 0, CV_OnOff, NULL);
|
2020-12-15 19:03:00 +00:00
|
|
|
consvar_t cv_renderthings = CVAR_INIT ("renderthings", "On", 0, CV_OnOff, NULL);
|
2016-06-12 19:58:03 +00:00
|
|
|
|
2019-12-31 04:29:41 +00:00
|
|
|
CV_PossibleValue_t cv_renderer_t[] = {
|
|
|
|
{1, "Software"},
|
|
|
|
#ifdef HWRENDER
|
|
|
|
{2, "OpenGL"},
|
|
|
|
#endif
|
|
|
|
{0, NULL}
|
|
|
|
};
|
2020-10-10 20:01:10 +00:00
|
|
|
|
2023-03-15 18:43:53 +00:00
|
|
|
consvar_t cv_renderer = CVAR_INIT ("renderer", "Software", CV_SAVE|CV_CALL, cv_renderer_t, SCR_ChangeRenderer);
|
2019-09-08 21:27:35 +00:00
|
|
|
|
2019-09-09 00:37:24 +00:00
|
|
|
static void SCR_ChangeFullscreen(void);
|
2016-06-12 19:58:03 +00:00
|
|
|
|
2020-10-07 06:04:23 +00:00
|
|
|
consvar_t cv_fullscreen = CVAR_INIT ("fullscreen", "Yes", CV_SAVE|CV_CALL, CV_YesNo, SCR_ChangeFullscreen);
|
2014-03-15 16:59:03 +00:00
|
|
|
|
|
|
|
// =========================================================================
|
|
|
|
// SCREEN VARIABLES
|
|
|
|
// =========================================================================
|
|
|
|
|
|
|
|
INT32 scr_bpp; // current video mode bytes per pixel
|
|
|
|
UINT8 *scr_borderpatch; // flat used to fill the reduced view borders set at ST_Init()
|
|
|
|
|
|
|
|
// =========================================================================
|
|
|
|
|
|
|
|
// Short and Tall sky drawer, for the current color mode
|
|
|
|
void (*walldrawerfunc)(void);
|
|
|
|
|
|
|
|
boolean R_486 = false;
|
|
|
|
boolean R_586 = false;
|
|
|
|
boolean R_MMX = false;
|
|
|
|
boolean R_SSE = false;
|
|
|
|
boolean R_3DNow = false;
|
|
|
|
boolean R_MMXExt = false;
|
|
|
|
boolean R_SSE2 = false;
|
|
|
|
|
2019-09-08 21:27:35 +00:00
|
|
|
void SCR_SetDrawFuncs(void)
|
2014-03-15 16:59:03 +00:00
|
|
|
{
|
|
|
|
//
|
|
|
|
// setup the right draw routines for either 8bpp or 16bpp
|
|
|
|
//
|
|
|
|
if (true)//vid.bpp == 1) //Always run in 8bpp. todo: remove all 16bpp code?
|
|
|
|
{
|
2019-12-13 15:26:47 +00:00
|
|
|
colfuncs[BASEDRAWFUNC] = R_DrawColumn_8;
|
|
|
|
spanfuncs[BASEDRAWFUNC] = R_DrawSpan_8;
|
|
|
|
|
|
|
|
colfunc = colfuncs[BASEDRAWFUNC];
|
|
|
|
spanfunc = spanfuncs[BASEDRAWFUNC];
|
|
|
|
|
|
|
|
colfuncs[COLDRAWFUNC_FUZZY] = R_DrawTranslucentColumn_8;
|
|
|
|
colfuncs[COLDRAWFUNC_TRANS] = R_DrawTranslatedColumn_8;
|
|
|
|
colfuncs[COLDRAWFUNC_SHADE] = R_DrawShadeColumn_8;
|
|
|
|
colfuncs[COLDRAWFUNC_SHADOWED] = R_DrawColumnShadowed_8;
|
|
|
|
colfuncs[COLDRAWFUNC_TRANSTRANS] = R_DrawTranslatedTranslucentColumn_8;
|
|
|
|
colfuncs[COLDRAWFUNC_TWOSMULTIPATCH] = R_Draw2sMultiPatchColumn_8;
|
|
|
|
colfuncs[COLDRAWFUNC_TWOSMULTIPATCHTRANS] = R_Draw2sMultiPatchTranslucentColumn_8;
|
|
|
|
colfuncs[COLDRAWFUNC_FOG] = R_DrawFogColumn_8;
|
|
|
|
|
|
|
|
spanfuncs[SPANDRAWFUNC_TRANS] = R_DrawTranslucentSpan_8;
|
2020-11-05 03:00:08 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_TILTED] = R_DrawTiltedSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDTRANS] = R_DrawTiltedTranslucentSpan_8;
|
2019-12-13 15:26:47 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_SPLAT] = R_DrawSplat_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TRANSSPLAT] = R_DrawTranslucentSplat_8;
|
2020-11-05 03:00:08 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDSPLAT] = R_DrawTiltedSplat_8;
|
2020-10-12 03:13:22 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_SPRITE] = R_DrawFloorSprite_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TRANSSPRITE] = R_DrawTranslucentFloorSprite_8;
|
2020-10-14 01:42:07 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDSPRITE] = R_DrawTiltedFloorSprite_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDTRANSSPRITE] = R_DrawTiltedTranslucentFloorSprite_8;
|
2022-11-16 01:40:54 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_WATER] = R_DrawWaterSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDWATER] = R_DrawTiltedWaterSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_SOLID] = R_DrawSolidColorSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TRANSSOLID] = R_DrawTransSolidColorSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDSOLID] = R_DrawTiltedSolidColorSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDTRANSSOLID] = R_DrawTiltedTransSolidColorSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_WATERSOLID] = R_DrawWaterSolidColorSpan_8;
|
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDWATERSOLID] = R_DrawTiltedWaterSolidColorSpan_8;
|
2020-11-05 03:00:08 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_FOG] = R_DrawFogSpan_8;
|
2021-10-12 17:10:48 +00:00
|
|
|
spanfuncs[SPANDRAWFUNC_TILTEDFOG] = R_DrawTiltedFogSpan_8;
|
2019-12-13 15:26:47 +00:00
|
|
|
|
2019-12-13 16:11:56 +00:00
|
|
|
// Lactozilla: Non-powers-of-two
|
|
|
|
spanfuncs_npo2[BASEDRAWFUNC] = R_DrawSpan_NPO2_8;
|
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TRANS] = R_DrawTranslucentSpan_NPO2_8;
|
2020-11-05 03:00:08 +00:00
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TILTED] = R_DrawTiltedSpan_NPO2_8;
|
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TILTEDTRANS] = R_DrawTiltedTranslucentSpan_NPO2_8;
|
2019-12-13 16:11:56 +00:00
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_SPLAT] = R_DrawSplat_NPO2_8;
|
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TRANSSPLAT] = R_DrawTranslucentSplat_NPO2_8;
|
2020-11-05 03:00:08 +00:00
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TILTEDSPLAT] = R_DrawTiltedSplat_NPO2_8;
|
2020-10-12 03:13:22 +00:00
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_SPRITE] = R_DrawFloorSprite_NPO2_8;
|
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TRANSSPRITE] = R_DrawTranslucentFloorSprite_NPO2_8;
|
2020-10-14 01:42:07 +00:00
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TILTEDSPRITE] = R_DrawTiltedFloorSprite_NPO2_8;
|
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TILTEDTRANSSPRITE] = R_DrawTiltedTranslucentFloorSprite_NPO2_8;
|
2022-11-16 01:40:54 +00:00
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_WATER] = R_DrawWaterSpan_NPO2_8;
|
|
|
|
spanfuncs_npo2[SPANDRAWFUNC_TILTEDWATER] = R_DrawTiltedWaterSpan_NPO2_8;
|
2019-12-13 15:26:47 +00:00
|
|
|
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
|
|
|
/* else if (vid.bpp > 1)
|
|
|
|
{
|
|
|
|
I_OutputMsg("using highcolor mode\n");
|
|
|
|
spanfunc = basespanfunc = R_DrawSpan_16;
|
|
|
|
transcolfunc = R_DrawTranslatedColumn_16;
|
|
|
|
transtransfunc = R_DrawTranslucentColumn_16; // No 16bit operation for this function
|
|
|
|
|
|
|
|
colfunc = basecolfunc = R_DrawColumn_16;
|
|
|
|
shadecolfunc = NULL; // detect error if used somewhere..
|
|
|
|
fuzzcolfunc = R_DrawTranslucentColumn_16;
|
|
|
|
walldrawerfunc = R_DrawWallColumn_16;
|
|
|
|
}*/
|
|
|
|
else
|
|
|
|
I_Error("unknown bytes per pixel mode %d\n", vid.bpp);
|
2017-09-29 20:03:28 +00:00
|
|
|
/*
|
2014-03-15 16:59:03 +00:00
|
|
|
if (SCR_IsAspectCorrect(vid.width, vid.height))
|
|
|
|
CONS_Alert(CONS_WARNING, M_GetText("Resolution is not aspect-correct!\nUse a multiple of %dx%d\n"), BASEVIDWIDTH, BASEVIDHEIGHT);
|
2017-09-29 20:03:28 +00:00
|
|
|
*/
|
2019-09-08 21:27:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SCR_SetMode(void)
|
|
|
|
{
|
|
|
|
if (dedicated)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!(setmodeneeded || setrenderneeded) || WipeInAction)
|
|
|
|
return; // should never happen and don't change it during a wipe, BAD!
|
|
|
|
|
2019-12-17 19:14:26 +00:00
|
|
|
// Lactozilla: Renderer switching
|
2019-09-08 21:27:35 +00:00
|
|
|
if (setrenderneeded)
|
|
|
|
{
|
2020-08-15 01:27:16 +00:00
|
|
|
// stop recording movies (APNG only)
|
|
|
|
if (setrenderneeded && (moviemode == MM_APNG))
|
|
|
|
M_StopMovie();
|
|
|
|
|
2020-11-22 20:22:18 +00:00
|
|
|
// VID_SetMode will call VID_CheckRenderer itself,
|
|
|
|
// so no need to do this in here.
|
|
|
|
if (!setmodeneeded)
|
|
|
|
VID_CheckRenderer();
|
|
|
|
|
2020-08-15 01:27:16 +00:00
|
|
|
vid.recalc = 1;
|
2019-09-08 21:27:35 +00:00
|
|
|
}
|
|
|
|
|
2020-08-15 01:27:16 +00:00
|
|
|
// Set the video mode in the video interface.
|
2019-09-08 21:27:35 +00:00
|
|
|
if (setmodeneeded)
|
2020-12-20 22:03:20 +00:00
|
|
|
VID_SetMode(setmodeneeded - 1);
|
2019-09-08 21:27:35 +00:00
|
|
|
|
|
|
|
V_SetPalette(0);
|
|
|
|
|
|
|
|
SCR_SetDrawFuncs();
|
2017-04-22 23:40:18 +00:00
|
|
|
|
2014-03-15 16:59:03 +00:00
|
|
|
// set the apprpriate drawer for the sky (tall or INT16)
|
|
|
|
setmodeneeded = 0;
|
2019-09-08 21:27:35 +00:00
|
|
|
setrenderneeded = 0;
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// do some initial settings for the game loading screen
|
|
|
|
//
|
|
|
|
void SCR_Startup(void)
|
|
|
|
{
|
|
|
|
const CPUInfoFlags *RCpuInfo = I_CPUInfo();
|
|
|
|
if (!M_CheckParm("-NOCPUID") && RCpuInfo)
|
|
|
|
{
|
|
|
|
#if defined (__i386__) || defined (_M_IX86) || defined (__WATCOMC__)
|
|
|
|
R_486 = true;
|
|
|
|
#endif
|
|
|
|
if (RCpuInfo->RDTSC)
|
|
|
|
R_586 = true;
|
|
|
|
if (RCpuInfo->MMX)
|
|
|
|
R_MMX = true;
|
|
|
|
if (RCpuInfo->AMD3DNow)
|
|
|
|
R_3DNow = true;
|
|
|
|
if (RCpuInfo->MMXExt)
|
|
|
|
R_MMXExt = true;
|
|
|
|
if (RCpuInfo->SSE)
|
|
|
|
R_SSE = true;
|
|
|
|
if (RCpuInfo->SSE2)
|
|
|
|
R_SSE2 = true;
|
|
|
|
CONS_Printf("CPU Info: 486: %i, 586: %i, MMX: %i, 3DNow: %i, MMXExt: %i, SSE2: %i\n", R_486, R_586, R_MMX, R_3DNow, R_MMXExt, R_SSE2);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (M_CheckParm("-486"))
|
|
|
|
R_486 = true;
|
|
|
|
if (M_CheckParm("-586"))
|
|
|
|
R_586 = true;
|
|
|
|
if (M_CheckParm("-MMX"))
|
|
|
|
R_MMX = true;
|
|
|
|
if (M_CheckParm("-3DNow"))
|
|
|
|
R_3DNow = true;
|
|
|
|
if (M_CheckParm("-MMXExt"))
|
|
|
|
R_MMXExt = true;
|
|
|
|
|
|
|
|
if (M_CheckParm("-SSE"))
|
|
|
|
R_SSE = true;
|
|
|
|
if (M_CheckParm("-noSSE"))
|
|
|
|
R_SSE = false;
|
|
|
|
|
|
|
|
if (M_CheckParm("-SSE2"))
|
|
|
|
R_SSE2 = true;
|
|
|
|
|
|
|
|
M_SetupMemcpy();
|
|
|
|
|
|
|
|
if (dedicated)
|
|
|
|
{
|
|
|
|
V_Init();
|
|
|
|
V_SetPalette(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
vid.modenum = 0;
|
|
|
|
|
|
|
|
V_Init();
|
2020-08-15 01:27:16 +00:00
|
|
|
V_Recalc();
|
|
|
|
|
2014-03-15 16:59:03 +00:00
|
|
|
CV_RegisterVar(&cv_ticrate);
|
|
|
|
CV_RegisterVar(&cv_constextsize);
|
|
|
|
|
|
|
|
V_SetPalette(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Called at new frame, if the video mode has changed
|
|
|
|
//
|
|
|
|
void SCR_Recalc(void)
|
|
|
|
{
|
|
|
|
if (dedicated)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// bytes per pixel quick access
|
|
|
|
scr_bpp = vid.bpp;
|
|
|
|
|
2020-08-15 01:27:16 +00:00
|
|
|
V_Recalc();
|
2014-03-15 16:59:03 +00:00
|
|
|
|
2020-01-27 02:12:28 +00:00
|
|
|
// toggle off (then back on) the automap because some screensize-dependent values will
|
2014-03-15 16:59:03 +00:00
|
|
|
// be calculated next time the automap is activated.
|
|
|
|
if (automapactive)
|
2020-01-27 02:12:28 +00:00
|
|
|
{
|
|
|
|
am_recalc = true;
|
|
|
|
AM_Start();
|
|
|
|
}
|
2014-03-15 16:59:03 +00:00
|
|
|
|
|
|
|
// set the screen[x] ptrs on the new vidbuffers
|
|
|
|
V_Init();
|
|
|
|
|
|
|
|
// scr_viewsize doesn't change, neither detailLevel, but the pixels
|
|
|
|
// per screenblock is different now, since we've changed resolution.
|
|
|
|
R_SetViewSize(); //just set setsizeneeded true now ..
|
|
|
|
|
|
|
|
// vid.recalc lasts only for the next refresh...
|
|
|
|
con_recalc = true;
|
|
|
|
am_recalc = true;
|
2020-08-15 01:27:16 +00:00
|
|
|
|
|
|
|
#ifdef HWRENDER
|
|
|
|
// Shoot! The screen texture was flushed!
|
|
|
|
if ((rendermode == render_opengl) && (gamestate == GS_INTERMISSION))
|
|
|
|
usebuffer = false;
|
|
|
|
#endif
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check for screen cmd-line parms: to force a resolution.
|
|
|
|
//
|
|
|
|
// Set the video mode to set at the 1st display loop (setmodeneeded)
|
|
|
|
//
|
|
|
|
|
|
|
|
void SCR_CheckDefaultMode(void)
|
|
|
|
{
|
|
|
|
INT32 scr_forcex, scr_forcey; // resolution asked from the cmd-line
|
|
|
|
|
|
|
|
if (dedicated)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// 0 means not set at the cmd-line
|
|
|
|
scr_forcex = scr_forcey = 0;
|
|
|
|
|
|
|
|
if (M_CheckParm("-width") && M_IsNextParm())
|
|
|
|
scr_forcex = atoi(M_GetNextParm());
|
|
|
|
|
|
|
|
if (M_CheckParm("-height") && M_IsNextParm())
|
|
|
|
scr_forcey = atoi(M_GetNextParm());
|
|
|
|
|
|
|
|
if (scr_forcex && scr_forcey)
|
|
|
|
{
|
|
|
|
CONS_Printf(M_GetText("Using resolution: %d x %d\n"), scr_forcex, scr_forcey);
|
|
|
|
// returns -1 if not found, thus will be 0 (no mode change) if not found
|
|
|
|
setmodeneeded = VID_GetModeForSize(scr_forcex, scr_forcey) + 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-02-06 15:13:45 +00:00
|
|
|
CONS_Printf(M_GetText("Default resolution: %d x %d\n"), cv_scr_width.value, cv_scr_height.value);
|
|
|
|
CONS_Printf(M_GetText("Windowed resolution: %d x %d\n"), cv_scr_width_w.value, cv_scr_height_w.value);
|
|
|
|
CONS_Printf(M_GetText("Default bit depth: %d bits\n"), cv_scr_depth.value);
|
|
|
|
if (cv_fullscreen.value)
|
2022-02-04 13:42:10 +00:00
|
|
|
setmodeneeded = VID_GetModeForSize(cv_scr_width.value, cv_scr_height.value) + 1; // see note above
|
2022-02-06 15:13:45 +00:00
|
|
|
else
|
|
|
|
setmodeneeded = VID_GetModeForSize(cv_scr_width_w.value, cv_scr_height_w.value) + 1; // see note above
|
2022-02-19 11:12:04 +00:00
|
|
|
|
|
|
|
if (setmodeneeded <= 0)
|
|
|
|
CONS_Alert(CONS_WARNING, "Invalid resolution given, defaulting to base resolution\n");
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
2019-09-09 00:37:24 +00:00
|
|
|
|
2020-08-15 01:27:16 +00:00
|
|
|
if (cv_renderer.value != (signed)rendermode)
|
|
|
|
{
|
|
|
|
if (chosenrendermode == render_none) // nothing set at command line
|
|
|
|
SCR_ChangeRenderer();
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Set cv_renderer to the current render mode
|
|
|
|
CV_StealthSetValue(&cv_renderer, rendermode);
|
|
|
|
}
|
|
|
|
}
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// sets the modenum as the new default video mode to be saved in the config file
|
|
|
|
void SCR_SetDefaultMode(void)
|
|
|
|
{
|
2022-02-06 15:13:45 +00:00
|
|
|
CV_SetValue(cv_fullscreen.value ? &cv_scr_width : &cv_scr_width_w, vid.width);
|
|
|
|
CV_SetValue(cv_fullscreen.value ? &cv_scr_height : &cv_scr_height_w, vid.height);
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Change fullscreen on/off according to cv_fullscreen
|
|
|
|
void SCR_ChangeFullscreen(void)
|
|
|
|
{
|
|
|
|
#ifdef DIRECTFULLSCREEN
|
|
|
|
// allow_fullscreen is set by VID_PrepareModeList
|
|
|
|
// it is used to prevent switching to fullscreen during startup
|
|
|
|
if (!allow_fullscreen)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (graphics_started)
|
|
|
|
{
|
|
|
|
VID_PrepareModeList();
|
2022-02-06 15:13:45 +00:00
|
|
|
if (cv_fullscreen.value)
|
2022-01-31 13:40:41 +00:00
|
|
|
setmodeneeded = VID_GetModeForSize(cv_scr_width.value, cv_scr_height.value) + 1;
|
2022-02-06 15:13:45 +00:00
|
|
|
else
|
|
|
|
setmodeneeded = VID_GetModeForSize(cv_scr_width_w.value, cv_scr_height_w.value) + 1;
|
2022-02-19 11:12:04 +00:00
|
|
|
|
|
|
|
if (setmodeneeded <= 0) // hacky safeguard
|
|
|
|
{
|
|
|
|
CONS_Alert(CONS_WARNING, "Invalid resolution given, defaulting to base resolution.\n");
|
|
|
|
setmodeneeded = VID_GetModeForSize(BASEVIDWIDTH, BASEVIDHEIGHT) + 1;
|
|
|
|
}
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-08-15 01:27:16 +00:00
|
|
|
void SCR_ChangeRenderer(void)
|
2019-09-09 00:37:24 +00:00
|
|
|
{
|
2020-11-22 20:22:18 +00:00
|
|
|
if (chosenrendermode != render_none
|
|
|
|
|| (signed)rendermode == cv_renderer.value)
|
2020-08-15 01:27:16 +00:00
|
|
|
return;
|
2020-02-17 15:14:51 +00:00
|
|
|
|
|
|
|
#ifdef HWRENDER
|
2020-08-15 01:27:16 +00:00
|
|
|
// Check if OpenGL loaded successfully (or wasn't disabled) before switching to it.
|
|
|
|
if ((vid.glstate == VID_GL_LIBRARY_ERROR)
|
|
|
|
&& (cv_renderer.value == render_opengl))
|
2020-02-17 15:14:51 +00:00
|
|
|
{
|
|
|
|
if (M_CheckParm("-nogl"))
|
|
|
|
CONS_Alert(CONS_ERROR, "OpenGL rendering was disabled!\n");
|
|
|
|
else
|
|
|
|
CONS_Alert(CONS_ERROR, "OpenGL never loaded\n");
|
|
|
|
return;
|
|
|
|
}
|
2021-03-28 17:22:04 +00:00
|
|
|
|
|
|
|
if (rendermode == render_opengl && (vid.glstate == VID_GL_LIBRARY_LOADED)) // Clear these out before switching to software
|
|
|
|
HWR_ClearAllTextures();
|
|
|
|
|
2020-02-17 15:14:51 +00:00
|
|
|
#endif
|
|
|
|
|
2020-08-15 01:27:16 +00:00
|
|
|
// Set the new render mode
|
|
|
|
setrenderneeded = cv_renderer.value;
|
2019-09-08 21:27:35 +00:00
|
|
|
}
|
|
|
|
|
2014-03-15 16:59:03 +00:00
|
|
|
boolean SCR_IsAspectCorrect(INT32 width, INT32 height)
|
|
|
|
{
|
|
|
|
return
|
|
|
|
( width % BASEVIDWIDTH == 0
|
|
|
|
&& height % BASEVIDHEIGHT == 0
|
|
|
|
&& width / BASEVIDWIDTH == height / BASEVIDHEIGHT
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-03-27 03:48:08 +00:00
|
|
|
double averageFPS = 0.0f;
|
2014-03-15 16:59:03 +00:00
|
|
|
|
2022-05-19 20:37:33 +00:00
|
|
|
#define USE_FPS_SAMPLES
|
|
|
|
|
|
|
|
#ifdef USE_FPS_SAMPLES
|
2023-07-15 10:51:07 +00:00
|
|
|
#define MAX_FRAME_TIME 0.05
|
2022-05-19 20:37:33 +00:00
|
|
|
#define NUM_FPS_SAMPLES (16) // Number of samples to store
|
2022-03-26 20:32:43 +00:00
|
|
|
|
2023-07-15 10:51:07 +00:00
|
|
|
static double total_frame_time = 0.0;
|
|
|
|
static int frame_index;
|
2022-05-19 20:37:33 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static boolean fps_init = false;
|
|
|
|
static precise_t fps_enter = 0;
|
2022-03-26 20:32:43 +00:00
|
|
|
|
2022-03-27 03:48:08 +00:00
|
|
|
void SCR_CalculateFPS(void)
|
2014-03-15 16:59:03 +00:00
|
|
|
{
|
2022-05-19 20:37:33 +00:00
|
|
|
precise_t fps_finish = 0;
|
2014-03-15 16:59:03 +00:00
|
|
|
|
2022-05-19 20:37:33 +00:00
|
|
|
double frameElapsed = 0.0;
|
2022-03-26 20:32:43 +00:00
|
|
|
|
2022-05-19 20:37:33 +00:00
|
|
|
if (fps_init == false)
|
2022-03-27 03:48:08 +00:00
|
|
|
{
|
2022-05-19 20:37:33 +00:00
|
|
|
fps_enter = I_GetPreciseTime();
|
|
|
|
fps_init = true;
|
2022-03-27 03:48:08 +00:00
|
|
|
}
|
2022-03-26 20:32:43 +00:00
|
|
|
|
2022-05-19 20:37:33 +00:00
|
|
|
fps_finish = I_GetPreciseTime();
|
|
|
|
frameElapsed = (double)((INT64)(fps_finish - fps_enter)) / I_GetPrecisePrecision();
|
|
|
|
fps_enter = fps_finish;
|
|
|
|
|
|
|
|
#ifdef USE_FPS_SAMPLES
|
2023-07-15 10:51:07 +00:00
|
|
|
total_frame_time += frameElapsed;
|
|
|
|
if (frame_index++ >= NUM_FPS_SAMPLES || total_frame_time >= MAX_FRAME_TIME)
|
2022-05-19 20:37:33 +00:00
|
|
|
{
|
2023-07-15 10:51:07 +00:00
|
|
|
averageFPS = 1.0 / (total_frame_time / frame_index);
|
|
|
|
total_frame_time = 0.0;
|
|
|
|
frame_index = 0;
|
2022-05-19 20:37:33 +00:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
// Direct, unsampled counter.
|
|
|
|
averageFPS = 1.0 / frameElapsed;
|
|
|
|
#endif
|
2022-03-26 20:32:43 +00:00
|
|
|
}
|
2014-03-15 16:59:03 +00:00
|
|
|
|
|
|
|
void SCR_DisplayTicRate(void)
|
|
|
|
{
|
|
|
|
INT32 ticcntcolor = 0;
|
2018-04-06 17:40:08 +00:00
|
|
|
const INT32 h = vid.height-(8*vid.dupy);
|
2022-03-27 03:48:08 +00:00
|
|
|
UINT32 cap = R_GetFramerateCap();
|
2022-04-19 16:06:17 +00:00
|
|
|
double fps = round(averageFPS);
|
2014-03-15 16:59:03 +00:00
|
|
|
|
2020-05-10 23:33:34 +00:00
|
|
|
if (gamestate == GS_NULL)
|
|
|
|
return;
|
|
|
|
|
2022-04-16 05:21:49 +00:00
|
|
|
if (cap > 0)
|
|
|
|
{
|
|
|
|
if (fps <= cap / 2.0) ticcntcolor = V_REDMAP;
|
|
|
|
else if (fps <= cap * 0.90) ticcntcolor = V_YELLOWMAP;
|
|
|
|
else ticcntcolor = V_GREENMAP;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ticcntcolor = V_GREENMAP;
|
|
|
|
}
|
2014-03-15 16:59:03 +00:00
|
|
|
|
2020-05-08 13:40:50 +00:00
|
|
|
if (cv_ticrate.value == 2) // compact counter
|
2022-04-19 16:06:17 +00:00
|
|
|
{
|
|
|
|
V_DrawRightAlignedString(vid.width, h,
|
|
|
|
ticcntcolor|V_NOSCALESTART|V_USERHUDTRANS, va("%04.2f", averageFPS)); // use averageFPS directly
|
|
|
|
}
|
2020-05-08 13:40:50 +00:00
|
|
|
else if (cv_ticrate.value == 1) // full counter
|
|
|
|
{
|
2022-11-08 00:49:39 +00:00
|
|
|
const char *drawnstr;
|
|
|
|
INT32 width;
|
|
|
|
|
|
|
|
// The highest assignable cap is < 1000, so 3 characters is fine.
|
2022-04-16 05:21:49 +00:00
|
|
|
if (cap > 0)
|
2022-11-08 00:49:39 +00:00
|
|
|
drawnstr = va("%3.0f/%3u", fps, cap);
|
2022-04-16 05:21:49 +00:00
|
|
|
else
|
2022-11-08 00:49:39 +00:00
|
|
|
drawnstr = va("%4.2f", averageFPS);
|
|
|
|
|
|
|
|
width = V_StringWidth(drawnstr, V_NOSCALESTART);
|
|
|
|
|
|
|
|
V_DrawString(vid.width - ((7 * 8 * vid.dupx) + V_StringWidth("FPS: ", V_NOSCALESTART)), h,
|
2020-05-08 13:40:50 +00:00
|
|
|
V_YELLOWMAP|V_NOSCALESTART|V_USERHUDTRANS, "FPS:");
|
2022-11-08 00:49:39 +00:00
|
|
|
V_DrawString(vid.width - width, h,
|
|
|
|
ticcntcolor|V_NOSCALESTART|V_USERHUDTRANS, drawnstr);
|
2020-05-08 13:40:50 +00:00
|
|
|
}
|
2014-03-15 16:59:03 +00:00
|
|
|
}
|
2017-04-17 00:09:47 +00:00
|
|
|
|
2019-11-18 01:22:47 +00:00
|
|
|
void SCR_DisplayLocalPing(void)
|
|
|
|
{
|
|
|
|
UINT32 ping = playerpingtable[consoleplayer]; // consoleplayer's ping is everyone's ping in a splitnetgame :P
|
|
|
|
if (cv_showping.value == 1 || (cv_showping.value == 2 && servermaxping && ping > servermaxping)) // only show 2 (warning) if our ping is at a bad level
|
|
|
|
{
|
2019-11-18 02:53:48 +00:00
|
|
|
INT32 dispy = cv_ticrate.value ? 180 : 189;
|
2019-11-18 02:52:05 +00:00
|
|
|
HU_drawPing(307, dispy, ping, true, V_SNAPTORIGHT | V_SNAPTOBOTTOM);
|
2019-11-18 01:22:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-17 00:09:47 +00:00
|
|
|
void SCR_ClosedCaptions(void)
|
|
|
|
{
|
|
|
|
UINT8 i;
|
2017-09-25 19:35:04 +00:00
|
|
|
boolean gamestopped = (paused || P_AutoPause());
|
2018-03-26 22:53:09 +00:00
|
|
|
INT32 basey = BASEVIDHEIGHT;
|
|
|
|
|
2018-06-14 21:23:20 +00:00
|
|
|
if (gamestate != wipegamestate)
|
|
|
|
return;
|
|
|
|
|
2018-03-26 22:53:09 +00:00
|
|
|
if (gamestate == GS_LEVEL)
|
|
|
|
{
|
2019-07-29 13:55:36 +00:00
|
|
|
if (promptactive)
|
2019-11-01 15:48:50 +00:00
|
|
|
basey -= 42;
|
2019-07-29 13:55:36 +00:00
|
|
|
else if (splitscreen)
|
2018-03-26 22:53:09 +00:00
|
|
|
basey -= 8;
|
2018-06-14 21:23:20 +00:00
|
|
|
else if ((modeattacking == ATTACKING_NIGHTS)
|
2018-06-03 21:41:54 +00:00
|
|
|
|| (!(maptol & TOL_NIGHTS)
|
2018-08-28 19:57:43 +00:00
|
|
|
&& ((cv_powerupdisplay.value == 2) // "Always"
|
2018-08-30 19:08:17 +00:00
|
|
|
|| (cv_powerupdisplay.value == 1 && !camera.chase)))) // "First-person only"
|
2018-03-26 22:53:09 +00:00
|
|
|
basey -= 16;
|
|
|
|
}
|
2017-04-17 00:09:47 +00:00
|
|
|
|
2017-04-17 17:15:42 +00:00
|
|
|
for (i = 0; i < NUMCAPTIONS; i++)
|
2017-04-17 00:09:47 +00:00
|
|
|
{
|
2017-04-29 15:40:07 +00:00
|
|
|
INT32 flags, y;
|
|
|
|
char dot;
|
|
|
|
boolean music;
|
|
|
|
|
|
|
|
if (!closedcaptions[i].s)
|
|
|
|
continue;
|
|
|
|
|
2017-09-25 19:35:04 +00:00
|
|
|
music = (closedcaptions[i].s-S_sfx == sfx_None);
|
|
|
|
|
|
|
|
if (music && !gamestopped && (closedcaptions[i].t < flashingtics) && (closedcaptions[i].t & 1))
|
2017-04-29 15:40:07 +00:00
|
|
|
continue;
|
|
|
|
|
2018-03-21 18:18:45 +00:00
|
|
|
flags = V_SNAPTORIGHT|V_SNAPTOBOTTOM|V_ALLOWLOWERCASE;
|
2018-03-26 22:53:09 +00:00
|
|
|
y = basey-((i + 2)*10);
|
2017-04-29 15:40:07 +00:00
|
|
|
|
|
|
|
if (closedcaptions[i].b)
|
2023-01-26 02:44:04 +00:00
|
|
|
{
|
|
|
|
y -= closedcaptions[i].b * vid.dupy;
|
|
|
|
if (renderisnewtic)
|
|
|
|
{
|
|
|
|
closedcaptions[i].b--;
|
|
|
|
}
|
|
|
|
}
|
2017-04-29 15:40:07 +00:00
|
|
|
|
|
|
|
if (closedcaptions[i].t < CAPTIONFADETICS)
|
|
|
|
flags |= (((CAPTIONFADETICS-closedcaptions[i].t)/2)*V_10TRANS);
|
|
|
|
|
|
|
|
if (music)
|
|
|
|
dot = '\x19';
|
|
|
|
else if (closedcaptions[i].c && closedcaptions[i].c->origin)
|
|
|
|
dot = '\x1E';
|
2018-03-21 18:18:45 +00:00
|
|
|
else
|
|
|
|
dot = ' ';
|
2017-04-29 15:40:07 +00:00
|
|
|
|
2018-03-21 18:18:45 +00:00
|
|
|
V_DrawRightAlignedString(BASEVIDWIDTH - 20, y, flags,
|
|
|
|
va("%c [%s]", dot, (closedcaptions[i].s->caption[0] ? closedcaptions[i].s->caption : closedcaptions[i].s->name)));
|
2017-04-17 00:09:47 +00:00
|
|
|
}
|
|
|
|
}
|
Introducing Marathon Run. (I was going to call it Marathon Mode, but NiGHTS Mode being right next to it on the menu looked terrible.)
Basically a dedicated Record Attack-like experience for speedrunning the game as a continuous chunk rather than ILs. Has several quality of life features.
Benefits include:
* An unambiguous real-time bar across the bottom of the screen, always displaying the current time, ticking up until you reach the ending.
* Disable the console (pausing is still allowed, but the timer will still increment).
* Automatically skip intermissions as if you're holding down the spin button.
* Show centiseconds on HUD automatically, like record attack.
* "Live Event Backups" - a category of run fit for major events like GDQ, where recovery from crashes or chokes makes for better entertainment. Essentially a modified SP savefile, down to using the same basic functions, but has its own filename and tweaked internal layout.
* "spmarathon_start" MainCfg block parameter and "marathonnext" mapheader parameter, allowing for a customised flow (makes this fit for purpose for an eventual SUGOI port).
* Disabling inter-level custom cutscenes by default with a menu option to toggle this (won't show up if the mod doesn't *have* any custom cutscenes), although either way ending cutscenes (vanilla or custom) remain intact since is time is called before them.
* Won't show up if you have a mod that consists of only one level (determined by spmarathon_start's nextlevel; this won't trip if you manually set its marathonnext).
* Unconditional gratitude on the evaluation screen, instead of a negging "Try again..." if you didn't get all the emeralds (which you may not have been aiming for).
* Gorgeous new menu (no new assets required, unless you wanna give it a header later).
Changes which were required for the above but affect other areas of the game include:
* "useBlackRock" MainCFG block parameter, which can be used to disable the presence of the Black Rock or Egg Rock in both the Evaluation screen and the Marathon Run menu (for total conversions with different stories).
* Disabling Continues in NiGHTS mode, to match the most common singleplayer experience post 2.2.4's release (is reverted if useContinues is set to true).
* Hiding the exitmove "powerup" outside of multiplayer. (Okay, this isn't really related, I just saw this bug in action a lot while doing test runs and got annoyed enough to fix it here.)
* The ability to use V_DrawPromptBack (in hardcode only at the moment, but) to draw in terms of pixels rather than rows of text, by providing negative instead of positive inputs).
* A refactoring of redundant game saves smattered across the ending, credits, and evaluation - in addition to saving the game slightly earlier.
* Minor m_menu.c touchups and refactorings here and there.
Built using feedback from the official server's #speedruns channel, among other places.
2020-05-14 22:10:00 +00:00
|
|
|
|
|
|
|
void SCR_DisplayMarathonInfo(void)
|
|
|
|
{
|
|
|
|
INT32 flags = V_SNAPTOBOTTOM;
|
2020-05-15 14:08:45 +00:00
|
|
|
static tic_t entertic, oldentertics = 0, antisplice[2] = {48,0};
|
Introducing Marathon Run. (I was going to call it Marathon Mode, but NiGHTS Mode being right next to it on the menu looked terrible.)
Basically a dedicated Record Attack-like experience for speedrunning the game as a continuous chunk rather than ILs. Has several quality of life features.
Benefits include:
* An unambiguous real-time bar across the bottom of the screen, always displaying the current time, ticking up until you reach the ending.
* Disable the console (pausing is still allowed, but the timer will still increment).
* Automatically skip intermissions as if you're holding down the spin button.
* Show centiseconds on HUD automatically, like record attack.
* "Live Event Backups" - a category of run fit for major events like GDQ, where recovery from crashes or chokes makes for better entertainment. Essentially a modified SP savefile, down to using the same basic functions, but has its own filename and tweaked internal layout.
* "spmarathon_start" MainCfg block parameter and "marathonnext" mapheader parameter, allowing for a customised flow (makes this fit for purpose for an eventual SUGOI port).
* Disabling inter-level custom cutscenes by default with a menu option to toggle this (won't show up if the mod doesn't *have* any custom cutscenes), although either way ending cutscenes (vanilla or custom) remain intact since is time is called before them.
* Won't show up if you have a mod that consists of only one level (determined by spmarathon_start's nextlevel; this won't trip if you manually set its marathonnext).
* Unconditional gratitude on the evaluation screen, instead of a negging "Try again..." if you didn't get all the emeralds (which you may not have been aiming for).
* Gorgeous new menu (no new assets required, unless you wanna give it a header later).
Changes which were required for the above but affect other areas of the game include:
* "useBlackRock" MainCFG block parameter, which can be used to disable the presence of the Black Rock or Egg Rock in both the Evaluation screen and the Marathon Run menu (for total conversions with different stories).
* Disabling Continues in NiGHTS mode, to match the most common singleplayer experience post 2.2.4's release (is reverted if useContinues is set to true).
* Hiding the exitmove "powerup" outside of multiplayer. (Okay, this isn't really related, I just saw this bug in action a lot while doing test runs and got annoyed enough to fix it here.)
* The ability to use V_DrawPromptBack (in hardcode only at the moment, but) to draw in terms of pixels rather than rows of text, by providing negative instead of positive inputs).
* A refactoring of redundant game saves smattered across the ending, credits, and evaluation - in addition to saving the game slightly earlier.
* Minor m_menu.c touchups and refactorings here and there.
Built using feedback from the official server's #speedruns channel, among other places.
2020-05-14 22:10:00 +00:00
|
|
|
const char *str;
|
|
|
|
#if 0 // eh, this probably isn't going to be a problem
|
|
|
|
if (((signed)marathontime) < 0)
|
|
|
|
{
|
|
|
|
flags |= V_REDMAP;
|
|
|
|
str = "No waiting out the clock to submit a bogus time.";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
entertic = I_GetTime();
|
|
|
|
if (gamecomplete)
|
|
|
|
flags |= V_YELLOWMAP;
|
2020-05-15 15:33:20 +00:00
|
|
|
else if (marathonmode & MA_INGAME)
|
|
|
|
; // see also G_Ticker
|
Introducing Marathon Run. (I was going to call it Marathon Mode, but NiGHTS Mode being right next to it on the menu looked terrible.)
Basically a dedicated Record Attack-like experience for speedrunning the game as a continuous chunk rather than ILs. Has several quality of life features.
Benefits include:
* An unambiguous real-time bar across the bottom of the screen, always displaying the current time, ticking up until you reach the ending.
* Disable the console (pausing is still allowed, but the timer will still increment).
* Automatically skip intermissions as if you're holding down the spin button.
* Show centiseconds on HUD automatically, like record attack.
* "Live Event Backups" - a category of run fit for major events like GDQ, where recovery from crashes or chokes makes for better entertainment. Essentially a modified SP savefile, down to using the same basic functions, but has its own filename and tweaked internal layout.
* "spmarathon_start" MainCfg block parameter and "marathonnext" mapheader parameter, allowing for a customised flow (makes this fit for purpose for an eventual SUGOI port).
* Disabling inter-level custom cutscenes by default with a menu option to toggle this (won't show up if the mod doesn't *have* any custom cutscenes), although either way ending cutscenes (vanilla or custom) remain intact since is time is called before them.
* Won't show up if you have a mod that consists of only one level (determined by spmarathon_start's nextlevel; this won't trip if you manually set its marathonnext).
* Unconditional gratitude on the evaluation screen, instead of a negging "Try again..." if you didn't get all the emeralds (which you may not have been aiming for).
* Gorgeous new menu (no new assets required, unless you wanna give it a header later).
Changes which were required for the above but affect other areas of the game include:
* "useBlackRock" MainCFG block parameter, which can be used to disable the presence of the Black Rock or Egg Rock in both the Evaluation screen and the Marathon Run menu (for total conversions with different stories).
* Disabling Continues in NiGHTS mode, to match the most common singleplayer experience post 2.2.4's release (is reverted if useContinues is set to true).
* Hiding the exitmove "powerup" outside of multiplayer. (Okay, this isn't really related, I just saw this bug in action a lot while doing test runs and got annoyed enough to fix it here.)
* The ability to use V_DrawPromptBack (in hardcode only at the moment, but) to draw in terms of pixels rather than rows of text, by providing negative instead of positive inputs).
* A refactoring of redundant game saves smattered across the ending, credits, and evaluation - in addition to saving the game slightly earlier.
* Minor m_menu.c touchups and refactorings here and there.
Built using feedback from the official server's #speedruns channel, among other places.
2020-05-14 22:10:00 +00:00
|
|
|
else if (marathonmode & MA_INIT)
|
|
|
|
marathonmode &= ~MA_INIT;
|
|
|
|
else
|
|
|
|
marathontime += entertic - oldentertics;
|
2020-05-15 15:33:20 +00:00
|
|
|
|
2020-05-15 14:08:45 +00:00
|
|
|
// Create a sequence of primes such that their LCM is nice and big.
|
|
|
|
#define PRIMEV1 13
|
|
|
|
#define PRIMEV2 17 // I can't believe it! I'm on TV!
|
|
|
|
antisplice[0] += (entertic - oldentertics)*PRIMEV2;
|
|
|
|
antisplice[0] %= PRIMEV1*((vid.width/vid.dupx)+1);
|
|
|
|
antisplice[1] += (entertic - oldentertics)*PRIMEV1;
|
|
|
|
antisplice[1] %= PRIMEV1*((vid.width/vid.dupx)+1);
|
Introducing Marathon Run. (I was going to call it Marathon Mode, but NiGHTS Mode being right next to it on the menu looked terrible.)
Basically a dedicated Record Attack-like experience for speedrunning the game as a continuous chunk rather than ILs. Has several quality of life features.
Benefits include:
* An unambiguous real-time bar across the bottom of the screen, always displaying the current time, ticking up until you reach the ending.
* Disable the console (pausing is still allowed, but the timer will still increment).
* Automatically skip intermissions as if you're holding down the spin button.
* Show centiseconds on HUD automatically, like record attack.
* "Live Event Backups" - a category of run fit for major events like GDQ, where recovery from crashes or chokes makes for better entertainment. Essentially a modified SP savefile, down to using the same basic functions, but has its own filename and tweaked internal layout.
* "spmarathon_start" MainCfg block parameter and "marathonnext" mapheader parameter, allowing for a customised flow (makes this fit for purpose for an eventual SUGOI port).
* Disabling inter-level custom cutscenes by default with a menu option to toggle this (won't show up if the mod doesn't *have* any custom cutscenes), although either way ending cutscenes (vanilla or custom) remain intact since is time is called before them.
* Won't show up if you have a mod that consists of only one level (determined by spmarathon_start's nextlevel; this won't trip if you manually set its marathonnext).
* Unconditional gratitude on the evaluation screen, instead of a negging "Try again..." if you didn't get all the emeralds (which you may not have been aiming for).
* Gorgeous new menu (no new assets required, unless you wanna give it a header later).
Changes which were required for the above but affect other areas of the game include:
* "useBlackRock" MainCFG block parameter, which can be used to disable the presence of the Black Rock or Egg Rock in both the Evaluation screen and the Marathon Run menu (for total conversions with different stories).
* Disabling Continues in NiGHTS mode, to match the most common singleplayer experience post 2.2.4's release (is reverted if useContinues is set to true).
* Hiding the exitmove "powerup" outside of multiplayer. (Okay, this isn't really related, I just saw this bug in action a lot while doing test runs and got annoyed enough to fix it here.)
* The ability to use V_DrawPromptBack (in hardcode only at the moment, but) to draw in terms of pixels rather than rows of text, by providing negative instead of positive inputs).
* A refactoring of redundant game saves smattered across the ending, credits, and evaluation - in addition to saving the game slightly earlier.
* Minor m_menu.c touchups and refactorings here and there.
Built using feedback from the official server's #speedruns channel, among other places.
2020-05-14 22:10:00 +00:00
|
|
|
str = va("%i:%02i:%02i.%02i",
|
|
|
|
G_TicsToHours(marathontime),
|
|
|
|
G_TicsToMinutes(marathontime, false),
|
|
|
|
G_TicsToSeconds(marathontime),
|
|
|
|
G_TicsToCentiseconds(marathontime));
|
|
|
|
oldentertics = entertic;
|
|
|
|
}
|
2020-05-15 14:08:45 +00:00
|
|
|
V_DrawFill((antisplice[0]/PRIMEV1)-1, BASEVIDHEIGHT-8, 1, 8, V_SNAPTOBOTTOM|V_SNAPTOLEFT);
|
|
|
|
V_DrawFill((antisplice[0]/PRIMEV1), BASEVIDHEIGHT-8, 1, 8, V_SNAPTOBOTTOM|V_SNAPTOLEFT|31);
|
|
|
|
V_DrawFill(BASEVIDWIDTH-((antisplice[1]/PRIMEV1)-1), BASEVIDHEIGHT-8, 1, 8, V_SNAPTOBOTTOM|V_SNAPTORIGHT);
|
|
|
|
V_DrawFill(BASEVIDWIDTH-((antisplice[1]/PRIMEV1)), BASEVIDHEIGHT-8, 1, 8, V_SNAPTOBOTTOM|V_SNAPTORIGHT|31);
|
|
|
|
#undef PRIMEV1
|
|
|
|
#undef PRIMEV2
|
Introducing Marathon Run. (I was going to call it Marathon Mode, but NiGHTS Mode being right next to it on the menu looked terrible.)
Basically a dedicated Record Attack-like experience for speedrunning the game as a continuous chunk rather than ILs. Has several quality of life features.
Benefits include:
* An unambiguous real-time bar across the bottom of the screen, always displaying the current time, ticking up until you reach the ending.
* Disable the console (pausing is still allowed, but the timer will still increment).
* Automatically skip intermissions as if you're holding down the spin button.
* Show centiseconds on HUD automatically, like record attack.
* "Live Event Backups" - a category of run fit for major events like GDQ, where recovery from crashes or chokes makes for better entertainment. Essentially a modified SP savefile, down to using the same basic functions, but has its own filename and tweaked internal layout.
* "spmarathon_start" MainCfg block parameter and "marathonnext" mapheader parameter, allowing for a customised flow (makes this fit for purpose for an eventual SUGOI port).
* Disabling inter-level custom cutscenes by default with a menu option to toggle this (won't show up if the mod doesn't *have* any custom cutscenes), although either way ending cutscenes (vanilla or custom) remain intact since is time is called before them.
* Won't show up if you have a mod that consists of only one level (determined by spmarathon_start's nextlevel; this won't trip if you manually set its marathonnext).
* Unconditional gratitude on the evaluation screen, instead of a negging "Try again..." if you didn't get all the emeralds (which you may not have been aiming for).
* Gorgeous new menu (no new assets required, unless you wanna give it a header later).
Changes which were required for the above but affect other areas of the game include:
* "useBlackRock" MainCFG block parameter, which can be used to disable the presence of the Black Rock or Egg Rock in both the Evaluation screen and the Marathon Run menu (for total conversions with different stories).
* Disabling Continues in NiGHTS mode, to match the most common singleplayer experience post 2.2.4's release (is reverted if useContinues is set to true).
* Hiding the exitmove "powerup" outside of multiplayer. (Okay, this isn't really related, I just saw this bug in action a lot while doing test runs and got annoyed enough to fix it here.)
* The ability to use V_DrawPromptBack (in hardcode only at the moment, but) to draw in terms of pixels rather than rows of text, by providing negative instead of positive inputs).
* A refactoring of redundant game saves smattered across the ending, credits, and evaluation - in addition to saving the game slightly earlier.
* Minor m_menu.c touchups and refactorings here and there.
Built using feedback from the official server's #speedruns channel, among other places.
2020-05-14 22:10:00 +00:00
|
|
|
V_DrawPromptBack(-8, cons_backcolor.value);
|
|
|
|
V_DrawCenteredString(BASEVIDWIDTH/2, BASEVIDHEIGHT-8, flags, str);
|
|
|
|
}
|