2015-05-19 21:54:34 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
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
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
2019-10-09 16:09:05 +00:00
|
|
|
#include "ns.h"
|
|
|
|
|
2015-05-19 21:54:34 +00:00
|
|
|
#include "build.h"
|
|
|
|
|
|
|
|
#include "names2.h"
|
|
|
|
#include "panel.h"
|
|
|
|
#include "game.h"
|
|
|
|
#include "tags.h"
|
|
|
|
#include "player.h"
|
|
|
|
#include "mclip.h"
|
|
|
|
|
2019-10-09 16:09:05 +00:00
|
|
|
BEGIN_SW_NS
|
|
|
|
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2022-09-03 20:20:12 +00:00
|
|
|
Collision MultiClipMove(PLAYER* pp, double zz, double floordist)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
|
|
|
int i;
|
2022-09-03 20:12:32 +00:00
|
|
|
DVector3 opos[MAX_CLIPBOX], pos[MAX_CLIPBOX];
|
2021-12-31 14:50:44 +00:00
|
|
|
SECTOR_OBJECT* sop = pp->sop;
|
2015-05-19 21:54:34 +00:00
|
|
|
short min_ndx = 0;
|
2022-09-03 20:12:32 +00:00
|
|
|
double min_dist = 999999;
|
|
|
|
double dist;
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
int ret;
|
2021-11-26 15:08:12 +00:00
|
|
|
Collision min_ret{};
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
for (i = 0; i < sop->clipbox_num; i++)
|
|
|
|
{
|
|
|
|
// move the box to position instead of using offset- this prevents small rounding errors
|
|
|
|
// allowing you to move through wall
|
2022-09-03 20:12:32 +00:00
|
|
|
DAngle ang = (pp->angle.ang + sop->clipbox_ang[i]);
|
|
|
|
DVector3 spos(pp->pos, zz);
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2022-09-03 20:12:32 +00:00
|
|
|
DVector2 vect = ang.ToVector() * 1024 * sop->clipbox_vdist[i];
|
2021-11-26 15:08:12 +00:00
|
|
|
Collision coll;
|
2022-09-03 20:12:32 +00:00
|
|
|
|
|
|
|
int xvect = vect.X * 16 * worldtoint; // note: this means clipmove input is Q18.14!
|
|
|
|
int yvect = vect.Y * 16 * worldtoint;
|
2022-09-03 20:20:12 +00:00
|
|
|
clipmove(spos, &pp->cursector, xvect, yvect, (int)sop->clipbox_dist[i], Z(4), int(floordist * zworldtoint), CLIPMASK_PLAYER, coll, 1);
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2021-11-26 15:08:12 +00:00
|
|
|
if (coll.type != kHitNone)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
|
|
|
// hit something moving into start position
|
|
|
|
min_dist = 0;
|
|
|
|
min_ndx = i;
|
|
|
|
// ox is where it should be
|
2022-09-03 20:12:32 +00:00
|
|
|
opos[i].XY() = pp->pos + ang.ToVector() * sop->clipbox_vdist[i];
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2021-10-30 10:59:47 +00:00
|
|
|
// spos.x is where it hit
|
2022-09-03 20:12:32 +00:00
|
|
|
pos[i].XY() = spos.XY();
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
// see the dist moved
|
2022-09-03 20:12:32 +00:00
|
|
|
dist = (pos[i].XY() - opos[i].XY()).Length();
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
// save it off
|
|
|
|
if (dist < min_dist)
|
|
|
|
{
|
|
|
|
min_dist = dist;
|
|
|
|
min_ndx = i;
|
2021-11-26 15:08:12 +00:00
|
|
|
min_ret = coll;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// save off the start position
|
2021-10-30 10:59:47 +00:00
|
|
|
opos[i] = pos[i] = spos;
|
2022-09-03 20:20:12 +00:00
|
|
|
pos[i].Z = zz;
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
// move the box
|
2022-09-03 20:20:12 +00:00
|
|
|
clipmove(pos[i], &pp->cursector, pp->vect.X, pp->vect.Y, (int)sop->clipbox_dist[i], Z(4), int(floordist * zworldtoint), CLIPMASK_PLAYER, coll);
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
// save the dist moved
|
2022-09-03 20:12:32 +00:00
|
|
|
dist = (pos[i].XY() - opos[i].XY()).Length();
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
if (dist < min_dist)
|
|
|
|
{
|
|
|
|
min_dist = dist;
|
|
|
|
min_ndx = i;
|
2021-11-26 15:08:12 +00:00
|
|
|
min_ret = coll;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// put posx and y off from offset
|
2022-09-03 20:12:32 +00:00
|
|
|
pp->pos.XY() += pos[min_ndx].XY() - opos[min_ndx].XY();
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
return min_ret;
|
|
|
|
}
|
|
|
|
|
2022-09-03 20:20:12 +00:00
|
|
|
int MultiClipTurn(PLAYER* pp, DAngle new_ang, double zz, double floordist)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
|
|
|
int i;
|
2021-12-31 14:50:44 +00:00
|
|
|
SECTOR_OBJECT* sop = pp->sop;
|
2015-05-19 21:54:34 +00:00
|
|
|
int ret;
|
2021-11-26 16:50:49 +00:00
|
|
|
auto cursect = pp->cursector;
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
for (i = 0; i < sop->clipbox_num; i++)
|
|
|
|
{
|
2022-09-03 20:12:32 +00:00
|
|
|
DAngle ang = new_ang + sop->clipbox_ang[i];
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2022-09-03 20:12:32 +00:00
|
|
|
DVector3 spos(pp->pos, zz);
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2022-09-03 20:12:32 +00:00
|
|
|
DVector2 vect = ang.ToVector() * 1024 * sop->clipbox_vdist[i];
|
2021-11-26 15:41:49 +00:00
|
|
|
Collision coll;
|
2022-09-03 20:12:32 +00:00
|
|
|
|
|
|
|
int xvect = vect.X * 16 * worldtoint; // note: this means clipmove input is Q18.14!
|
|
|
|
int yvect = vect.Y * 16 * worldtoint;
|
2022-09-03 20:20:12 +00:00
|
|
|
clipmove(spos, &cursect, xvect, yvect, (int)sop->clipbox_dist[i], Z(4), int(floordist * zworldtoint), CLIPMASK_PLAYER, coll);
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2021-11-26 15:41:49 +00:00
|
|
|
ASSERT(cursect);
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2021-11-26 15:41:49 +00:00
|
|
|
if (coll.type != kHitNone)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
|
|
|
// attempt to move a bit when turning against a wall
|
|
|
|
//ang = NORM_ANGLE(ang + 1024);
|
2020-11-15 09:54:59 +00:00
|
|
|
//pp->xvect += 20 * bcos(ang);
|
|
|
|
//pp->yvect += 20 * bsin(ang);
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-09 17:52:52 +00:00
|
|
|
return true;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
2022-09-03 19:36:47 +00:00
|
|
|
int testquadinsect(int *point_num, DVector2 const * qp, sectortype* sect)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
|
|
|
int i,next_i;
|
|
|
|
|
|
|
|
*point_num = -1;
|
|
|
|
|
|
|
|
for (i=0; i < 4; i++)
|
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
if (!inside(qp[i].X, qp[i].Y, sect))
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
|
|
|
*point_num = i;
|
|
|
|
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i=0; i<4; i++)
|
|
|
|
{
|
2021-11-01 14:31:42 +00:00
|
|
|
next_i = (i+1) & 3;
|
2022-09-03 19:36:47 +00:00
|
|
|
if (!cansee(DVector3(qp[i], 0x3fffff), sect, DVector3(qp[next_i], 0x3fffff), sect))
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-09 17:52:52 +00:00
|
|
|
return true;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//Ken gives the tank clippin' a try...
|
2022-09-03 19:36:47 +00:00
|
|
|
int RectClipMove(PLAYER* pp, DVector2* qpos)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2017-01-23 11:21:36 +00:00
|
|
|
int i;
|
2022-09-03 19:36:47 +00:00
|
|
|
DVector2 xy[4];
|
2015-05-19 21:54:34 +00:00
|
|
|
int point_num;
|
2022-09-03 19:36:47 +00:00
|
|
|
DVector2 pvect((pp->vect.X >> 14) * inttoworld, (pp->vect.Y >> 14) * inttoworld);
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
xy[i] = qpos[i] + pvect;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//Given the 4 points: x[4], y[4]
|
2021-11-26 16:50:49 +00:00
|
|
|
if (testquadinsect(&point_num, xy, pp->cursector))
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
pp->pos += pvect;
|
2020-09-09 17:52:52 +00:00
|
|
|
return true;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (point_num < 0)
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
|
|
|
|
if ((point_num == 0) || (point_num == 3)) //Left side bad - strafe right
|
|
|
|
{
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
xy[i].X = qpos[i].X - pvect.Y * 0.5;
|
|
|
|
xy[i].Y = qpos[i].Y + pvect.X * 0.5;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
2021-11-26 16:50:49 +00:00
|
|
|
if (testquadinsect(&point_num, xy, pp->cursector))
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
pp->pos.XY() += { -pvect.X * 0.5, pvect.X * 0.5 };
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((point_num == 1) || (point_num == 2)) //Right side bad - strafe left
|
|
|
|
{
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
xy[i].X = qpos[i].X + pvect.Y * 0.5;
|
|
|
|
xy[i].Y = qpos[i].Y - pvect.X * 0.5;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
2021-11-26 16:50:49 +00:00
|
|
|
if (testquadinsect(&point_num, xy, pp->cursector))
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
pp->pos.XY() += { pvect.X * 0.5, -pvect.X * 0.5 };
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
2022-09-03 19:36:47 +00:00
|
|
|
int testpointinquad(const DVector2& pt, const DVector2* quad)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
for (int i = 0; i < 4; i++)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
double dist = PointOnLineSide(pt.X, pt.Y, quad[i].X, quad[i].Y, quad[(i + 1) & 3].X - quad[i].X, quad[(i + 1) & 3].Y - quad[i].Y);
|
|
|
|
if (dist > 0) return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
2022-09-03 19:36:47 +00:00
|
|
|
return true;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
2022-09-03 19:36:47 +00:00
|
|
|
short RectClipTurn(PLAYER* pp, DAngle new_angl, DVector2* qpos, DVector2* opos)
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
2016-05-23 04:46:30 +00:00
|
|
|
int i;
|
2022-09-03 19:36:47 +00:00
|
|
|
DVector2 xy[4];
|
2021-12-31 14:50:44 +00:00
|
|
|
SECTOR_OBJECT* sop = pp->sop;
|
2022-09-03 19:36:47 +00:00
|
|
|
DAngle rot_angl;
|
2015-05-19 21:54:34 +00:00
|
|
|
int point_num;
|
|
|
|
|
2022-10-03 19:47:05 +00:00
|
|
|
rot_angl = new_angl + sop->spin_ang - sop->ang_orig;
|
2015-05-19 21:54:34 +00:00
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
xy[i] = rotatepoint(pp->pos.XY(), opos[i], rot_angl);
|
2015-05-19 21:54:34 +00:00
|
|
|
// cannot use sop->xmid and ymid because the SO is off the map at this point
|
|
|
|
}
|
|
|
|
|
|
|
|
//Given the 4 points: x[4], y[4]
|
2021-11-26 16:50:49 +00:00
|
|
|
if (testquadinsect(&point_num, xy, pp->cursector))
|
2015-05-19 21:54:34 +00:00
|
|
|
{
|
|
|
|
// move to new pos
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
2022-09-03 19:36:47 +00:00
|
|
|
qpos[i] = xy[i];
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
2020-09-09 17:52:52 +00:00
|
|
|
return true;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (point_num < 0)
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
|
2020-09-09 17:52:52 +00:00
|
|
|
return false;
|
2015-05-19 21:54:34 +00:00
|
|
|
}
|
2019-10-09 16:09:05 +00:00
|
|
|
END_SW_NS
|