mirror of
https://github.com/Rockbox/rockbox.git
synced 2025-10-24 07:27:39 -04:00
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@11738 a1c6a512-1295-4272-9138-f99709370657
1702 lines
43 KiB
C
1702 lines
43 KiB
C
/* Emacs style mode select -*- C++ -*-
|
|
*-----------------------------------------------------------------------------
|
|
*
|
|
*
|
|
* PrBoom a Doom port merged with LxDoom and LSDLDoom
|
|
* based on BOOM, a modified and improved DOOM engine
|
|
* Copyright (C) 1999 by
|
|
* id Software, Chi Hoang, Lee Killough, Jim Flynn, Rand Phares, Ty Halderman
|
|
* Copyright (C) 1999-2000 by
|
|
* Jess Haas, Nicolas Kalkhof, Colin Phipps, Florian Schulze
|
|
*
|
|
* This program 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., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*
|
|
* DESCRIPTION:
|
|
* the automap code
|
|
*
|
|
*-----------------------------------------------------------------------------
|
|
*/
|
|
|
|
#include "doomstat.h"
|
|
|
|
#include "st_stuff.h"
|
|
#include "r_main.h"
|
|
#include "p_setup.h"
|
|
#include "p_maputl.h"
|
|
#include "w_wad.h"
|
|
#include "v_video.h"
|
|
#include "p_spec.h"
|
|
#include "am_map.h"
|
|
#include "dstrings.h"
|
|
#include "d_deh.h" // Ty 03/27/98 - externalizations
|
|
#include "g_game.h"
|
|
#include "rockmacros.h"
|
|
|
|
|
|
//jff 1/7/98 default automap colors added
|
|
int mapcolor_back; // map background
|
|
int mapcolor_grid; // grid lines color
|
|
int mapcolor_wall; // normal 1s wall color
|
|
int mapcolor_fchg; // line at floor height change color
|
|
int mapcolor_cchg; // line at ceiling height change color
|
|
int mapcolor_clsd; // line at sector with floor=ceiling color
|
|
int mapcolor_rkey; // red key color
|
|
int mapcolor_bkey; // blue key color
|
|
int mapcolor_ykey; // yellow key color
|
|
int mapcolor_rdor; // red door color (diff from keys to allow option)
|
|
int mapcolor_bdor; // blue door color (of enabling one but not other )
|
|
int mapcolor_ydor; // yellow door color
|
|
int mapcolor_tele; // teleporter line color
|
|
int mapcolor_secr; // secret sector boundary color
|
|
int mapcolor_exit; // jff 4/23/98 add exit line color
|
|
int mapcolor_unsn; // computer map unseen line color
|
|
int mapcolor_flat; // line with no floor/ceiling changes
|
|
int mapcolor_sprt; // general sprite color
|
|
int mapcolor_item; // item sprite color
|
|
int mapcolor_frnd; // friendly sprite color
|
|
int mapcolor_hair; // crosshair color
|
|
int mapcolor_sngl; // single player arrow color
|
|
int mapcolor_plyr[4] = { 112, 88, 64, 176 }; // colors for player arrows in multiplayer
|
|
|
|
//jff 3/9/98 add option to not show secret sectors until entered
|
|
int map_secret_after;
|
|
//jff 4/3/98 add symbols for "no-color" for disable and "black color" for black
|
|
#define NC 0
|
|
#define BC 247
|
|
|
|
// drawing stuff
|
|
#define FB 0
|
|
|
|
// scale on entry
|
|
#define INITSCALEMTOF (.2*FRACUNIT)
|
|
// how much the automap moves window per tic in frame-buffer coordinates
|
|
// moves 140 pixels in 1 second
|
|
#define F_PANINC 4
|
|
// how much zoom-in per tic
|
|
// goes to 2x in 1 second
|
|
#define M_ZOOMIN ((int) (1.02*FRACUNIT))
|
|
// how much zoom-out per tic
|
|
// pulls out to 0.5x in 1 second
|
|
#define M_ZOOMOUT ((int) (FRACUNIT/1.02))
|
|
|
|
// translates between frame-buffer and map distances
|
|
#define FTOM(x) FixedMul(((x)<<16),scale_ftom)
|
|
#define MTOF(x) (FixedMul((x),scale_mtof)>>16)
|
|
// translates between frame-buffer and map coordinates
|
|
#define CXMTOF(x) (f_x + MTOF((x)-m_x))
|
|
#define CYMTOF(y) (f_y + (f_h - MTOF((y)-m_y)))
|
|
|
|
typedef struct
|
|
{
|
|
int x, y;
|
|
} fpoint_t;
|
|
|
|
typedef struct
|
|
{
|
|
fpoint_t a, b;
|
|
} fline_t;
|
|
|
|
typedef struct
|
|
{
|
|
mpoint_t a, b;
|
|
} mline_t;
|
|
|
|
typedef struct
|
|
{
|
|
fixed_t slp, islp;
|
|
} islope_t;
|
|
|
|
//
|
|
// The vector graphics for the automap.
|
|
// A line drawing of the player pointing right,
|
|
// starting from the middle.
|
|
//
|
|
#define R ((8*PLAYERRADIUS)/7)
|
|
mline_t player_arrow[] =
|
|
{
|
|
{ { -R+R/8, 0 }, { R, 0 } }, // -----
|
|
{ { R, 0 }, { R-R/2, R/4 } }, // ----->
|
|
{ { R, 0 }, { R-R/2, -R/4 } },
|
|
{ { -R+R/8, 0 }, { -R-R/8, R/4 } }, // >---->
|
|
{ { -R+R/8, 0 }, { -R-R/8, -R/4 } },
|
|
{ { -R+3*R/8, 0 }, { -R+R/8, R/4 } }, // >>--->
|
|
{ { -R+3*R/8, 0 }, { -R+R/8, -R/4 } }
|
|
};
|
|
#undef R
|
|
#define NUMPLYRLINES (sizeof(player_arrow)/sizeof(mline_t))
|
|
|
|
#define R ((8*PLAYERRADIUS)/7)
|
|
mline_t cheat_player_arrow[] =
|
|
{ // killough 3/22/98: He's alive, Jim :)
|
|
{ { -R+R/8, 0 }, { R, 0 } }, // -----
|
|
{ { R, 0 }, { R-R/2, R/4 } }, // ----->
|
|
{ { R, 0 }, { R-R/2, -R/4 } },
|
|
{ { -R+R/8, 0 }, { -R-R/8, R/4 } }, // >---->
|
|
{ { -R+R/8, 0 }, { -R-R/8, -R/4 } },
|
|
{ { -R+3*R/8, 0 }, { -R+R/8, R/4 } }, // >>--->
|
|
{ { -R+3*R/8, 0 }, { -R+R/8, -R/4 } },
|
|
{ { -R/10-R/6, R/4}, {-R/10-R/6, -R/4} }, // J
|
|
{ { -R/10-R/6, -R/4}, {-R/10-R/6-R/8, -R/4} },
|
|
{ { -R/10-R/6-R/8, -R/4}, {-R/10-R/6-R/8, -R/8} },
|
|
{ { -R/10, R/4}, {-R/10, -R/4}}, // F
|
|
{ { -R/10, R/4}, {-R/10+R/8, R/4}},
|
|
{ { -R/10+R/4, R/4}, {-R/10+R/4, -R/4}}, // F
|
|
{ { -R/10+R/4, R/4}, {-R/10+R/4+R/8, R/4}},
|
|
};
|
|
#undef R
|
|
#define NUMCHEATPLYRLINES (sizeof(cheat_player_arrow)/sizeof(mline_t))
|
|
|
|
#define R (FRACUNIT)
|
|
mline_t triangle_guy[] =
|
|
{
|
|
{ { (fixed_t)(-.867*R), (fixed_t)(-.5*R) }, { (fixed_t)( .867*R), (fixed_t)(-.5*R) } },
|
|
{ { (fixed_t)( .867*R), (fixed_t)(-.5*R) }, { (fixed_t)(0 ), (fixed_t)( R) } },
|
|
{ { (fixed_t)(0 ), (fixed_t)( R) }, { (fixed_t)(-.867*R), (fixed_t)(-.5*R) } }
|
|
};
|
|
#undef R
|
|
#define NUMTRIANGLEGUYLINES (sizeof(triangle_guy)/sizeof(mline_t))
|
|
|
|
//jff 1/5/98 new symbol for keys on automap
|
|
#define R (FRACUNIT)
|
|
mline_t cross_mark[] =
|
|
{
|
|
{ { -R, 0 }, { R, 0} },
|
|
{ { 0, -R }, { 0, R } },
|
|
};
|
|
#undef R
|
|
#define NUMCROSSMARKLINES (sizeof(cross_mark)/sizeof(mline_t))
|
|
//jff 1/5/98 end of new symbol
|
|
|
|
#define R (FRACUNIT)
|
|
mline_t thintriangle_guy[] =
|
|
{
|
|
{ { (fixed_t)(-.5*R), (fixed_t)(-.7*R) }, { (fixed_t)( R), (fixed_t)( 0) } },
|
|
{ { (fixed_t)( R), (fixed_t)( 0) }, { (fixed_t)(-.5*R), (fixed_t)( .7*R) } },
|
|
{ { (fixed_t)(-.5*R), (fixed_t)( .7*R) }, { (fixed_t)(-.5*R), (fixed_t)(-.7*R) } }
|
|
};
|
|
#undef R
|
|
#define NUMTHINTRIANGLEGUYLINES (sizeof(thintriangle_guy)/sizeof(mline_t))
|
|
|
|
int ddt_cheating = 0; // killough 2/7/98: make global, rename to ddt_*
|
|
|
|
static int leveljuststarted = 1; // kluge until AM_LevelInit() is called
|
|
|
|
enum automapmode_e automapmode; // Mode that the automap is in
|
|
|
|
// location of window on screen
|
|
static int f_x;
|
|
static int f_y;
|
|
|
|
// size of window on screen
|
|
static int f_w;
|
|
static int f_h;
|
|
|
|
static mpoint_t m_paninc; // how far the window pans each tic (map coords)
|
|
static fixed_t mtof_zoommul; // how far the window zooms each tic (map coords)
|
|
static fixed_t ftom_zoommul; // how far the window zooms each tic (fb coords)
|
|
|
|
static fixed_t m_x, m_y; // LL x,y window location on the map (map coords)
|
|
static fixed_t m_x2, m_y2; // UR x,y window location on the map (map coords)
|
|
|
|
//
|
|
// width/height of window on map (map coords)
|
|
//
|
|
static fixed_t m_w;
|
|
static fixed_t m_h;
|
|
|
|
// based on level size
|
|
static fixed_t min_x;
|
|
static fixed_t min_y;
|
|
static fixed_t max_x;
|
|
static fixed_t max_y;
|
|
|
|
static fixed_t max_w; // max_x-min_x,
|
|
static fixed_t max_h; // max_y-min_y
|
|
|
|
// based on player size
|
|
static fixed_t min_w;
|
|
static fixed_t min_h;
|
|
|
|
|
|
static fixed_t min_scale_mtof; // used to tell when to stop zooming out
|
|
static fixed_t max_scale_mtof; // used to tell when to stop zooming in
|
|
|
|
// old stuff for recovery later
|
|
static fixed_t old_m_w, old_m_h;
|
|
static fixed_t old_m_x, old_m_y;
|
|
|
|
// old location used by the Follower routine
|
|
static mpoint_t f_oldloc;
|
|
|
|
// used by MTOF to scale from map-to-frame-buffer coords
|
|
static fixed_t scale_mtof = (fixed_t)INITSCALEMTOF;
|
|
// used by FTOM to scale from frame-buffer-to-map coords (=1/scale_mtof)
|
|
static fixed_t scale_ftom;
|
|
|
|
static player_t *plr; // the player represented by an arrow
|
|
|
|
// killough 2/22/98: Remove limit on automap marks,
|
|
// and make variables external for use in savegames.
|
|
|
|
mpoint_t *markpoints = NULL; // where the points are
|
|
int markpointnum = 0; // next point to be assigned (also number of points now)
|
|
int markpointnum_max = 0; // killough 2/22/98
|
|
|
|
static boolean stopped = true;
|
|
|
|
//
|
|
// AM_getIslope()
|
|
//
|
|
// Calculates the slope and slope according to the x-axis of a line
|
|
// segment in map coordinates (with the upright y-axis n' all) so
|
|
// that it can be used with the brain-dead drawing stuff.
|
|
//
|
|
// Passed the line slope is desired for and an islope_t structure for return
|
|
// Returns nothing
|
|
//
|
|
void AM_getIslope
|
|
( mline_t* ml,
|
|
islope_t* is )
|
|
{
|
|
int dx, dy;
|
|
|
|
dy = ml->a.y - ml->b.y;
|
|
dx = ml->b.x - ml->a.x;
|
|
if (!dy)
|
|
is->islp = (dx<0?-INT_MAX:INT_MAX);
|
|
else
|
|
is->islp = FixedDiv(dx, dy);
|
|
if (!dx)
|
|
is->slp = (dy<0?-INT_MAX:INT_MAX);
|
|
else
|
|
is->slp = FixedDiv(dy, dx);
|
|
}
|
|
|
|
//
|
|
// AM_activateNewScale()
|
|
//
|
|
// Changes the map scale after zooming or translating
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_activateNewScale(void)
|
|
{
|
|
m_x += m_w/2;
|
|
m_y += m_h/2;
|
|
m_w = FTOM(f_w);
|
|
m_h = FTOM(f_h);
|
|
m_x -= m_w/2;
|
|
m_y -= m_h/2;
|
|
m_x2 = m_x + m_w;
|
|
m_y2 = m_y + m_h;
|
|
}
|
|
|
|
//
|
|
// AM_saveScaleAndLoc()
|
|
//
|
|
// Saves the current center and zoom
|
|
// Affects the variables that remember old scale and loc
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_saveScaleAndLoc(void)
|
|
{
|
|
old_m_x = m_x;
|
|
old_m_y = m_y;
|
|
old_m_w = m_w;
|
|
old_m_h = m_h;
|
|
}
|
|
|
|
//
|
|
// AM_restoreScaleAndLoc()
|
|
//
|
|
// restores the center and zoom from locally saved values
|
|
// Affects global variables for location and scale
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_restoreScaleAndLoc(void)
|
|
{
|
|
m_w = old_m_w;
|
|
m_h = old_m_h;
|
|
if (!(automapmode & am_follow))
|
|
{
|
|
m_x = old_m_x;
|
|
m_y = old_m_y;
|
|
}
|
|
else
|
|
{
|
|
m_x = plr->mo->x - m_w/2;
|
|
m_y = plr->mo->y - m_h/2;
|
|
}
|
|
m_x2 = m_x + m_w;
|
|
m_y2 = m_y + m_h;
|
|
|
|
// Change the scaling multipliers
|
|
scale_mtof = FixedDiv(f_w<<FRACBITS, m_w);
|
|
scale_ftom = FixedDiv(FRACUNIT, scale_mtof);
|
|
}
|
|
|
|
//
|
|
// AM_addMark()
|
|
//
|
|
// Adds a marker at the current location
|
|
// Affects global variables for marked points
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_addMark(void)
|
|
{
|
|
// killough 2/22/98:
|
|
// remove limit on automap marks
|
|
|
|
if (markpointnum >= markpointnum_max)
|
|
markpoints = realloc(markpoints,
|
|
(markpointnum_max = markpointnum_max ?
|
|
markpointnum_max*2 : 16) * sizeof(*markpoints));
|
|
|
|
markpoints[markpointnum].x = m_x + m_w/2;
|
|
markpoints[markpointnum].y = m_y + m_h/2;
|
|
markpointnum++;
|
|
}
|
|
|
|
//
|
|
// AM_findMinMaxBoundaries()
|
|
//
|
|
// Determines bounding box of all vertices,
|
|
// sets global variables controlling zoom range.
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_findMinMaxBoundaries(void)
|
|
{
|
|
int i;
|
|
fixed_t a;
|
|
fixed_t b;
|
|
|
|
min_x = min_y = INT_MAX;
|
|
max_x = max_y = -INT_MAX;
|
|
|
|
for (i=0;i<numvertexes;i++)
|
|
{
|
|
if (vertexes[i].x < min_x)
|
|
min_x = vertexes[i].x;
|
|
else if (vertexes[i].x > max_x)
|
|
max_x = vertexes[i].x;
|
|
|
|
if (vertexes[i].y < min_y)
|
|
min_y = vertexes[i].y;
|
|
else if (vertexes[i].y > max_y)
|
|
max_y = vertexes[i].y;
|
|
}
|
|
|
|
max_w = max_x - min_x;
|
|
max_h = max_y - min_y;
|
|
|
|
min_w = 2*PLAYERRADIUS; // const? never changed?
|
|
min_h = 2*PLAYERRADIUS;
|
|
|
|
a = FixedDiv(f_w<<FRACBITS, max_w);
|
|
b = FixedDiv(f_h<<FRACBITS, max_h);
|
|
|
|
min_scale_mtof = a < b ? a : b;
|
|
max_scale_mtof = FixedDiv(f_h<<FRACBITS, 2*PLAYERRADIUS);
|
|
}
|
|
|
|
//
|
|
// AM_changeWindowLoc()
|
|
//
|
|
// Moves the map window by the global variables m_paninc.x, m_paninc.y
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_changeWindowLoc(void)
|
|
{
|
|
if (m_paninc.x || m_paninc.y)
|
|
{
|
|
automapmode &= ~am_follow;
|
|
f_oldloc.x = INT_MAX;
|
|
}
|
|
|
|
m_x += m_paninc.x;
|
|
m_y += m_paninc.y;
|
|
|
|
if (m_x + m_w/2 > max_x)
|
|
m_x = max_x - m_w/2;
|
|
else if (m_x + m_w/2 < min_x)
|
|
m_x = min_x - m_w/2;
|
|
|
|
if (m_y + m_h/2 > max_y)
|
|
m_y = max_y - m_h/2;
|
|
else if (m_y + m_h/2 < min_y)
|
|
m_y = min_y - m_h/2;
|
|
|
|
m_x2 = m_x + m_w;
|
|
m_y2 = m_y + m_h;
|
|
}
|
|
|
|
|
|
//
|
|
// AM_initVariables()
|
|
//
|
|
// Initialize the variables for the automap
|
|
//
|
|
// Affects the automap global variables
|
|
// Status bar is notified that the automap has been entered
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_initVariables(void)
|
|
{
|
|
int pnum;
|
|
static event_t st_notify = { ev_keyup, AM_MSGENTERED, 0 , 0 };
|
|
|
|
automapmode |= am_active;
|
|
|
|
f_oldloc.x = INT_MAX;
|
|
|
|
m_paninc.x = m_paninc.y = 0;
|
|
ftom_zoommul = FRACUNIT;
|
|
mtof_zoommul = FRACUNIT;
|
|
|
|
m_w = FTOM(f_w);
|
|
m_h = FTOM(f_h);
|
|
|
|
// find player to center on initially
|
|
if (!playeringame[pnum = consoleplayer])
|
|
for (pnum=0;pnum<MAXPLAYERS;pnum++)
|
|
if (playeringame[pnum])
|
|
break;
|
|
|
|
plr = &players[pnum];
|
|
m_x = plr->mo->x - m_w/2;
|
|
m_y = plr->mo->y - m_h/2;
|
|
AM_changeWindowLoc();
|
|
|
|
// for saving & restoring
|
|
old_m_x = m_x;
|
|
old_m_y = m_y;
|
|
old_m_w = m_w;
|
|
old_m_h = m_h;
|
|
|
|
// inform the status bar of the change
|
|
ST_Responder(&st_notify);
|
|
}
|
|
|
|
//
|
|
// AM_loadPics()
|
|
//
|
|
void AM_loadPics(void)
|
|
{
|
|
// cph - mark numbers no longer needed cached
|
|
}
|
|
|
|
//
|
|
// AM_unloadPics()
|
|
//
|
|
void AM_unloadPics(void)
|
|
{
|
|
// cph - mark numbers no longer needed cached
|
|
}
|
|
|
|
//
|
|
// AM_clearMarks()
|
|
//
|
|
// Sets the number of marks to 0, thereby clearing them from the display
|
|
//
|
|
// Affects the global variable markpointnum
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_clearMarks(void)
|
|
{
|
|
markpointnum = 0;
|
|
}
|
|
|
|
//
|
|
// AM_LevelInit()
|
|
//
|
|
// Initialize the automap at the start of a new level
|
|
// should be called at the start of every level
|
|
//
|
|
// Passed nothing, returns nothing
|
|
// Affects automap's global variables
|
|
//
|
|
// CPhipps - get status bar height from status bar code
|
|
void AM_LevelInit(void)
|
|
{
|
|
leveljuststarted = 0;
|
|
|
|
f_x = f_y = 0;
|
|
f_w = SCREENWIDTH; // killough 2/7/98: get rid of finit_ vars
|
|
f_h = SCREENHEIGHT-ST_SCALED_HEIGHT;// to allow runtime setting of width/height
|
|
|
|
AM_findMinMaxBoundaries();
|
|
scale_mtof = FixedDiv(min_scale_mtof, (int) (0.7*FRACUNIT));
|
|
if (scale_mtof > max_scale_mtof)
|
|
scale_mtof = min_scale_mtof;
|
|
scale_ftom = FixedDiv(FRACUNIT, scale_mtof);
|
|
}
|
|
|
|
//
|
|
// AM_Stop()
|
|
//
|
|
// Cease automap operations, unload patches, notify status bar
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_Stop (void)
|
|
{
|
|
static event_t st_notify = { 0, ev_keyup, AM_MSGEXITED, 0 };
|
|
|
|
AM_unloadPics();
|
|
automapmode &= ~am_active;
|
|
ST_Responder(&st_notify);
|
|
stopped = true;
|
|
}
|
|
|
|
//
|
|
// AM_Start()
|
|
//
|
|
// Start up automap operations,
|
|
// if a new level, or game start, (re)initialize level variables
|
|
// init map variables
|
|
// load mark patches
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_Start()
|
|
{
|
|
static int lastlevel = -1, lastepisode = -1;
|
|
|
|
if (!stopped)
|
|
AM_Stop();
|
|
stopped = false;
|
|
if (lastlevel != gamemap || lastepisode != gameepisode)
|
|
{
|
|
AM_LevelInit();
|
|
lastlevel = gamemap;
|
|
lastepisode = gameepisode;
|
|
}
|
|
AM_initVariables();
|
|
AM_loadPics();
|
|
}
|
|
|
|
//
|
|
// AM_minOutWindowScale()
|
|
//
|
|
// Set the window scale to the maximum size
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_minOutWindowScale()
|
|
{
|
|
scale_mtof = min_scale_mtof;
|
|
scale_ftom = FixedDiv(FRACUNIT, scale_mtof);
|
|
AM_activateNewScale();
|
|
}
|
|
|
|
//
|
|
// AM_maxOutWindowScale(void)
|
|
//
|
|
// Set the window scale to the minimum size
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_maxOutWindowScale(void)
|
|
{
|
|
scale_mtof = max_scale_mtof;
|
|
scale_ftom = FixedDiv(FRACUNIT, scale_mtof);
|
|
AM_activateNewScale();
|
|
}
|
|
|
|
//
|
|
// AM_Responder()
|
|
//
|
|
// Handle events (user inputs) in automap mode
|
|
//
|
|
// Passed an input event, returns true if its handled
|
|
//
|
|
boolean AM_Responder
|
|
( event_t* ev )
|
|
{
|
|
int rc;
|
|
static int cheatstate=0;
|
|
static int bigstate=0;
|
|
static char buffer[20];
|
|
int ch; // phares
|
|
|
|
rc = false;
|
|
|
|
if (!(automapmode & am_active))
|
|
{
|
|
if (ev->type == ev_keydown && ev->data1 == key_map) // phares
|
|
{
|
|
AM_Start ();
|
|
rc = true;
|
|
}
|
|
}
|
|
else if (ev->type == ev_keydown)
|
|
{
|
|
rc = true;
|
|
ch = ev->data1; // phares
|
|
if (ch == key_map_right) // |
|
|
if (!(automapmode & am_follow)) // V
|
|
m_paninc.x = FTOM(F_PANINC);
|
|
else
|
|
rc = false;
|
|
else if (ch == key_map_left)
|
|
if (!(automapmode & am_follow))
|
|
m_paninc.x = -FTOM(F_PANINC);
|
|
else
|
|
rc = false;
|
|
else if (ch == key_map_up)
|
|
if (!(automapmode & am_follow))
|
|
m_paninc.y = FTOM(F_PANINC);
|
|
else
|
|
rc = false;
|
|
else if (ch == key_map_down)
|
|
if (!(automapmode & am_follow))
|
|
m_paninc.y = -FTOM(F_PANINC);
|
|
else
|
|
rc = false;
|
|
else if (ch == key_map_zoomout)
|
|
{
|
|
mtof_zoommul = M_ZOOMOUT;
|
|
ftom_zoommul = M_ZOOMIN;
|
|
}
|
|
else if (ch == key_map_zoomin)
|
|
{
|
|
mtof_zoommul = M_ZOOMIN;
|
|
ftom_zoommul = M_ZOOMOUT;
|
|
}
|
|
else if (ch == key_map)
|
|
{
|
|
bigstate = 0;
|
|
AM_Stop ();
|
|
}
|
|
else if (ch == key_map_gobig)
|
|
{
|
|
bigstate = !bigstate;
|
|
if (bigstate)
|
|
{
|
|
AM_saveScaleAndLoc();
|
|
AM_minOutWindowScale();
|
|
}
|
|
else
|
|
AM_restoreScaleAndLoc();
|
|
}
|
|
else if (ch == key_map_follow)
|
|
{
|
|
automapmode ^= am_follow; // CPhipps - put all automap mode stuff into one enum
|
|
f_oldloc.x = INT_MAX;
|
|
// Ty 03/27/98 - externalized
|
|
plr->message = (automapmode & am_follow) ? s_AMSTR_FOLLOWON : s_AMSTR_FOLLOWOFF;
|
|
}
|
|
else if (ch == key_map_grid)
|
|
{
|
|
automapmode ^= am_grid; // CPhipps
|
|
// Ty 03/27/98 - *not* externalized
|
|
plr->message = (automapmode & am_grid) ? s_AMSTR_GRIDON : s_AMSTR_GRIDOFF;
|
|
}
|
|
else if (ch == key_map_mark)
|
|
{
|
|
// Ty 03/27/98 - *not* externalized
|
|
snprintf(buffer, sizeof(buffer), "%s %d", s_AMSTR_MARKEDSPOT, markpointnum);
|
|
plr->message = buffer;
|
|
AM_addMark();
|
|
}
|
|
else if (ch == key_map_clear)
|
|
{
|
|
AM_clearMarks(); // Ty 03/27/98 - *not* externalized
|
|
plr->message = s_AMSTR_MARKSCLEARED; // ^
|
|
} // |
|
|
else if (ch == key_map_rotate) {
|
|
automapmode ^= am_rotate;
|
|
plr->message = (automapmode & am_rotate) ? s_AMSTR_ROTATEON : s_AMSTR_ROTATEOFF;
|
|
}
|
|
else if (ch == key_map_overlay) {
|
|
automapmode ^= am_overlay;
|
|
plr->message = (automapmode & am_overlay) ? s_AMSTR_OVERLAYON : s_AMSTR_OVERLAYOFF;
|
|
}
|
|
else // phares
|
|
{
|
|
cheatstate=0;
|
|
rc = false;
|
|
}
|
|
}
|
|
else if (ev->type == ev_keyup)
|
|
{
|
|
rc = false;
|
|
ch = ev->data1;
|
|
if (ch == key_map_right)
|
|
{
|
|
if (!(automapmode & am_follow))
|
|
m_paninc.x = 0;
|
|
}
|
|
else if (ch == key_map_left)
|
|
{
|
|
if (!(automapmode & am_follow))
|
|
m_paninc.x = 0;
|
|
}
|
|
else if (ch == key_map_up)
|
|
{
|
|
if (!(automapmode & am_follow))
|
|
m_paninc.y = 0;
|
|
}
|
|
else if (ch == key_map_down)
|
|
{
|
|
if (!(automapmode & am_follow))
|
|
m_paninc.y = 0;
|
|
}
|
|
else if ((ch == key_map_zoomout) || (ch == key_map_zoomin))
|
|
{
|
|
mtof_zoommul = FRACUNIT;
|
|
ftom_zoommul = FRACUNIT;
|
|
}
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
//
|
|
// AM_rotate()
|
|
//
|
|
// Rotation in 2D.
|
|
// Used to rotate player arrow line character.
|
|
//
|
|
// Passed the coordinates of a point, and an angle
|
|
// Returns the coordinates rotated by the angle
|
|
//
|
|
// CPhipps - made static & enhanced for automap rotation
|
|
|
|
static void AM_rotate(fixed_t* x, fixed_t* y, angle_t a, fixed_t xorig, fixed_t yorig)
|
|
{
|
|
fixed_t tmpx;
|
|
|
|
tmpx =
|
|
FixedMul(*x - xorig,finecosine[a>>ANGLETOFINESHIFT])
|
|
- FixedMul(*y - yorig,finesine[a>>ANGLETOFINESHIFT]);
|
|
|
|
*y = yorig +
|
|
FixedMul(*x - xorig,finesine[a>>ANGLETOFINESHIFT])
|
|
+ FixedMul(*y - yorig,finecosine[a>>ANGLETOFINESHIFT]);
|
|
|
|
*x = tmpx + xorig;
|
|
}
|
|
|
|
//
|
|
// AM_changeWindowScale()
|
|
//
|
|
// Automap zooming
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_changeWindowScale(void)
|
|
{
|
|
// Change the scaling multipliers
|
|
scale_mtof = FixedMul(scale_mtof, mtof_zoommul);
|
|
scale_ftom = FixedDiv(FRACUNIT, scale_mtof);
|
|
|
|
if (scale_mtof < min_scale_mtof)
|
|
AM_minOutWindowScale();
|
|
else if (scale_mtof > max_scale_mtof)
|
|
AM_maxOutWindowScale();
|
|
else
|
|
AM_activateNewScale();
|
|
}
|
|
|
|
//
|
|
// AM_doFollowPlayer()
|
|
//
|
|
// Turn on follow mode - the map scrolls opposite to player motion
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_doFollowPlayer(void)
|
|
{
|
|
if (f_oldloc.x != plr->mo->x || f_oldloc.y != plr->mo->y)
|
|
{
|
|
m_x = FTOM(MTOF(plr->mo->x)) - m_w/2;
|
|
m_y = FTOM(MTOF(plr->mo->y)) - m_h/2;
|
|
m_x2 = m_x + m_w;
|
|
m_y2 = m_y + m_h;
|
|
f_oldloc.x = plr->mo->x;
|
|
f_oldloc.y = plr->mo->y;
|
|
}
|
|
}
|
|
|
|
//
|
|
// AM_Ticker()
|
|
//
|
|
// Updates on gametic - enter follow mode, zoom, or change map location
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_Ticker (void)
|
|
{
|
|
if (!(automapmode & am_active))
|
|
return;
|
|
|
|
if (automapmode & am_follow)
|
|
AM_doFollowPlayer();
|
|
|
|
// Change the zoom if necessary
|
|
if (ftom_zoommul != FRACUNIT)
|
|
AM_changeWindowScale();
|
|
|
|
// Change x,y location
|
|
if (m_paninc.x || m_paninc.y)
|
|
AM_changeWindowLoc();
|
|
}
|
|
|
|
//
|
|
// AM_clipMline()
|
|
//
|
|
// Automap clipping of lines.
|
|
//
|
|
// Based on Cohen-Sutherland clipping algorithm but with a slightly
|
|
// faster reject and precalculated slopes. If the speed is needed,
|
|
// use a hash algorithm to handle the common cases.
|
|
//
|
|
// Passed the line's coordinates on map and in the frame buffer performs
|
|
// clipping on them in the lines frame coordinates.
|
|
// Returns true if any part of line was not clipped
|
|
//
|
|
boolean AM_clipMline
|
|
( mline_t* ml,
|
|
fline_t* fl )
|
|
{
|
|
enum
|
|
{
|
|
LEFT =1,
|
|
RIGHT =2,
|
|
BOTTOM =4,
|
|
TOP =8
|
|
};
|
|
|
|
register int outcode1 = 0;
|
|
register int outcode2 = 0;
|
|
register int outside;
|
|
|
|
fpoint_t tmp = {0,0};
|
|
int dx;
|
|
int dy;
|
|
|
|
|
|
#define DOOUTCODE(oc, mx, my) \
|
|
(oc) = 0; \
|
|
if ((my) < 0) (oc) |= TOP; \
|
|
else if ((my) >= f_h) (oc) |= BOTTOM; \
|
|
if ((mx) < 0) (oc) |= LEFT; \
|
|
else if ((mx) >= f_w) (oc) |= RIGHT;
|
|
|
|
|
|
// do trivial rejects and outcodes
|
|
if (ml->a.y > m_y2)
|
|
outcode1 = TOP;
|
|
else if (ml->a.y < m_y)
|
|
outcode1 = BOTTOM;
|
|
|
|
if (ml->b.y > m_y2)
|
|
outcode2 = TOP;
|
|
else if (ml->b.y < m_y)
|
|
outcode2 = BOTTOM;
|
|
|
|
if (outcode1 & outcode2)
|
|
return false; // trivially outside
|
|
|
|
if (ml->a.x < m_x)
|
|
outcode1 |= LEFT;
|
|
else if (ml->a.x > m_x2)
|
|
outcode1 |= RIGHT;
|
|
|
|
if (ml->b.x < m_x)
|
|
outcode2 |= LEFT;
|
|
else if (ml->b.x > m_x2)
|
|
outcode2 |= RIGHT;
|
|
|
|
if (outcode1 & outcode2)
|
|
return false; // trivially outside
|
|
|
|
// transform to frame-buffer coordinates.
|
|
fl->a.x = CXMTOF(ml->a.x);
|
|
fl->a.y = CYMTOF(ml->a.y);
|
|
fl->b.x = CXMTOF(ml->b.x);
|
|
fl->b.y = CYMTOF(ml->b.y);
|
|
|
|
DOOUTCODE(outcode1, fl->a.x, fl->a.y);
|
|
DOOUTCODE(outcode2, fl->b.x, fl->b.y);
|
|
|
|
if (outcode1 & outcode2)
|
|
return false;
|
|
|
|
while (outcode1 | outcode2)
|
|
{
|
|
// may be partially inside box
|
|
// find an outside point
|
|
if (outcode1)
|
|
outside = outcode1;
|
|
else
|
|
outside = outcode2;
|
|
|
|
// clip to each side
|
|
if (outside & TOP)
|
|
{
|
|
dy = fl->a.y - fl->b.y;
|
|
dx = fl->b.x - fl->a.x;
|
|
tmp.x = fl->a.x + (dx*(fl->a.y))/dy;
|
|
tmp.y = 0;
|
|
}
|
|
else if (outside & BOTTOM)
|
|
{
|
|
dy = fl->a.y - fl->b.y;
|
|
dx = fl->b.x - fl->a.x;
|
|
tmp.x = fl->a.x + (dx*(fl->a.y-f_h))/dy;
|
|
tmp.y = f_h-1;
|
|
}
|
|
else if (outside & RIGHT)
|
|
{
|
|
dy = fl->b.y - fl->a.y;
|
|
dx = fl->b.x - fl->a.x;
|
|
tmp.y = fl->a.y + (dy*(f_w-1 - fl->a.x))/dx;
|
|
tmp.x = f_w-1;
|
|
}
|
|
else if (outside & LEFT)
|
|
{
|
|
dy = fl->b.y - fl->a.y;
|
|
dx = fl->b.x - fl->a.x;
|
|
tmp.y = fl->a.y + (dy*(-fl->a.x))/dx;
|
|
tmp.x = 0;
|
|
}
|
|
|
|
if (outside == outcode1)
|
|
{
|
|
fl->a = tmp;
|
|
DOOUTCODE(outcode1, fl->a.x, fl->a.y);
|
|
}
|
|
else
|
|
{
|
|
fl->b = tmp;
|
|
DOOUTCODE(outcode2, fl->b.x, fl->b.y);
|
|
}
|
|
|
|
if (outcode1 & outcode2)
|
|
return false; // trivially outside
|
|
}
|
|
|
|
return true;
|
|
}
|
|
#undef DOOUTCODE
|
|
|
|
//
|
|
// AM_drawFline()
|
|
//
|
|
// Draw a line in the frame buffer.
|
|
// Classic Bresenham w/ whatever optimizations needed for speed
|
|
//
|
|
// Passed the frame coordinates of line, and the color to be drawn
|
|
// Returns nothing
|
|
//
|
|
|
|
void AM_drawFline
|
|
( fline_t* fl,
|
|
int color )
|
|
{
|
|
register int x;
|
|
register int y;
|
|
register int dx;
|
|
register int dy;
|
|
register int sx;
|
|
register int sy;
|
|
register int ax;
|
|
register int ay;
|
|
register int d;
|
|
|
|
#ifdef RANGECHECK // killough 2/22/98
|
|
static int fuck = 0;
|
|
|
|
// For debugging only
|
|
if
|
|
(
|
|
fl->a.x < 0 || fl->a.x >= f_w
|
|
|| fl->a.y < 0 || fl->a.y >= f_h
|
|
|| fl->b.x < 0 || fl->b.x >= f_w
|
|
|| fl->b.y < 0 || fl->b.y >= f_h
|
|
)
|
|
{
|
|
//jff 8/3/98 use logical output routine
|
|
printf("fuck %d \r", fuck++);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
#define PUTDOT(xx,yy,cc) V_PlotPixel(FB,xx,yy,(byte)cc)
|
|
|
|
dx = fl->b.x - fl->a.x;
|
|
ax = 2 * (dx<0 ? -dx : dx);
|
|
sx = dx<0 ? -1 : 1;
|
|
|
|
dy = fl->b.y - fl->a.y;
|
|
ay = 2 * (dy<0 ? -dy : dy);
|
|
sy = dy<0 ? -1 : 1;
|
|
|
|
x = fl->a.x;
|
|
y = fl->a.y;
|
|
|
|
if (ax > ay)
|
|
{
|
|
d = ay - ax/2;
|
|
while (1)
|
|
{
|
|
PUTDOT(x,y,color);
|
|
if (x == fl->b.x) return;
|
|
if (d>=0)
|
|
{
|
|
y += sy;
|
|
d -= ax;
|
|
}
|
|
x += sx;
|
|
d += ay;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
d = ax - ay/2;
|
|
while (1)
|
|
{
|
|
PUTDOT(x, y, color);
|
|
if (y == fl->b.y) return;
|
|
if (d >= 0)
|
|
{
|
|
x += sx;
|
|
d -= ay;
|
|
}
|
|
y += sy;
|
|
d += ax;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// AM_drawMline()
|
|
//
|
|
// Clip lines, draw visible parts of lines.
|
|
//
|
|
// Passed the map coordinates of the line, and the color to draw it
|
|
// Color -1 is special and prevents drawing. Color 247 is special and
|
|
// is translated to black, allowing Color 0 to represent feature disable
|
|
// in the defaults file.
|
|
// Returns nothing.
|
|
//
|
|
void AM_drawMline
|
|
( mline_t* ml,
|
|
int color )
|
|
{
|
|
static fline_t fl;
|
|
|
|
if (color==-1) // jff 4/3/98 allow not drawing any sort of line
|
|
return; // by setting its color to -1
|
|
if (color==247) // jff 4/3/98 if color is 247 (xparent), use black
|
|
color=0;
|
|
|
|
if (AM_clipMline(ml, &fl))
|
|
AM_drawFline(&fl, color); // draws it on frame buffer using fb coords
|
|
}
|
|
|
|
//
|
|
// AM_drawGrid()
|
|
//
|
|
// Draws blockmap aligned grid lines.
|
|
//
|
|
// Passed the color to draw the grid lines
|
|
// Returns nothing
|
|
//
|
|
void AM_drawGrid(int color)
|
|
{
|
|
fixed_t x, y;
|
|
fixed_t start, end;
|
|
mline_t ml;
|
|
|
|
// Figure out start of vertical gridlines
|
|
start = m_x;
|
|
if ((start-bmaporgx)%(MAPBLOCKUNITS<<FRACBITS))
|
|
start += (MAPBLOCKUNITS<<FRACBITS)
|
|
- ((start-bmaporgx)%(MAPBLOCKUNITS<<FRACBITS));
|
|
end = m_x + m_w;
|
|
|
|
// draw vertical gridlines
|
|
ml.a.y = m_y;
|
|
ml.b.y = m_y+m_h;
|
|
for (x=start; x<end; x+=(MAPBLOCKUNITS<<FRACBITS))
|
|
{
|
|
ml.a.x = x;
|
|
ml.b.x = x;
|
|
AM_drawMline(&ml, color);
|
|
}
|
|
|
|
// Figure out start of horizontal gridlines
|
|
start = m_y;
|
|
if ((start-bmaporgy)%(MAPBLOCKUNITS<<FRACBITS))
|
|
start += (MAPBLOCKUNITS<<FRACBITS)
|
|
- ((start-bmaporgy)%(MAPBLOCKUNITS<<FRACBITS));
|
|
end = m_y + m_h;
|
|
|
|
// draw horizontal gridlines
|
|
ml.a.x = m_x;
|
|
ml.b.x = m_x + m_w;
|
|
for (y=start; y<end; y+=(MAPBLOCKUNITS<<FRACBITS))
|
|
{
|
|
ml.a.y = y;
|
|
ml.b.y = y;
|
|
AM_drawMline(&ml, color);
|
|
}
|
|
}
|
|
|
|
//
|
|
// AM_DoorColor()
|
|
//
|
|
// Returns the 'color' or key needed for a door linedef type
|
|
//
|
|
// Passed the type of linedef, returns:
|
|
// -1 if not a keyed door
|
|
// 0 if a red key required
|
|
// 1 if a blue key required
|
|
// 2 if a yellow key required
|
|
// 3 if a multiple keys required
|
|
//
|
|
// jff 4/3/98 add routine to get color of generalized keyed door
|
|
//
|
|
int AM_DoorColor(int type)
|
|
{
|
|
if (GenLockedBase <= type && type< GenDoorBase)
|
|
{
|
|
type -= GenLockedBase;
|
|
type = (type & LockedKey) >> LockedKeyShift;
|
|
if (!type || type==7)
|
|
return 3; //any or all keys
|
|
else return (type-1)%3;
|
|
}
|
|
switch (type) // closed keyed door
|
|
{
|
|
case 26: case 32: case 99: case 133:
|
|
/*bluekey*/
|
|
return 1;
|
|
case 27: case 34: case 136: case 137:
|
|
/*yellowkey*/
|
|
return 2;
|
|
case 28: case 33: case 134: case 135:
|
|
/*redkey*/
|
|
return 0;
|
|
default:
|
|
return -1; //not a keyed door
|
|
}
|
|
return -1; //not a keyed door
|
|
}
|
|
|
|
//
|
|
// Determines visible lines, draws them.
|
|
// This is LineDef based, not LineSeg based.
|
|
//
|
|
// jff 1/5/98 many changes in this routine
|
|
// backward compatibility not needed, so just changes, no ifs
|
|
// addition of clauses for:
|
|
// doors opening, keyed door id, secret sectors,
|
|
// teleports, exit lines, key things
|
|
// ability to suppress any of added features or lines with no height changes
|
|
//
|
|
// support for gamma correction in automap abandoned
|
|
//
|
|
// jff 4/3/98 changed mapcolor_xxxx=0 as control to disable feature
|
|
// jff 4/3/98 changed mapcolor_xxxx=-1 to disable drawing line completely
|
|
//
|
|
void AM_drawWalls(void)
|
|
{
|
|
int i;
|
|
static mline_t l;
|
|
|
|
// draw the unclipped visible portions of all lines
|
|
for (i=0;i<numlines;i++)
|
|
{
|
|
l.a.x = lines[i].v1->x;
|
|
l.a.y = lines[i].v1->y;
|
|
l.b.x = lines[i].v2->x;
|
|
l.b.y = lines[i].v2->y;
|
|
|
|
if (automapmode & am_rotate) {
|
|
AM_rotate(&l.a.x, &l.a.y, ANG90-plr->mo->angle, plr->mo->x, plr->mo->y);
|
|
AM_rotate(&l.b.x, &l.b.y, ANG90-plr->mo->angle, plr->mo->x, plr->mo->y);
|
|
}
|
|
|
|
// if line has been seen or IDDT has been used
|
|
if (ddt_cheating || (lines[i].flags & ML_MAPPED))
|
|
{
|
|
if ((lines[i].flags & ML_DONTDRAW) && !ddt_cheating)
|
|
continue;
|
|
{
|
|
/* cph - show keyed doors and lines */
|
|
int amd;
|
|
if ((mapcolor_bdor || mapcolor_ydor || mapcolor_rdor) &&
|
|
!(lines[i].flags & ML_SECRET) && /* non-secret */
|
|
(amd = AM_DoorColor(lines[i].special)) != -1
|
|
)
|
|
{
|
|
{
|
|
switch (amd) /* closed keyed door */
|
|
{
|
|
case 1:
|
|
/*bluekey*/
|
|
AM_drawMline(&l,
|
|
mapcolor_bdor? mapcolor_bdor : mapcolor_cchg);
|
|
continue;
|
|
case 2:
|
|
/*yellowkey*/
|
|
AM_drawMline(&l,
|
|
mapcolor_ydor? mapcolor_ydor : mapcolor_cchg);
|
|
continue;
|
|
case 0:
|
|
/*redkey*/
|
|
AM_drawMline(&l,
|
|
mapcolor_rdor? mapcolor_rdor : mapcolor_cchg);
|
|
continue;
|
|
case 3:
|
|
/*any or all*/
|
|
AM_drawMline(&l,
|
|
mapcolor_clsd? mapcolor_clsd : mapcolor_cchg);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if /* jff 4/23/98 add exit lines to automap */
|
|
(
|
|
mapcolor_exit &&
|
|
(
|
|
lines[i].special==11 ||
|
|
lines[i].special==52 ||
|
|
lines[i].special==197 ||
|
|
lines[i].special==51 ||
|
|
lines[i].special==124 ||
|
|
lines[i].special==198
|
|
)
|
|
) {
|
|
AM_drawMline(&l, mapcolor_exit); /* exit line */
|
|
continue;
|
|
}
|
|
|
|
if (!lines[i].backsector)
|
|
{
|
|
// jff 1/10/98 add new color for 1S secret sector boundary
|
|
if (mapcolor_secr && //jff 4/3/98 0 is disable
|
|
(
|
|
(
|
|
map_secret_after &&
|
|
P_WasSecret(lines[i].frontsector) &&
|
|
!P_IsSecret(lines[i].frontsector)
|
|
)
|
|
||
|
|
(
|
|
!map_secret_after &&
|
|
P_WasSecret(lines[i].frontsector)
|
|
)
|
|
)
|
|
)
|
|
AM_drawMline(&l, mapcolor_secr); // line bounding secret sector
|
|
else //jff 2/16/98 fixed bug
|
|
AM_drawMline(&l, mapcolor_wall); // special was cleared
|
|
}
|
|
else /* now for 2S lines */
|
|
{
|
|
// jff 1/10/98 add color change for all teleporter types
|
|
if
|
|
(
|
|
mapcolor_tele && !(lines[i].flags & ML_SECRET) &&
|
|
(lines[i].special == 39 || lines[i].special == 97 ||
|
|
lines[i].special == 125 || lines[i].special == 126)
|
|
)
|
|
{ // teleporters
|
|
AM_drawMline(&l, mapcolor_tele);
|
|
}
|
|
else if (lines[i].flags & ML_SECRET) // secret door
|
|
{
|
|
AM_drawMline(&l, mapcolor_wall); // wall color
|
|
}
|
|
else if
|
|
(
|
|
mapcolor_clsd &&
|
|
!(lines[i].flags & ML_SECRET) && // non-secret closed door
|
|
((lines[i].backsector->floorheight==lines[i].backsector->ceilingheight) ||
|
|
(lines[i].frontsector->floorheight==lines[i].frontsector->ceilingheight))
|
|
)
|
|
{
|
|
AM_drawMline(&l, mapcolor_clsd); // non-secret closed door
|
|
} //jff 1/6/98 show secret sector 2S lines
|
|
else if
|
|
(
|
|
mapcolor_secr && //jff 2/16/98 fixed bug
|
|
( // special was cleared after getting it
|
|
(map_secret_after &&
|
|
(
|
|
(P_WasSecret(lines[i].frontsector)
|
|
&& !P_IsSecret(lines[i].frontsector)) ||
|
|
(P_WasSecret(lines[i].backsector)
|
|
&& !P_IsSecret(lines[i].backsector))
|
|
)
|
|
)
|
|
|| //jff 3/9/98 add logic to not show secret til after entered
|
|
( // if map_secret_after is true
|
|
!map_secret_after &&
|
|
(P_WasSecret(lines[i].frontsector) ||
|
|
P_WasSecret(lines[i].backsector))
|
|
)
|
|
)
|
|
)
|
|
{
|
|
AM_drawMline(&l, mapcolor_secr); // line bounding secret sector
|
|
} //jff 1/6/98 end secret sector line change
|
|
else if (lines[i].backsector->floorheight !=
|
|
lines[i].frontsector->floorheight)
|
|
{
|
|
AM_drawMline(&l, mapcolor_fchg); // floor level change
|
|
}
|
|
else if (lines[i].backsector->ceilingheight !=
|
|
lines[i].frontsector->ceilingheight)
|
|
{
|
|
AM_drawMline(&l, mapcolor_cchg); // ceiling level change
|
|
}
|
|
else if (mapcolor_flat && ddt_cheating)
|
|
{
|
|
AM_drawMline(&l, mapcolor_flat); //2S lines that appear only in IDDT
|
|
}
|
|
}
|
|
} // now draw the lines only visible because the player has computermap
|
|
else if (plr->powers[pw_allmap]) // computermap visible lines
|
|
{
|
|
if (!(lines[i].flags & ML_DONTDRAW)) // invisible flag lines do not show
|
|
{
|
|
if
|
|
(
|
|
mapcolor_flat
|
|
||
|
|
!lines[i].backsector
|
|
||
|
|
lines[i].backsector->floorheight
|
|
!= lines[i].frontsector->floorheight
|
|
||
|
|
lines[i].backsector->ceilingheight
|
|
!= lines[i].frontsector->ceilingheight
|
|
)
|
|
AM_drawMline(&l, mapcolor_unsn);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// AM_drawLineCharacter()
|
|
//
|
|
// Draws a vector graphic according to numerous parameters
|
|
//
|
|
// Passed the structure defining the vector graphic shape, the number
|
|
// of vectors in it, the scale to draw it at, the angle to draw it at,
|
|
// the color to draw it with, and the map coordinates to draw it at.
|
|
// Returns nothing
|
|
//
|
|
void AM_drawLineCharacter
|
|
( mline_t* lineguy,
|
|
int lineguylines,
|
|
fixed_t scale,
|
|
angle_t angle,
|
|
int color,
|
|
fixed_t x,
|
|
fixed_t y )
|
|
{
|
|
int i;
|
|
mline_t l;
|
|
|
|
if (automapmode & am_rotate) angle -= plr->mo->angle - ANG90; // cph
|
|
|
|
for (i=0;i<lineguylines;i++)
|
|
{
|
|
l.a.x = lineguy[i].a.x;
|
|
l.a.y = lineguy[i].a.y;
|
|
|
|
if (scale)
|
|
{
|
|
l.a.x = FixedMul(scale, l.a.x);
|
|
l.a.y = FixedMul(scale, l.a.y);
|
|
}
|
|
|
|
if (angle)
|
|
AM_rotate(&l.a.x, &l.a.y, angle, 0, 0);
|
|
|
|
l.a.x += x;
|
|
l.a.y += y;
|
|
|
|
l.b.x = lineguy[i].b.x;
|
|
l.b.y = lineguy[i].b.y;
|
|
|
|
if (scale)
|
|
{
|
|
l.b.x = FixedMul(scale, l.b.x);
|
|
l.b.y = FixedMul(scale, l.b.y);
|
|
}
|
|
|
|
if (angle)
|
|
AM_rotate(&l.b.x, &l.b.y, angle, 0, 0);
|
|
|
|
l.b.x += x;
|
|
l.b.y += y;
|
|
|
|
AM_drawMline(&l, color);
|
|
}
|
|
}
|
|
|
|
//
|
|
// AM_drawPlayers()
|
|
//
|
|
// Draws the player arrow in single player,
|
|
// or all the player arrows in a netgame.
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_drawPlayers(void)
|
|
{
|
|
int i;
|
|
|
|
if (!netgame)
|
|
{
|
|
if (ddt_cheating)
|
|
AM_drawLineCharacter
|
|
(
|
|
cheat_player_arrow,
|
|
NUMCHEATPLYRLINES,
|
|
0,
|
|
plr->mo->angle,
|
|
mapcolor_sngl, //jff color
|
|
plr->mo->x,
|
|
plr->mo->y
|
|
);
|
|
else
|
|
AM_drawLineCharacter
|
|
(
|
|
player_arrow,
|
|
NUMPLYRLINES,
|
|
0,
|
|
plr->mo->angle,
|
|
mapcolor_sngl, //jff color
|
|
plr->mo->x,
|
|
plr->mo->y);
|
|
return;
|
|
}
|
|
|
|
for (i=0;i<MAXPLAYERS;i++) {
|
|
player_t* p = &players[i];
|
|
|
|
if ( (deathmatch && !singledemo) && p != plr)
|
|
continue;
|
|
|
|
if (playeringame[i]) {
|
|
fixed_t x = p->mo->x, y = p->mo->y;
|
|
if (automapmode & am_rotate)
|
|
AM_rotate(&x, &y, ANG90-plr->mo->angle, plr->mo->x, plr->mo->y);
|
|
|
|
AM_drawLineCharacter (player_arrow, NUMPLYRLINES, 0, p->mo->angle,
|
|
p->powers[pw_invisibility] ? 246 /* *close* to black */
|
|
: mapcolor_plyr[i], //jff 1/6/98 use default color
|
|
x, y);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// AM_drawThings()
|
|
//
|
|
// Draws the things on the automap in double IDDT cheat mode
|
|
//
|
|
// Passed colors and colorrange, no longer used
|
|
// Returns nothing
|
|
//
|
|
void AM_drawThings
|
|
( int colors,
|
|
int colorrange)
|
|
{
|
|
(void)colors;
|
|
(void)colorrange;
|
|
int i;
|
|
mobj_t* t;
|
|
|
|
// for all sectors
|
|
for (i=0;i<numsectors;i++)
|
|
{
|
|
t = sectors[i].thinglist;
|
|
while (t) // for all things in that sector
|
|
{
|
|
fixed_t x = t->x, y = t->y;
|
|
|
|
if (automapmode & am_rotate)
|
|
AM_rotate(&x, &y, ANG90-plr->mo->angle, plr->mo->x, plr->mo->y);
|
|
|
|
//jff 1/5/98 case over doomednum of thing being drawn
|
|
if (mapcolor_rkey || mapcolor_ykey || mapcolor_bkey)
|
|
{
|
|
switch(t->info->doomednum)
|
|
{
|
|
//jff 1/5/98 treat keys special
|
|
case 38: case 13: //jff red key
|
|
AM_drawLineCharacter
|
|
(
|
|
cross_mark,
|
|
NUMCROSSMARKLINES,
|
|
16<<FRACBITS,
|
|
t->angle,
|
|
mapcolor_rkey!=-1? mapcolor_rkey : mapcolor_sprt,
|
|
x, y
|
|
);
|
|
t = t->snext;
|
|
continue;
|
|
case 39: case 6: //jff yellow key
|
|
AM_drawLineCharacter
|
|
(
|
|
cross_mark,
|
|
NUMCROSSMARKLINES,
|
|
16<<FRACBITS,
|
|
t->angle,
|
|
mapcolor_ykey!=-1? mapcolor_ykey : mapcolor_sprt,
|
|
x, y
|
|
);
|
|
t = t->snext;
|
|
continue;
|
|
case 40: case 5: //jff blue key
|
|
AM_drawLineCharacter
|
|
(
|
|
cross_mark,
|
|
NUMCROSSMARKLINES,
|
|
16<<FRACBITS,
|
|
t->angle,
|
|
mapcolor_bkey!=-1? mapcolor_bkey : mapcolor_sprt,
|
|
x, y
|
|
);
|
|
t = t->snext;
|
|
continue;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
//jff 1/5/98 end added code for keys
|
|
//jff previously entire code
|
|
AM_drawLineCharacter
|
|
(
|
|
thintriangle_guy,
|
|
NUMTHINTRIANGLEGUYLINES,
|
|
16<<FRACBITS,
|
|
t->angle,
|
|
t->flags & MF_FRIEND && !t->player ? mapcolor_frnd :
|
|
/* bbm 2/28/03 Show countable items in yellow. */
|
|
t->flags & MF_COUNTITEM ? mapcolor_item : mapcolor_sprt,
|
|
x, y
|
|
);
|
|
t = t->snext;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// AM_drawMarks()
|
|
//
|
|
// Draw the marked locations on the automap
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
// killough 2/22/98:
|
|
// Rewrote AM_drawMarks(). Removed limit on marks.
|
|
//
|
|
void AM_drawMarks(void)
|
|
{
|
|
int i;
|
|
for (i=0;i<markpointnum;i++) // killough 2/22/98: remove automap mark limit
|
|
if (markpoints[i].x != -1)
|
|
{
|
|
int w = 5;
|
|
int h = 6;
|
|
int fx = markpoints[i].x;
|
|
int fy = markpoints[i].y;
|
|
int j = i;
|
|
|
|
if (automapmode & am_rotate)
|
|
AM_rotate(&fx, &fy, ANG90-plr->mo->angle, plr->mo->x, plr->mo->y);
|
|
|
|
fx = CXMTOF(fx); fy = CYMTOF(fy);
|
|
|
|
do
|
|
{
|
|
int d = j % 10;
|
|
if (d==1) // killough 2/22/98: less spacing for '1'
|
|
fx++;
|
|
|
|
if (fx >= f_x && fx < f_w - w && fy >= f_y && fy < f_h - h) {
|
|
// cph - construct patch name and draw marker
|
|
char namebuf[] = { 'A', 'M', 'M', 'N', 'U', 'M', '0'+d, 0 };
|
|
|
|
V_DrawNamePatch(fx, fy, FB, namebuf, CR_DEFAULT, VPT_NONE);
|
|
}
|
|
fx -= w-1; // killough 2/22/98: 1 space backwards
|
|
j /= 10;
|
|
}
|
|
while (j>0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// AM_drawCrosshair()
|
|
//
|
|
// Draw the single point crosshair representing map center
|
|
//
|
|
// Passed the color to draw the pixel with
|
|
// Returns nothing
|
|
//
|
|
// CPhipps - made static inline, and use the general pixel plotter function
|
|
|
|
inline static void AM_drawCrosshair(int color)
|
|
{
|
|
// single point for now
|
|
V_PlotPixel(FB, f_w/2, f_h/2, (byte)color);
|
|
}
|
|
|
|
//
|
|
// AM_Drawer()
|
|
//
|
|
// Draws the entire automap
|
|
//
|
|
// Passed nothing, returns nothing
|
|
//
|
|
void AM_Drawer (void)
|
|
{
|
|
// CPhipps - all automap modes put into one enum
|
|
if (!(automapmode & am_active)) return;
|
|
|
|
if (!(automapmode & am_overlay)) // cph - If not overlay mode, clear background for the automap
|
|
V_FillRect(FB, f_x, f_y, f_w, f_h, (byte)mapcolor_back); //jff 1/5/98 background default color
|
|
if (automapmode & am_grid)
|
|
AM_drawGrid(mapcolor_grid); //jff 1/7/98 grid default color
|
|
AM_drawWalls();
|
|
AM_drawPlayers();
|
|
if (ddt_cheating==2)
|
|
AM_drawThings(mapcolor_sprt, 0); //jff 1/5/98 default double IDDT sprite
|
|
AM_drawCrosshair(mapcolor_hair); //jff 1/7/98 default crosshair color
|
|
|
|
AM_drawMarks();
|
|
|
|
V_MarkRect(f_x, f_y, f_w, f_h);
|
|
}
|