mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-24 01:11:21 +00:00
560 lines
15 KiB
C++
560 lines
15 KiB
C++
//-------------------------------------------------------------------------
|
|
/*
|
|
Copyright (C) 1997, 2005 - 3D Realms Entertainment
|
|
|
|
This file is part of Shadow Warrior version 1.2
|
|
|
|
Shadow Warrior 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 2
|
|
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.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
Original Source: 1997 - Frank Maddin and Jim Norwood
|
|
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
|
|
*/
|
|
//-------------------------------------------------------------------------
|
|
#include "ns.h"
|
|
|
|
#include "build.h"
|
|
|
|
#include "keys.h"
|
|
#include "names2.h"
|
|
#include "panel.h"
|
|
#include "lists.h"
|
|
#include "game.h"
|
|
#include "common_game.h"
|
|
#include "network.h"
|
|
#include "text.h"
|
|
#include "menus.h"
|
|
|
|
BEGIN_SW_NS
|
|
|
|
|
|
|
|
#define BAR_HEIGHT 48
|
|
#define XDIM 320
|
|
#define YDIM 200
|
|
|
|
short DebugBorderShade = 0;
|
|
|
|
short RegBorderTest[] =
|
|
{
|
|
51, 53, 127, 128, 140, 145, 152, 197, 198, 201, 205, 206, 213, 218, 242, 243,
|
|
245, 246, 247, 257, 2560, 2561, 2562, 2570, 2571, 2572, 2573, 2576, 2578,
|
|
2579, 2580, 2581, 2582, 2583, 2584, 2585, 2593, 2594
|
|
};
|
|
short SWBorderTest[] =
|
|
{
|
|
51, 53, 127, 128, 140, 145, 201, 205, 206, 213, 218,
|
|
245, 2560, 2573, 2576, 2580, 2581, 2582, 2583, 2584, 2593
|
|
};
|
|
|
|
#undef BORDER_TILE
|
|
#define BORDER_TILE \
|
|
(isShareware ? \
|
|
SWBorderTest[gs.BorderTile % SIZ(SWBorderTest)] : \
|
|
RegBorderTest[gs.BorderTile % SIZ(RegBorderTest)] \
|
|
)
|
|
|
|
#define f_320 FIXED(320,0)
|
|
#define f_200 FIXED(200,0)
|
|
|
|
#define X_TO_FIXED(val) (x_aspect_mul*(val))
|
|
#define Y_TO_FIXED(val) (y_aspect_mul*(val))
|
|
|
|
SWBOOL RedrawScreen = FALSE;
|
|
|
|
int f_xdim, f_ydim, x_pix_size, y_pix_size, x_aspect_mul, y_aspect_mul;
|
|
int CrosshairX, CrosshairY;
|
|
|
|
extern SWBOOL BorderAdjust;
|
|
SWBOOL GlobSpriteBoxUpdateEveryFrame = FALSE;
|
|
|
|
PANEL_SPRITEp
|
|
pSpawnFullScreenSpriteBox(PLAYERp pp, short id, short pic, short pri, int x, int y, short x1, short y1, short x2, short y2)
|
|
{
|
|
PANEL_SPRITEp psp;
|
|
|
|
psp = pSpawnSprite(pp, NULL, pri, x, y);
|
|
|
|
psp->ID = id;
|
|
psp->numpages = numpages;
|
|
if (GlobSpriteBoxUpdateEveryFrame)
|
|
{
|
|
psp->numpages = 1;
|
|
}
|
|
psp->picndx = -1;
|
|
psp->picnum = pic;
|
|
psp->x1 = x1;
|
|
psp->y1 = y1;
|
|
psp->x2 = x2;
|
|
psp->y2 = y2;
|
|
psp->shade = DebugBorderShade;
|
|
|
|
//SET(psp->flags, PANF_STATUS_AREA | PANF_KILL_AFTER_SHOW | PANF_IGNORE_START_MOST | PANF_DRAW_BEFORE_VIEW | PANF_NOT_ALL_PAGES);
|
|
SET(psp->flags, PANF_STATUS_AREA | PANF_KILL_AFTER_SHOW | PANF_IGNORE_START_MOST | PANF_DRAW_BEFORE_VIEW);
|
|
//extern SWBOOL DrawBeforeView;
|
|
//DrawBeforeView = TRUE;
|
|
|
|
//SET(psp->flags, PANF_SCREEN_CLIP | PANF_KILL_AFTER_SHOW | PANF_IGNORE_START_MOST);
|
|
|
|
return psp;
|
|
}
|
|
|
|
void SetCrosshair(void)
|
|
{
|
|
int wdx,wdy,x,y;
|
|
|
|
wdx = ((windowxy2.x-windowxy1.x)/2);
|
|
wdy = ((windowxy2.y-windowxy1.y)/2);
|
|
x = windowxy1.x + wdx;
|
|
y = windowxy1.y + wdy;
|
|
|
|
CrosshairX = x / (xdim/320.0);
|
|
CrosshairY = y / (ydim/200.0);
|
|
|
|
// rotatesprite takes FIXED point number
|
|
CrosshairX <<= 16;
|
|
CrosshairY <<= 16;
|
|
}
|
|
|
|
|
|
void
|
|
SetupAspectRatio(void)
|
|
{
|
|
f_xdim = FIXED(xdim, 0);
|
|
f_ydim = FIXED(ydim, 0);
|
|
|
|
x_pix_size = (f_320 / xdim);
|
|
y_pix_size = (f_200 / ydim);
|
|
|
|
x_aspect_mul = (f_xdim / 320);
|
|
y_aspect_mul = (f_ydim / 200);
|
|
}
|
|
|
|
void
|
|
SetConsoleDmost(void)
|
|
{
|
|
int ystart;
|
|
|
|
int i;
|
|
int adj=0;
|
|
|
|
// dont setup the startumost/dmost arrays if border is 0
|
|
if (gs.BorderNum == BORDER_NONE || gs.BorderNum == BORDER_MINI_BAR)
|
|
return;
|
|
|
|
//
|
|
// Set the whole thing to the size of the bar
|
|
//
|
|
|
|
ystart = f_ydim - Y_TO_FIXED(BAR_HEIGHT);
|
|
|
|
if (ydim == 480 && gs.BorderNum == 2)
|
|
adj = 1;
|
|
}
|
|
|
|
void
|
|
SetFragBar(PLAYERp pp)
|
|
{
|
|
short i, num_frag_bars;
|
|
int y;
|
|
extern int16_t OrigCommPlayers;
|
|
|
|
if (numplayers <= 1)
|
|
return;
|
|
|
|
if (gNet.MultiGameType == MULTI_GAME_COOPERATIVE)
|
|
return;
|
|
|
|
// if player sprite has not been initialized we have no business
|
|
// sticking a frag bar up. Prevents processing from MenuLevel etc.
|
|
if (!pp->SpriteP)
|
|
return;
|
|
|
|
//num_frag_bars = ((numplayers-1)/4)+1;
|
|
num_frag_bars = ((OrigCommPlayers-1)/4)+1;
|
|
|
|
for (i = windowxy1.x; i <= windowxy2.x; i++)
|
|
{
|
|
y = (tilesiz[FRAG_BAR].y * num_frag_bars) - (2 * (num_frag_bars-1));
|
|
y = y * (ydim/200.0);
|
|
}
|
|
|
|
for (i = 0, y = 0; i < num_frag_bars; i++)
|
|
{
|
|
pSpawnFullScreenSprite(pp, FRAG_BAR, PRI_MID, 0, y);
|
|
y += tilesiz[FRAG_BAR].y - 2;
|
|
}
|
|
|
|
// write each persons kill info to everybody
|
|
// for (i = 0; i < numplayers; i++)
|
|
TRAVERSE_CONNECT(i)
|
|
{
|
|
PlayerUpdateKills(Player + i, 0);
|
|
DisplayFragNames(Player + i);
|
|
}
|
|
}
|
|
|
|
SWBOOL RectOverlap(short tx1, short ty1, short bx1, short by1, short tx2, short ty2, short bx2, short by2)
|
|
{
|
|
if (bx1 >= tx2)
|
|
if (tx1 <= bx2)
|
|
if (ty1 <= by2)
|
|
if (by1 >= ty2)
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void DrawBorderShade(PLAYERp pp, short shade_num, short wx1, short wy1, short wx2, short wy2)
|
|
{
|
|
short i,j,k,l;
|
|
PANEL_SPRITEp psp;
|
|
int dark_shade = 27 - (shade_num * 6);
|
|
int light_shade = 20 - (shade_num * 6);
|
|
|
|
for (i = 0; i < xdim; i += tilesiz[BORDER_TILE].x)
|
|
{
|
|
for (j = 0; j < ydim; j += tilesiz[BORDER_TILE].y)
|
|
{
|
|
k = i + tilesiz[BORDER_TILE].x;
|
|
l = j + tilesiz[BORDER_TILE].y;
|
|
|
|
if (RectOverlap(i, j, k, l, wx1 - 1, wy1 - 1, wx2 + 1, wy1))
|
|
{
|
|
// draw top box of the border
|
|
psp = pSpawnFullScreenSpriteBox(pp, ID_BORDER_TOP, BORDER_TILE, PRI_BACK + 1, i, j, wx1 - 1, wy1 - 1, wx2 + 1, wy1);
|
|
psp->shade = dark_shade;
|
|
psp->ID = ID_BORDER_SHADE;
|
|
}
|
|
|
|
if (RectOverlap(i, j, k, l, wx1 - 1, wy2, wx2 + 1, wy2 + 1))
|
|
{
|
|
// draw bottom box of the border
|
|
psp = pSpawnFullScreenSpriteBox(pp, ID_BORDER_BOTTOM, BORDER_TILE, PRI_BACK + 1, i, j, wx1 - 1, wy2, wx2 + 1, wy2 + 1);
|
|
psp->shade = light_shade;
|
|
psp->ID = ID_BORDER_SHADE;
|
|
}
|
|
if (RectOverlap(i, j, k, l, wx1 - 1, wy1 - 1, wx1, wy2 + 1))
|
|
{
|
|
// draw left box of the border
|
|
psp = pSpawnFullScreenSpriteBox(pp, ID_BORDER_LEFT, BORDER_TILE, PRI_BACK + 1, i, j, wx1 - 1, wy1 - 1, wx1, wy2 + 1);
|
|
psp->shade = dark_shade;
|
|
psp->ID = ID_BORDER_SHADE;
|
|
}
|
|
if (RectOverlap(i, j, k, l, wx2, wy1 - 1, wx2 + 1, wy2 + 1))
|
|
{
|
|
// draw right box of the border
|
|
psp = pSpawnFullScreenSpriteBox(pp, ID_BORDER_RIGHT, BORDER_TILE, PRI_BACK + 1, i, j, wx2, wy1 - 1, wx2 + 1, wy2 + 1);
|
|
psp->shade = light_shade;
|
|
psp->ID = ID_BORDER_SHADE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
BorderShade(PLAYERp pp, SWBOOL refresh)
|
|
{
|
|
int wx1, wx2, wy1, wy2;
|
|
uint8_t lines;
|
|
|
|
wx1 = windowxy1.x - 1;
|
|
wy1 = windowxy1.y - 1;
|
|
wx2 = windowxy2.x + 1;
|
|
wy2 = windowxy2.y + 1;
|
|
|
|
for (lines = 0; lines < 4; lines++)
|
|
{
|
|
|
|
// make sure that these values dont go out of bound - which they do
|
|
wx1 = max(wx1, 0);
|
|
wx2 = min(wx2, xdim - 1);
|
|
wy1 = max(wy1, 0);
|
|
|
|
if (refresh)
|
|
{
|
|
// silly thing seems off by 1
|
|
wy2 = min(wy2, ydim - (Y_TO_FIXED(BAR_HEIGHT) >> 16) - 2);
|
|
}
|
|
else
|
|
{
|
|
if (gs.BorderNum >= BORDER_BAR+1 && gs.BorderNum <= BORDER_BAR+2)
|
|
wy2 = min(wy2, ydim - 1);
|
|
else
|
|
wy2 = min(wy2, ydim - (Y_TO_FIXED(BAR_HEIGHT) >> 16) - 1);
|
|
}
|
|
|
|
DrawBorderShade(pp, lines, wx1, wy1, wx2, wy2);
|
|
// increase view size by one - dont do a set view though
|
|
wx1--;
|
|
wy1--;
|
|
wx2++;
|
|
wy2++;
|
|
}
|
|
}
|
|
|
|
|
|
BORDER_INFO BorderInfoValues[] =
|
|
{
|
|
// x,y,screensize
|
|
{0, 0, 0},
|
|
{0, 0, 0},
|
|
{0, BAR_HEIGHT, 0},
|
|
|
|
{0, BAR_HEIGHT, (1 * 16)},
|
|
{0, BAR_HEIGHT, (2 * 16)},
|
|
{0, BAR_HEIGHT, (3 * 16)},
|
|
{0, BAR_HEIGHT, (4 * 16)},
|
|
{0, BAR_HEIGHT, (5 * 16)},
|
|
{0, BAR_HEIGHT, (6 * 16)},
|
|
{0, BAR_HEIGHT, (7 * 16)},
|
|
{0, BAR_HEIGHT, (8 * 16)},
|
|
{0, BAR_HEIGHT, (9 * 16)},
|
|
{0, BAR_HEIGHT, (10 * 16)},
|
|
{0, BAR_HEIGHT, (11 * 16)},
|
|
{0, BAR_HEIGHT, (12 * 16)}
|
|
};
|
|
|
|
|
|
void DrawBorder(PLAYERp pp, short x, short y, short x2, short y2)
|
|
{
|
|
short i,j,k,l;
|
|
short count = 0;
|
|
|
|
for (i = 0; i < xdim; i += tilesiz[BORDER_TILE].x)
|
|
{
|
|
for (j = 0; j < ydim; j += tilesiz[BORDER_TILE].y)
|
|
{
|
|
k = i + tilesiz[BORDER_TILE].x;
|
|
l = j + tilesiz[BORDER_TILE].y;
|
|
|
|
if (RectOverlap(i, j, k, l, x, y, windowxy1.x-1, y2))
|
|
{
|
|
// draw top box of the border
|
|
pSpawnFullScreenSpriteBox(pp, ID_BORDER_TOP, BORDER_TILE, PRI_BACK, i, j, x, y, windowxy1.x-1, y2);
|
|
count++;
|
|
}
|
|
|
|
if (RectOverlap(i, j, k, l, windowxy2.x+1, y, x2, y2))
|
|
{
|
|
// draw bottom box of the border
|
|
pSpawnFullScreenSpriteBox(pp, ID_BORDER_BOTTOM, BORDER_TILE, PRI_BACK, i, j, windowxy2.x+1, y, x2, y2);
|
|
count++;
|
|
}
|
|
|
|
if (RectOverlap(i, j, k, l, windowxy1.x, y, windowxy2.x, windowxy1.y-1))
|
|
{
|
|
// draw left box of the border
|
|
pSpawnFullScreenSpriteBox(pp, ID_BORDER_LEFT, BORDER_TILE, PRI_BACK, i, j, windowxy1.x, y, windowxy2.x, windowxy1.y-1);
|
|
count++;
|
|
}
|
|
|
|
if (RectOverlap(i, j, k, l, windowxy1.x, windowxy2.y+1, windowxy2.x, y2))
|
|
{
|
|
// draw right box of the border
|
|
pSpawnFullScreenSpriteBox(pp, ID_BORDER_RIGHT, BORDER_TILE, PRI_BACK, i, j, windowxy1.x, windowxy2.y+1, windowxy2.x, y2);
|
|
count++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void DrawPanelBorderSides(PLAYERp pp, short x, short y, short x2, short y2, short panl, short panr)
|
|
{
|
|
short i,j,k,l;
|
|
short count = 0;
|
|
|
|
for (i = 0; i < xdim; i += tilesiz[BORDER_TILE].x)
|
|
{
|
|
for (j = 0; j < ydim; j += tilesiz[BORDER_TILE].y)
|
|
{
|
|
k = i + tilesiz[BORDER_TILE].x;
|
|
l = j + tilesiz[BORDER_TILE].y;
|
|
|
|
if (RectOverlap(i, j, k, l, x, y, panl, y2))
|
|
{
|
|
pSpawnFullScreenSpriteBox(pp, ID_PANEL_BORDER_LEFT, BORDER_TILE, PRI_BACK, i, j, x, y, panl, y2);
|
|
count++;
|
|
}
|
|
|
|
if (RectOverlap(i, j, k, l, panr, y, x2, y2))
|
|
{
|
|
pSpawnFullScreenSpriteBox(pp, ID_PANEL_BORDER_RIGHT, BORDER_TILE, PRI_BACK, i, j, panr, y, x2, y2);
|
|
count++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void BorderSetView(PLAYERp, int *Xdim, int *Ydim, int *ScreenSize)
|
|
{
|
|
int x, x2, y, y2;
|
|
BORDER_INFO *b;
|
|
|
|
BorderInfo = BorderInfoValues[gs.BorderNum];
|
|
|
|
b = &BorderInfo;
|
|
|
|
// figure out the viewing window x and y dimensions
|
|
*Xdim = MSW(f_xdim - X_TO_FIXED(b->Xdim));
|
|
*Ydim = MSW(f_ydim - Y_TO_FIXED(b->Ydim));
|
|
*ScreenSize = MSW(f_xdim - X_TO_FIXED(b->ScreenSize));
|
|
|
|
// figure out the viewing window x and y coordinates
|
|
x = DIV2(*Xdim) - DIV2(*ScreenSize);
|
|
x2 = x + *ScreenSize - 1;
|
|
y = DIV2(*Ydim) - DIV2((*ScreenSize **Ydim) / *Xdim);
|
|
y2 = y + ((*ScreenSize **Ydim) / *Xdim) - 1;
|
|
|
|
// avoid a one-pixel tall HOM
|
|
if (gs.BorderNum == BORDER_BAR)
|
|
++y2;
|
|
|
|
// global windowxy1, windowxy2 coords set here
|
|
videoSetViewableArea(x, y, x2, y2);
|
|
SetCrosshair();
|
|
}
|
|
|
|
//
|
|
// Redraw the border without changing the view
|
|
//
|
|
|
|
static void
|
|
BorderRefresh(PLAYERp pp)
|
|
{
|
|
int x, x2, y, y2;
|
|
BORDER_INFO *b;
|
|
|
|
if (pp != Player + myconnectindex)
|
|
return;
|
|
|
|
if (!BorderAdjust)
|
|
return;
|
|
|
|
if (gs.BorderNum < BORDER_BAR)
|
|
return;
|
|
|
|
// Redraw the BORDER_TILE only if getting smaller
|
|
BorderInfo = BorderInfoValues[gs.BorderNum];
|
|
|
|
b = &BorderInfo;
|
|
|
|
// A refresh does not change the view size so we dont need to do a
|
|
// setview
|
|
// We don't need the calculations for the border drawing boxes - its
|
|
// the whole screen
|
|
// minus the border if necessary
|
|
|
|
// fill in the sides of the panel when the screen is wide
|
|
if (gs.BorderNum >= BORDER_BAR && r_usenewaspect)
|
|
{
|
|
const int sidew = (xdim - scale(4, ydim, 3)) / 2;
|
|
|
|
x = 0;
|
|
x2 = xdim - 1;
|
|
|
|
y = ydim - (Y_TO_FIXED(b->Ydim) >> 16);
|
|
y2 = ydim - 1;
|
|
|
|
DrawPanelBorderSides(pp, x, y, x2, y2, sidew, xdim-sidew);
|
|
}
|
|
|
|
// only need a border if border is > BORDER_BAR
|
|
if (gs.BorderNum > BORDER_BAR)
|
|
{
|
|
// make sure that these values dont go out of bound - which they do
|
|
x = 0;
|
|
x2 = xdim - 1;
|
|
|
|
y = 0;
|
|
y2 = ydim - (Y_TO_FIXED(b->Ydim) >> 16) - 1;
|
|
|
|
DrawBorder(pp, x, y, x2, y2);
|
|
|
|
// kill ALL outstanding (not yet drawn) border shade sprites before
|
|
// doing more shading
|
|
pKillScreenSpiteIDs(pp, ID_BORDER_SHADE);
|
|
|
|
BorderShade(pp, TRUE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Redraw the whole screen
|
|
//
|
|
|
|
void SetBorder(PLAYERp pp, int value)
|
|
{
|
|
int Xdim, Ydim, ScreenSize;
|
|
|
|
if (pp != Player + myconnectindex)
|
|
return;
|
|
|
|
if (!BorderAdjust)
|
|
return;
|
|
|
|
if (value >= 0) // just refresh
|
|
gs.BorderNum = value;
|
|
|
|
if (gs.BorderNum < BORDER_NONE)
|
|
{
|
|
gs.BorderNum = BORDER_NONE;
|
|
//return;
|
|
}
|
|
|
|
if (gs.BorderNum > (int)SIZ(BorderInfoValues) - 1)
|
|
{
|
|
gs.BorderNum = SIZ(BorderInfoValues) - 1;
|
|
return;
|
|
}
|
|
if (xdim == 0) return; // game not set up yet.
|
|
|
|
BorderSetView(pp, &Xdim, &Ydim, &ScreenSize);
|
|
|
|
if (gs.BorderNum >= BORDER_BAR)
|
|
{
|
|
BorderRefresh(pp);
|
|
|
|
if (gs.BorderNum == BORDER_BAR)
|
|
SetConsoleDmost();
|
|
|
|
pSpawnFullScreenSprite(pp, STATUS_BAR, PRI_FRONT, 0, 200 - tilesiz[STATUS_BAR].y);
|
|
PlayerUpdatePanelInfo(Player + screenpeek);
|
|
}
|
|
|
|
SetFragBar(pp);
|
|
}
|
|
|
|
void
|
|
SetRedrawScreen(PLAYERp pp)
|
|
{
|
|
|
|
if (pp != Player + myconnectindex)
|
|
return;
|
|
|
|
if (!BorderAdjust)
|
|
return;
|
|
|
|
if (gs.BorderNum < BORDER_NONE)
|
|
gs.BorderNum = BORDER_NONE;
|
|
|
|
// Redraw the BORDER_TILE only if getting smaller
|
|
BorderInfo = BorderInfoValues[gs.BorderNum];
|
|
|
|
// test at redrawing the whole screen
|
|
RedrawScreen = TRUE;
|
|
}
|
|
|
|
END_SW_NS
|