Merge remote-tracking branch 'remotes/james/netreplays-viewcommand' into new_netreplays

# Conflicts:
#	src/g_game.c
This commit is contained in:
fickleheart 2019-02-16 20:20:11 -06:00
commit 7a7ea255bf
4 changed files with 219 additions and 108 deletions

View file

@ -1914,11 +1914,10 @@ static INT32 LookupPlayer(const char *s)
for (playernum = 0; playernum < MAXPLAYERS; ++playernum) for (playernum = 0; playernum < MAXPLAYERS; ++playernum)
{ {
/* Consider strcasestr? */ /* Match name case-insensitively: fully, or partially the start. */
/* Match name (case-insensitively) fully, or partially the start. */
if (playeringame[playernum]) if (playeringame[playernum])
if (stricmp(player_names[playernum], s) == 0 || if (stricmp(player_names[playernum], s) == 0 ||
strstr(player_names[playernum], s) == player_names[playernum] ) strnicmp(player_names[playernum], s, strlen(s)) == 0)
{ {
return playernum; return playernum;
} }
@ -1969,7 +1968,7 @@ static void Command_View_f(void)
olddisplayplayer = (*displayplayerp); olddisplayplayer = (*displayplayerp);
(*displayplayerp) = playernum; (*displayplayerp) = playernum;
G_ResetView(viewnum); G_ResetView(viewnum, playernum);
/* The player we wanted was corrected to who it already was. */ /* The player we wanted was corrected to who it already was. */
if ((*displayplayerp) == olddisplayplayer) if ((*displayplayerp) == olddisplayplayer)

View file

@ -1915,8 +1915,7 @@ boolean G_Responder(event_t *ev)
displayplayer = consoleplayer; displayplayer = consoleplayer;
else else
{ {
displayplayer++; G_AdjustView(1, 1);
G_ResetView(1);
// change statusbar also if playing back demo // change statusbar also if playing back demo
if (singledemo) if (singledemo)
@ -1930,22 +1929,19 @@ boolean G_Responder(event_t *ev)
{ {
if (ev->data1 == gamecontrolbis[gc_viewpoint][0] || ev->data1 == gamecontrolbis[gc_viewpoint][1]) if (ev->data1 == gamecontrolbis[gc_viewpoint][0] || ev->data1 == gamecontrolbis[gc_viewpoint][1])
{ {
secondarydisplayplayer++; G_AdjustView(2, 1);
G_ResetView(2);
return true; return true;
} }
else if (ev->data1 == gamecontrol3[gc_viewpoint][0] || ev->data1 == gamecontrol3[gc_viewpoint][1]) else if (ev->data1 == gamecontrol3[gc_viewpoint][0] || ev->data1 == gamecontrol3[gc_viewpoint][1])
{ {
thirddisplayplayer++; G_AdjustView(3, 1);
G_ResetView(3);
return true; return true;
} }
else if (ev->data1 == gamecontrol4[gc_viewpoint][0] || ev->data1 == gamecontrol4[gc_viewpoint][1]) else if (ev->data1 == gamecontrol4[gc_viewpoint][0] || ev->data1 == gamecontrol4[gc_viewpoint][1])
{ {
fourthdisplayplayer++; G_AdjustView(4, 1);
G_ResetView(4);
return true; return true;
} }
@ -2073,81 +2069,131 @@ boolean G_Responder(event_t *ev)
return false; return false;
} }
static INT32 G_FindView(INT32 startview) //
// G_CouldView
// Return whether a player could be viewed by any means.
//
boolean G_CouldView(INT32 playernum)
{ {
UINT8 i = 0; // spy mode player_t *player;
startview--; // Ensures view doesn't move if the current view is valid if (playernum < 0 || playernum > MAXPLAYERS-1)
for (i = 0; i < MAXPLAYERS; i++) return false;
{
startview++;
if (startview == MAXPLAYERS)
startview = 0;
if (!demoplayback && startview == consoleplayer) if (!playeringame[playernum])
break; // End loop return false;
if (startview == displayplayer) player = &players[playernum];
continue;
if (splitscreen && startview == secondarydisplayplayer) if (player->spectator)
continue; return false;
if (splitscreen >= 2 && startview == thirddisplayplayer)
continue;
if (splitscreen == 3 && startview == fourthdisplayplayer)
continue;
if (!playeringame[startview])
continue;
if (players[startview].spectator)
continue;
// SRB2Kart: Only go through players who are actually playing // SRB2Kart: Only go through players who are actually playing
if (players[startview].exiting) if (player->exiting)
continue; return false;
if (( player->pflags & PF_TIMEOVER ))
if (players[startview].pflags & PF_TIMEOVER) return false;
continue;
// I don't know if we want this actually, but I'll humor the suggestion anyway // I don't know if we want this actually, but I'll humor the suggestion anyway
if (G_BattleGametype()) if (G_BattleGametype())
{ {
if (players[startview].kartstuff[k_bumper] <= 0) if (player->kartstuff[k_bumper] <= 0)
continue; return false;
} }
// SRB2Kart: we have no team-based modes, YET... // SRB2Kart: we have no team-based modes, YET...
/*if (G_GametypeHasTeams()) /*if (G_GametypeHasTeams())
{ {
if (players[consoleplayer].ctfteam if (players[consoleplayer].ctfteam
&& players[startview].ctfteam != players[consoleplayer].ctfteam) && player->ctfteam != players[consoleplayer].ctfteam)
continue; return false;
} }
else if (gametype == GT_HIDEANDSEEK) else if (gametype == GT_HIDEANDSEEK)
{ {
if (players[consoleplayer].pflags & PF_TAGIT) if (players[consoleplayer].pflags & PF_TAGIT)
continue; return false;
} }
// Other Tag-based gametypes? // Other Tag-based gametypes?
else if (G_TagGametype()) else if (G_TagGametype())
{ {
if (!players[consoleplayer].spectator if (!players[consoleplayer].spectator
&& (players[consoleplayer].pflags & PF_TAGIT) != (players[startview].pflags & PF_TAGIT)) && (players[consoleplayer].pflags & PF_TAGIT) != (player->pflags & PF_TAGIT))
continue; return false;
} }
else if (G_GametypeHasSpectators() && G_BattleGametype()) else if (G_GametypeHasSpectators() && G_BattleGametype())
{ {
if (!players[consoleplayer].spectator) if (!players[consoleplayer].spectator)
continue; return false;
}*/ }*/
break; return true;
}
//
// G_CanView
// Return whether a player can be viewed on a particular view (splitscreen).
//
boolean G_CanView(INT32 playernum, UINT8 viewnum)
{
UINT8 splits;
UINT8 viewd;
INT32 *displayplayerp;
if (!G_CouldView(playernum))
return false;
splits = splitscreen+1;
if (viewnum > splits)
viewnum = splits;
for (viewd = 1; viewd < viewnum; ++viewd)
{
displayplayerp = (G_GetDisplayplayerPtr(viewd));
if ((*displayplayerp) == playernum)
return false;
}
for (viewd = viewnum + 1; viewd <= splits; ++viewd)
{
displayplayerp = (G_GetDisplayplayerPtr(viewd));
if ((*displayplayerp) == playernum)
return false;
} }
return startview; return true;
}
//
// G_FindView
// Return the next player that can be viewed on a view, wraps forward.
// An out of range startview is corrected.
//
INT32 G_FindView(INT32 startview, UINT8 viewnum)
{
INT32 i;
startview = min(max(startview, 0), MAXPLAYERS);
for (i = startview; i < MAXPLAYERS; ++i)
{
if (G_CanView(i, viewnum))
return i;
}
for (i = 0; i < startview; ++i)
{
if (G_CanView(i, viewnum))
return i;
}
return -1;
}
INT32 G_CountPlayersPotentiallyViewable(void)
{
INT32 total = 0;
INT32 i;
for (i = 0; i < MAXPLAYERS; ++i)
{
if (G_CouldView(i))
total++;
}
return total;
} }
INT32 *G_GetDisplayplayerPtr(UINT8 viewnum) INT32 *G_GetDisplayplayerPtr(UINT8 viewnum)
@ -2163,64 +2209,114 @@ INT32 *G_GetDisplayplayerPtr(UINT8 viewnum)
// //
// G_ResetView // G_ResetView
// Ensures a viewpoint is valid. // Correct a viewpoint to playernum or the next available, wraps forward.
// Also promotes splitscreen up to available viewable players.
// An out of range playernum is corrected.
// //
void G_ResetView(UINT8 viewnum) void G_ResetView(UINT8 viewnum, INT32 playernum)
{ {
UINT8 splits;
UINT8 viewd;
INT32 *displayplayerp; INT32 *displayplayerp;
camera_t *camerap; camera_t *camerap;
INT32 newdisplayplayer; INT32 olddisplayplayer;
UINT8 viewd, splits; INT32 playersviewable;
splits = splitscreen+1; splits = splitscreen+1;
displayplayerp = (G_GetDisplayplayerPtr(viewnum)); /* Promote splits */
newdisplayplayer = (*displayplayerp); if (viewnum > splits)
{
playersviewable = G_CountPlayersPotentiallyViewable();
if (playersviewable < splits)/* do not demote */
return;
if (viewnum > playersviewable)
viewnum = playersviewable;
splitscreen = viewnum-1;
/* Prepare extra views for G_FindView to pass. */
for (viewd = splits+1; viewd < viewnum; ++viewd)
{
displayplayerp = (G_GetDisplayplayerPtr(viewd));
(*displayplayerp) = INT32_MAX; (*displayplayerp) = INT32_MAX;
(*displayplayerp) = G_FindView(newdisplayplayer); }
R_ExecuteSetViewSize();
}
/* Focus our target view first so that we don't take its player. */
displayplayerp = (G_GetDisplayplayerPtr(viewnum));
olddisplayplayer = (*displayplayerp);
(*displayplayerp) = G_FindView(playernum, viewnum);
if ((*displayplayerp) != olddisplayplayer)
{
camerap = (P_GetCameraPtr(viewnum));
P_ResetCamera(&players[(*displayplayerp)], camerap);
}
if (viewnum > splits) if (viewnum > splits)
{ {
splitscreen = viewnum-1;
for (viewd = splits+1; viewd < viewnum; ++viewd) for (viewd = splits+1; viewd < viewnum; ++viewd)
{
(*(G_GetDisplayplayerPtr(viewd))) = INT32_MAX;/* ensure clean */
}
/* Initialise views up from current splitscreen. */
for (viewd = splits+1 ;; )
{ {
displayplayerp = (G_GetDisplayplayerPtr(viewd)); displayplayerp = (G_GetDisplayplayerPtr(viewd));
camerap = (P_GetCameraPtr(viewd)); camerap = (P_GetCameraPtr(viewd));
(*displayplayerp) = G_FindView(0, viewd);
P_ResetCamera(&players[(*displayplayerp)], camerap); P_ResetCamera(&players[(*displayplayerp)], camerap);
if (++viewd > viewnum)
break;
/* Correct up to but viewnum */
(*displayplayerp) = G_FindView(displayplayer);
} }
R_ExecuteSetViewSize();
} }
if (viewnum == 1 && demoplayback) if (viewnum == 1 && demoplayback)
consoleplayer = displayplayer; consoleplayer = displayplayer;
} }
//
// G_AdjustView
// Increment a viewpoint by offset from the current player. A negative value
// decrements.
//
void G_AdjustView(UINT8 viewnum, INT32 offset)
{
INT32 *displayplayerp;
displayplayerp = G_GetDisplayplayerPtr(viewnum);
G_ResetView(viewnum, ( (*displayplayerp) + offset ));
}
// //
// G_ResetViews // G_ResetViews
// Ensures all viewpoints are valid // Ensures all viewpoints are valid
// Also demotes splitscreen down to one player.
// //
void G_ResetViews(void) void G_ResetViews(void)
{ {
UINT8 viewnum = splitscreen+1; UINT8 splits;
do UINT8 viewd;
INT32 playersviewable;
splits = splitscreen+1;
playersviewable = G_CountPlayersPotentiallyViewable();
/* Demote splits */
if (playersviewable < splits)
{ {
G_ResetView(viewnum); splits = playersviewable;
splitscreen = max(splits-1, 0);
R_ExecuteSetViewSize();
}
/*
Consider installing a method to focus the last
view elsewhere if all players spectate?
*/
for (viewd = 1; viewd <= splits; ++viewd)
{
G_AdjustView(viewd, 0);
} }
while (--viewnum > 0) ;
} }
// //

View file

@ -232,8 +232,15 @@ boolean G_Responder(event_t *ev);
INT32 *G_GetDisplayplayerPtr(UINT8 viewnum); INT32 *G_GetDisplayplayerPtr(UINT8 viewnum);
boolean G_CouldView(INT32 playernum);
boolean G_CanView(INT32 playernum, UINT8 viewnum);
INT32 G_FindView(INT32 startview, UINT8 viewnum);
INT32 G_CountPlayersPotentiallyViewable(void);
void G_ResetViews(void); void G_ResetViews(void);
void G_ResetView(UINT8 viewnum); void G_ResetView(UINT8 viewnum, INT32 playernum);
void G_AdjustView(UINT8 viewnum, INT32 offset);
void G_AddPlayer(INT32 playernum); void G_AddPlayer(INT32 playernum);

View file

@ -2014,6 +2014,15 @@ static void ST_overlayDrawer(void)
} }
} }
// Replay manual-save stuff
if (demorecording && multiplayer && demosavebutton && demosavebutton + 3*TICRATE < leveltime)
{
if (demodefersave || cv_recordmultiplayerdemos.value == 2)
V_DrawCenteredString(BASEVIDWIDTH/2, 178, V_HUDTRANSHALF|V_ALLOWLOWERCASE|(G_BattleGametype() ? V_REDMAP : V_SKYMAP), "Replay will be saved.");
else
V_DrawCenteredString(BASEVIDWIDTH/2, 178, V_HUDTRANSHALF|V_ALLOWLOWERCASE|(G_BattleGametype() ? V_REDMAP : V_SKYMAP), "Press Look Backward to save the replay");
}
ST_drawDebugInfo(); ST_drawDebugInfo();
} }