2018-11-05 07:28:01 +00:00
|
|
|
// "Build Engine & Tools" Copyright (c) 1993-1997 Ken Silverman
|
|
|
|
// Ken Silverman's official web site: "http://www.advsys.net/ken"
|
|
|
|
// See the included license file "BUILDLIC.TXT" for license info.
|
|
|
|
//
|
|
|
|
// This file has been modified from Ken Silverman's original release
|
|
|
|
// by Jonathon Fowler (jf@jonof.id.au)
|
|
|
|
// by the EDuke32 team (development@voidpoint.com)
|
|
|
|
|
2016-06-21 00:33:30 +00:00
|
|
|
#include "build.h"
|
2019-04-10 09:31:01 +00:00
|
|
|
#include "clip.h"
|
2020-08-28 07:06:49 +00:00
|
|
|
#include "printf.h"
|
2021-03-20 11:47:51 +00:00
|
|
|
#include "gamefuncs.h"
|
2021-12-03 23:47:08 +00:00
|
|
|
#include "coreactor.h"
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-03-20 15:46:06 +00:00
|
|
|
enum { MAXCLIPDIST = 1024 };
|
2020-04-11 22:04:02 +00:00
|
|
|
|
2021-11-11 19:20:00 +00:00
|
|
|
static int clipnum;
|
2019-04-10 09:31:01 +00:00
|
|
|
static linetype clipit[MAXCLIPNUM];
|
|
|
|
static int32_t clipsectnum, origclipsectnum, clipspritenum;
|
2021-11-11 19:20:00 +00:00
|
|
|
int clipsectorlist[MAXCLIPSECTORS];
|
|
|
|
static int origclipsectorlist[MAXCLIPSECTORS];
|
2021-12-04 00:05:18 +00:00
|
|
|
static CollisionBase clipobjectval[MAXCLIPNUM];
|
2019-07-24 12:56:44 +00:00
|
|
|
static uint8_t clipignore[(MAXCLIPNUM+7)>>3];
|
2021-03-27 14:03:57 +00:00
|
|
|
static int32_t rxi[8], ryi[8];
|
|
|
|
|
2021-12-06 19:08:32 +00:00
|
|
|
BitArray clipsectormap;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
int32_t quickloadboard=0;
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
vec2_t hitscangoal = { (1<<29)-1, (1<<29)-1 };
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
////////// CLIPMOVE //////////
|
2022-08-04 17:13:55 +00:00
|
|
|
inline uint8_t bitmap_test(uint8_t const* const ptr, int const n) { return ptr[n >> 3] & (1 << (n & 7)); }
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-03-27 14:03:57 +00:00
|
|
|
|
|
|
|
// x1, y1: in/out
|
|
|
|
// rest x/y: out
|
2022-02-02 17:37:01 +00:00
|
|
|
static inline void get_wallspr_points(DCoreActor* actor, int32_t *x1, int32_t *x2, int32_t *y1, int32_t *y2)
|
2021-03-27 14:03:57 +00:00
|
|
|
{
|
2022-08-17 20:06:39 +00:00
|
|
|
DVector2 out[2];
|
|
|
|
GetWallSpritePosition(&actor->spr, DVector2(*x1 * inttoworld, *y1 * inttoworld), out);
|
|
|
|
*x1 = int(out[0].X * worldtoint); *y1 = int(out[0].Y * worldtoint);
|
|
|
|
*x2 = int(out[1].X * worldtoint); *y2 = int(out[1].Y * worldtoint);
|
2021-03-27 14:03:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// x1, y1: in/out
|
|
|
|
// rest x/y: out
|
2021-12-30 10:35:49 +00:00
|
|
|
static inline void get_floorspr_points(DCoreActor *spr, int32_t px, int32_t py,
|
2021-03-27 14:03:57 +00:00
|
|
|
int32_t *x1, int32_t *x2, int32_t *x3, int32_t *x4,
|
2022-08-17 19:37:16 +00:00
|
|
|
int32_t *y1, int32_t *y2, int32_t *y3, int32_t *y4)
|
2021-03-27 14:03:57 +00:00
|
|
|
{
|
2022-08-17 19:37:16 +00:00
|
|
|
DVector2 out[4];
|
|
|
|
// very messed up interface... :(
|
|
|
|
GetFlatSpritePosition(spr, DVector2((*x1 - px) * inttoworld, (*y1 - py) * inttoworld), out);
|
|
|
|
*x1 = int(out[0].X * worldtoint); *y1 = int(out[0].Y * worldtoint);
|
|
|
|
*x2 = int(out[1].X * worldtoint); *y2 = int(out[1].Y * worldtoint);
|
|
|
|
*x3 = int(out[2].X * worldtoint); *y3 = int(out[2].Y * worldtoint);
|
|
|
|
*x4 = int(out[3].X * worldtoint); *y4 = int(out[3].Y * worldtoint);
|
2021-03-27 14:03:57 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-06-21 00:33:30 +00:00
|
|
|
//
|
|
|
|
// clipinsideboxline
|
|
|
|
//
|
2022-08-17 21:24:44 +00:00
|
|
|
static int clipinsideboxline(int x, int y, int x1, int y1, int x2, int y2, int walldist)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-08-17 21:24:44 +00:00
|
|
|
return (int)IsCloseToLine(DVector2(x * inttoworld, y * inttoworld), DVector2(x1 * inttoworld, y1 * inttoworld), DVector2(x2 * inttoworld, y2 * inttoworld), walldist * inttoworld);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
2019-04-18 17:23:42 +00:00
|
|
|
static int32_t clipmove_warned;
|
|
|
|
|
|
|
|
static inline void addclipsect(int const sectnum)
|
|
|
|
{
|
2019-08-27 06:52:30 +00:00
|
|
|
if (clipsectnum < MAXCLIPSECTORS)
|
2019-04-18 17:23:42 +00:00
|
|
|
{
|
2021-12-06 19:08:32 +00:00
|
|
|
clipsectormap.Set(sectnum);
|
2019-04-18 17:23:42 +00:00
|
|
|
clipsectorlist[clipsectnum++] = sectnum;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
clipmove_warned |= 1;
|
|
|
|
}
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
|
2021-12-04 00:05:18 +00:00
|
|
|
static void addclipline(int32_t dax1, int32_t day1, int32_t dax2, int32_t day2, const CollisionBase& daoval, int nofix)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-08-27 06:52:30 +00:00
|
|
|
if (clipnum >= MAXCLIPNUM)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-07-24 12:56:44 +00:00
|
|
|
clipmove_warned |= 2;
|
|
|
|
return;
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
2019-07-24 12:56:44 +00:00
|
|
|
|
|
|
|
clipit[clipnum].x1 = dax1; clipit[clipnum].y1 = day1;
|
|
|
|
clipit[clipnum].x2 = dax2; clipit[clipnum].y2 = day2;
|
2019-07-26 21:54:52 +00:00
|
|
|
clipobjectval[clipnum] = daoval;
|
2019-07-24 12:56:44 +00:00
|
|
|
|
2021-03-24 19:37:20 +00:00
|
|
|
uint32_t const mask = (1 << (clipnum&7));
|
2019-08-04 02:51:50 +00:00
|
|
|
uint8_t &value = clipignore[clipnum>>3];
|
2019-07-26 21:54:52 +00:00
|
|
|
value = (value & ~mask) | (-nofix & mask);
|
2019-07-24 12:56:44 +00:00
|
|
|
|
|
|
|
clipnum++;
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
2021-03-24 20:43:36 +00:00
|
|
|
inline void clipmove_tweak_pos(const vec3_t *pos, int32_t gx, int32_t gy, int32_t x1, int32_t y1, int32_t x2,
|
2016-06-21 00:33:35 +00:00
|
|
|
int32_t y2, int32_t *daxptr, int32_t *dayptr)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
|
|
|
int32_t daz;
|
|
|
|
|
2019-11-08 16:15:00 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_19950829 ||
|
2021-12-22 09:40:26 +00:00
|
|
|
rintersect(pos->X, pos->Y, 0, gx, gy, 0, x1, y1, x2, y2, daxptr, dayptr, &daz) == -1)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-22 09:36:09 +00:00
|
|
|
*daxptr = pos->X;
|
2021-12-22 09:40:26 +00:00
|
|
|
*dayptr = pos->Y;
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns: should clip?
|
2019-08-27 06:52:42 +00:00
|
|
|
static int cliptestsector(int const dasect, int const nextsect, int32_t const flordist, int32_t const ceildist, vec2_t const pos, int32_t const posz)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-11-07 17:09:19 +00:00
|
|
|
assert(validSectorIndex(dasect) && validSectorIndex(nextsect));
|
2019-08-27 06:52:42 +00:00
|
|
|
|
2021-12-30 10:45:58 +00:00
|
|
|
auto const sec2 = §or[nextsect];
|
2019-08-27 06:52:42 +00:00
|
|
|
|
2019-09-22 19:26:07 +00:00
|
|
|
switch (enginecompatibility_mode)
|
|
|
|
{
|
|
|
|
case ENGINECOMPATIBILITY_NONE:
|
|
|
|
break;
|
|
|
|
default:
|
2019-07-29 11:12:06 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t daz = getflorzofslopeptr(§or[dasect], pos.X, pos.Y);
|
|
|
|
int32_t daz2 = getflorzofslopeptr(sec2, pos.X, pos.Y);
|
2019-07-29 11:12:06 +00:00
|
|
|
|
2021-12-18 12:14:56 +00:00
|
|
|
if (daz2 < daz-(1<<8) && (sec2->floorstat & CSTAT_SECTOR_SKY) == 0)
|
2019-09-21 11:02:17 +00:00
|
|
|
if (posz >= daz2-(flordist-1)) return 1;
|
2021-12-22 09:28:51 +00:00
|
|
|
daz = getceilzofslopeptr(§or[dasect], pos.X, pos.Y);
|
|
|
|
daz2 = getceilzofslopeptr(sec2, pos.X, pos.Y);
|
2021-12-18 12:14:56 +00:00
|
|
|
if (daz2 > daz+(1<<8) && (sec2->ceilingstat & CSTAT_SECTOR_SKY) == 0)
|
2019-09-21 11:02:17 +00:00
|
|
|
if (posz <= daz2+(ceildist-1)) return 1;
|
2019-07-29 11:12:06 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2019-09-22 19:26:07 +00:00
|
|
|
}
|
2019-07-29 11:12:06 +00:00
|
|
|
|
2022-02-02 23:46:04 +00:00
|
|
|
int32_t daz2 = sec2->int_floorz();
|
|
|
|
int32_t dacz2 = sec2->int_ceilingz();
|
2019-04-19 08:31:54 +00:00
|
|
|
|
2021-12-18 12:14:56 +00:00
|
|
|
if ((sec2->floorstat|sec2->ceilingstat) & CSTAT_SECTOR_SLOPE)
|
2021-12-22 09:28:51 +00:00
|
|
|
getcorrectzsofslope(nextsect, pos.X, pos.Y, &dacz2, &daz2);
|
2019-07-24 01:37:43 +00:00
|
|
|
|
2019-07-24 10:56:20 +00:00
|
|
|
if (daz2 <= dacz2)
|
2019-07-24 01:37:43 +00:00
|
|
|
return 1;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-30 10:45:58 +00:00
|
|
|
auto const sec = §or[dasect];
|
2019-09-22 19:26:07 +00:00
|
|
|
|
2022-02-02 23:46:04 +00:00
|
|
|
int32_t daz = sec->int_floorz();
|
|
|
|
int32_t dacz = sec->int_ceilingz();
|
2019-07-24 01:37:43 +00:00
|
|
|
|
2021-12-18 12:14:56 +00:00
|
|
|
if ((sec->floorstat|sec->ceilingstat) & CSTAT_SECTOR_SLOPE)
|
2021-12-22 09:28:51 +00:00
|
|
|
getcorrectzsofslope(dasect, pos.X, pos.Y, &dacz, &daz);
|
2019-07-24 01:37:43 +00:00
|
|
|
|
2021-01-04 12:02:00 +00:00
|
|
|
int32_t const sec2height = abs(daz2-dacz2);
|
2019-08-01 06:49:47 +00:00
|
|
|
|
2021-01-04 12:02:00 +00:00
|
|
|
return ((abs(daz-dacz) > sec2height && // clip if the current sector is taller and the next is too small
|
2019-09-17 03:21:14 +00:00
|
|
|
sec2height < (ceildist+(CLIPCURBHEIGHT<<1))) ||
|
2019-08-01 06:49:47 +00:00
|
|
|
|
2021-12-18 12:14:56 +00:00
|
|
|
((sec2->floorstat & CSTAT_SECTOR_SKY) == 0 && // parallaxed floor curbs don't clip
|
2019-08-01 06:49:47 +00:00
|
|
|
posz >= daz2-(flordist-1) && // also account for desired z distance tolerance
|
|
|
|
daz2 < daz-CLIPCURBHEIGHT) || // curbs less tall than 256 z units don't clip
|
|
|
|
|
2021-12-18 12:14:56 +00:00
|
|
|
((sec2->ceilingstat & CSTAT_SECTOR_SKY) == 0 &&
|
2019-08-01 06:49:47 +00:00
|
|
|
posz <= dacz2+(ceildist-1) &&
|
2020-02-11 09:21:52 +00:00
|
|
|
dacz2 > dacz+CLIPCURBHEIGHT)); // ceilings check the same conditions ^^^^^
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// raytrace (internal)
|
|
|
|
//
|
2019-08-27 06:52:42 +00:00
|
|
|
static inline int32_t cliptrace(vec2_t const pos, vec2_t * const goal)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
|
|
|
int32_t hitwall = -1;
|
|
|
|
|
2021-05-11 23:29:18 +00:00
|
|
|
for (int z=clipnum-1; z>=0; z--)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-07-24 01:38:08 +00:00
|
|
|
vec2_t const p1 = { clipit[z].x1, clipit[z].y1 };
|
|
|
|
vec2_t const p2 = { clipit[z].x2, clipit[z].y2 };
|
2021-12-22 09:28:51 +00:00
|
|
|
vec2_t const area = { p2.X-p1.X, p2.Y-p1.Y };
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t topu = area.X*(pos.Y-p1.Y) - (pos.X-p1.X)*area.Y;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (topu <= 0 || area.X*(goal->Y-p1.Y) > (goal->X-p1.X)*area.Y)
|
2016-06-21 00:33:30 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
vec2_t const diff = { goal->X-pos.X, goal->Y-pos.Y };
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (diff.X*(p1.Y-pos.Y) > (p1.X-pos.X)*diff.Y || diff.X*(p2.Y-pos.Y) <= (p2.X-pos.X)*diff.Y)
|
2016-06-21 00:33:30 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t const bot = diff.X*area.Y - area.X*diff.Y;
|
2021-12-14 09:15:58 +00:00
|
|
|
int cnt = 256;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2019-07-24 01:38:08 +00:00
|
|
|
if (!bot)
|
2016-06-21 00:33:30 +00:00
|
|
|
continue;
|
|
|
|
|
2019-07-24 01:38:08 +00:00
|
|
|
vec2_t n;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
do
|
|
|
|
{
|
2019-07-24 01:38:08 +00:00
|
|
|
if (--cnt < 0)
|
|
|
|
{
|
|
|
|
*goal = pos;
|
|
|
|
return z;
|
|
|
|
}
|
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
n = { pos.X+Scale(diff.X, topu, bot), pos.Y+Scale(diff.Y, topu, bot) };
|
2016-06-21 00:33:30 +00:00
|
|
|
topu--;
|
2021-12-22 09:28:51 +00:00
|
|
|
} while (area.X*(n.Y-p1.Y) <= (n.X-p1.X)*area.Y);
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (abs(pos.X-n.X)+abs(pos.Y-n.Y) < abs(pos.X-goal->X)+abs(pos.Y-goal->Y))
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-07-24 01:38:08 +00:00
|
|
|
*goal = n;
|
|
|
|
hitwall = z;
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return hitwall;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// keepaway (internal)
|
|
|
|
//
|
|
|
|
static inline void keepaway(int32_t *x, int32_t *y, int32_t w)
|
|
|
|
{
|
|
|
|
const int32_t x1 = clipit[w].x1, dx = clipit[w].x2-x1;
|
|
|
|
const int32_t y1 = clipit[w].y1, dy = clipit[w].y2-y1;
|
2021-01-04 12:35:33 +00:00
|
|
|
const int32_t ox = Sgn(-dy), oy = Sgn(dx);
|
2022-08-04 17:13:55 +00:00
|
|
|
uint8_t first = (abs(dx) <= abs(dy));
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2019-03-30 19:35:54 +00:00
|
|
|
do
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
|
|
|
if (dx*(*y-y1) > (*x-x1)*dy)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (first == 0)
|
|
|
|
*x += ox;
|
|
|
|
else
|
|
|
|
*y += oy;
|
|
|
|
|
|
|
|
first ^= 1;
|
|
|
|
}
|
2019-03-30 19:35:54 +00:00
|
|
|
while (1);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
2019-08-27 06:52:42 +00:00
|
|
|
static int get_floorspr_clipyou(vec2_t const v1, vec2_t const v2, vec2_t const v3, vec2_t const v4)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2019-08-27 06:52:42 +00:00
|
|
|
int clipyou = 0;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v1.Y^v2.Y) < 0)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v1.X^v2.X) < 0) clipyou ^= (v1.X*v2.Y < v2.X*v1.Y)^(v1.Y<v2.Y);
|
2021-12-22 09:26:51 +00:00
|
|
|
else if (v1.X >= 0) clipyou ^= 1;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v2.Y^v3.Y) < 0)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v2.X^v3.X) < 0) clipyou ^= (v2.X*v3.Y < v3.X*v2.Y)^(v2.Y<v3.Y);
|
2021-12-22 09:26:51 +00:00
|
|
|
else if (v2.X >= 0) clipyou ^= 1;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v3.Y^v4.Y) < 0)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v3.X^v4.X) < 0) clipyou ^= (v3.X*v4.Y < v4.X*v3.Y)^(v3.Y<v4.Y);
|
2021-12-22 09:26:51 +00:00
|
|
|
else if (v3.X >= 0) clipyou ^= 1;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v4.Y^v1.Y) < 0)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if ((v4.X^v1.X) < 0) clipyou ^= (v4.X*v1.Y < v1.X*v4.Y)^(v4.Y<v1.Y);
|
2021-12-22 09:26:51 +00:00
|
|
|
else if (v4.X >= 0) clipyou ^= 1;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return clipyou;
|
|
|
|
}
|
|
|
|
|
2022-08-04 21:47:29 +00:00
|
|
|
static int32_t getwalldist(vec2_t const in, int const wallnum)
|
|
|
|
{
|
|
|
|
auto dvec = NearestPointOnWall(in.X * maptoworld, in.Y * maptoworld, &wall[wallnum]);
|
|
|
|
vec2_t closest = { int(dvec.X * worldtoint), int(dvec.Y * worldtoint) };
|
|
|
|
return abs(closest.X - in.X) + abs(closest.Y - in.Y);
|
|
|
|
}
|
|
|
|
|
2021-11-07 18:21:46 +00:00
|
|
|
static void clipupdatesector(vec2_t const pos, int * const sectnum, int walldist)
|
2019-04-18 17:23:50 +00:00
|
|
|
{
|
2020-09-10 19:39:52 +00:00
|
|
|
#if 0
|
2019-09-22 19:26:07 +00:00
|
|
|
if (enginecompatibility_mode != ENGINECOMPATIBILITY_NONE)
|
2019-07-29 11:12:06 +00:00
|
|
|
{
|
|
|
|
updatesector(pos.x, pos.y, sectnum);
|
|
|
|
return;
|
|
|
|
}
|
2020-09-10 19:39:52 +00:00
|
|
|
#endif
|
2019-07-29 11:12:06 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (inside_p(pos.X, pos.Y, *sectnum))
|
2019-04-09 19:21:22 +00:00
|
|
|
return;
|
|
|
|
|
2022-08-04 21:47:47 +00:00
|
|
|
double nsecs = SquareDistToSector(pos.X * inttoworld, pos.Y * inttoworld, §or[*sectnum]);
|
2019-10-20 17:55:20 +00:00
|
|
|
|
2022-08-04 21:47:47 +00:00
|
|
|
double wd = (walldist + 8) * inttoworld; wd *= wd;
|
|
|
|
if (nsecs > wd)
|
2019-10-20 17:55:20 +00:00
|
|
|
{
|
|
|
|
walldist = 0x7fff;
|
|
|
|
}
|
|
|
|
|
2019-04-09 19:21:22 +00:00
|
|
|
{
|
2021-12-21 09:51:41 +00:00
|
|
|
BFSSearch search(sector.Size(), *sectnum);
|
2019-04-18 17:25:11 +00:00
|
|
|
|
2021-11-16 23:13:34 +00:00
|
|
|
for (unsigned listsectnum; (listsectnum = search.GetNext()) != BFSSearch::EOL;)
|
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if (inside_p(pos.X, pos.Y, listsectnum))
|
2021-11-16 23:13:34 +00:00
|
|
|
{
|
|
|
|
*sectnum = listsectnum;
|
|
|
|
return;
|
|
|
|
}
|
2019-04-09 19:21:22 +00:00
|
|
|
|
2021-11-16 23:13:34 +00:00
|
|
|
for (auto& wal : wallsofsector(listsectnum))
|
|
|
|
{
|
2021-12-06 19:08:32 +00:00
|
|
|
if (wal.nextsector >= 0 && clipsectormap[wal.nextsector])
|
2021-11-16 23:13:34 +00:00
|
|
|
search.Add(wal.nextsector);
|
|
|
|
}
|
|
|
|
}
|
2019-04-09 19:21:22 +00:00
|
|
|
}
|
|
|
|
|
2019-04-10 09:31:09 +00:00
|
|
|
{
|
2021-12-21 09:51:41 +00:00
|
|
|
BFSSearch search(sector.Size(), *sectnum);
|
2019-04-18 17:25:11 +00:00
|
|
|
|
2021-11-16 23:13:34 +00:00
|
|
|
for (unsigned listsectnum; (listsectnum = search.GetNext()) != BFSSearch::EOL;)
|
2019-04-10 09:31:09 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if (inside_p(pos.X, pos.Y, listsectnum))
|
2021-11-16 23:13:34 +00:00
|
|
|
{
|
|
|
|
*sectnum = listsectnum;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (auto& wal : wallsofsector(listsectnum))
|
|
|
|
{
|
|
|
|
if (wal.nextsector >= 0 && getwalldist(pos, wallnum(&wal)) <= (walldist + 8))
|
|
|
|
search.Add(wal.nextsector);
|
|
|
|
}
|
2019-04-10 09:31:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-09 19:21:22 +00:00
|
|
|
*sectnum = -1;
|
|
|
|
}
|
|
|
|
|
2016-06-21 00:33:30 +00:00
|
|
|
//
|
|
|
|
// clipmove
|
|
|
|
//
|
2021-12-04 00:05:18 +00:00
|
|
|
CollisionBase clipmove_(vec3_t * const pos, int * const sectnum, int32_t xvect, int32_t yvect,
|
2021-11-07 19:00:22 +00:00
|
|
|
int32_t const walldist, int32_t const ceildist, int32_t const flordist, uint32_t const cliptype, int clipmoveboxtracenum)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-04 00:05:18 +00:00
|
|
|
CollisionBase b{};
|
2016-06-21 00:33:35 +00:00
|
|
|
if ((xvect|yvect) == 0 || *sectnum < 0)
|
2021-12-04 00:05:18 +00:00
|
|
|
return b;
|
2016-06-21 00:33:35 +00:00
|
|
|
|
2021-12-04 00:05:18 +00:00
|
|
|
DCoreActor* curspr=NULL; // non-NULL when handling sprite with sector-like clipping
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2019-04-19 08:31:50 +00:00
|
|
|
int const initialsectnum = *sectnum;
|
|
|
|
|
2021-12-18 14:36:50 +00:00
|
|
|
int32_t const dawalclipmask = (cliptype & 65535); // CLIPMASK0 = 0x00010001 (in desperate need of getting fixed!)
|
2021-07-04 11:05:33 +00:00
|
|
|
int32_t const dasprclipmask = (cliptype >> 16); // CLIPMASK1 = 0x01000040
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2019-03-30 19:36:18 +00:00
|
|
|
vec2_t const move = { xvect, yvect };
|
2021-12-22 09:40:26 +00:00
|
|
|
vec2_t goal = { pos->X + (xvect >> 14), pos->Y + (yvect >> 14) };
|
|
|
|
vec2_t const cent = { (pos->X + goal.X) >> 1, (pos->Y + goal.Y) >> 1 };
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
//Extra walldist for sprites on sector lines
|
2021-12-22 09:40:26 +00:00
|
|
|
vec2_t const diff = { goal.X - (pos->X), goal.Y - (pos->Y) };
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t const rad = ksqrt(compat_maybe_truncate_to_int32(uhypsq(diff.X, diff.Y))) + MAXCLIPDIST + walldist + 8;
|
|
|
|
vec2_t const clipMin = { cent.X - rad, cent.Y - rad };
|
|
|
|
vec2_t const clipMax = { cent.X + rad, cent.Y + rad };
|
2019-03-30 19:36:18 +00:00
|
|
|
|
|
|
|
int clipsectcnt = 0;
|
|
|
|
int clipspritecnt = 0;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2019-03-30 19:36:18 +00:00
|
|
|
clipsectorlist[0] = *sectnum;
|
|
|
|
|
|
|
|
clipsectnum = 1;
|
|
|
|
clipnum = 0;
|
|
|
|
clipspritenum = 0;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
clipmove_warned = 0;
|
|
|
|
|
2021-12-06 19:08:32 +00:00
|
|
|
clipsectormap.Zero();
|
|
|
|
clipsectormap.Set(*sectnum);
|
2019-04-10 09:31:12 +00:00
|
|
|
|
2016-06-21 00:33:30 +00:00
|
|
|
do
|
|
|
|
{
|
2019-03-30 19:36:18 +00:00
|
|
|
int const dasect = clipsectorlist[clipsectcnt++];
|
2016-06-21 00:33:30 +00:00
|
|
|
//if (curspr)
|
2020-04-11 21:45:45 +00:00
|
|
|
// Printf("sprite %d/%d: sect %d/%d (%d)\n", clipspritecnt,clipspritenum, clipsectcnt,clipsectnum,dasect);
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
////////// Walls //////////
|
|
|
|
|
2021-12-30 10:45:58 +00:00
|
|
|
auto const sec = §or[dasect];
|
2019-03-30 19:36:29 +00:00
|
|
|
int const startwall = sec->wallptr;
|
|
|
|
int const endwall = startwall + sec->wallnum;
|
2021-12-30 10:38:34 +00:00
|
|
|
auto wal = &wall[startwall];
|
2019-03-30 19:36:18 +00:00
|
|
|
|
2021-12-14 09:15:58 +00:00
|
|
|
for (int j=startwall; j<endwall; j++, wal++)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-30 10:38:34 +00:00
|
|
|
auto const wal2 = wal->point2Wall();
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2022-01-27 16:41:10 +00:00
|
|
|
if ((wal->wall_int_pos().X < clipMin.X && wal2->wall_int_pos().X < clipMin.X) || (wal->wall_int_pos().X > clipMax.X && wal2->wall_int_pos().X > clipMax.X) ||
|
|
|
|
(wal->wall_int_pos().Y < clipMin.Y && wal2->wall_int_pos().Y < clipMin.Y) || (wal->wall_int_pos().Y > clipMax.Y && wal2->wall_int_pos().Y > clipMax.Y))
|
2016-06-21 00:33:35 +00:00
|
|
|
continue;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2022-01-27 16:41:10 +00:00
|
|
|
vec2_t p1 = wal->wall_int_pos();
|
|
|
|
vec2_t p2 = wal2->wall_int_pos();
|
2021-12-22 09:28:51 +00:00
|
|
|
vec2_t d = { p2.X-p1.X, p2.Y-p1.Y };
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if (d.X * (pos->Y-p1.Y) < (pos->X-p1.X) * d.Y)
|
2019-03-30 19:36:18 +00:00
|
|
|
continue; //If wall's not facing you
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
vec2_t const r = { (d.Y > 0) ? clipMax.X : clipMin.X, (d.X > 0) ? clipMin.Y : clipMax.Y };
|
|
|
|
vec2_t v = { d.X * (r.Y - p1.Y), d.Y * (r.X - p1.X) };
|
2019-03-30 19:36:18 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (v.X >= v.Y)
|
2019-03-30 19:36:18 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
int clipyou = 0;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-18 14:36:50 +00:00
|
|
|
if (wal->nextsector < 0 || (wal->cstat & EWallFlags::FromInt(dawalclipmask)))
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
|
|
|
clipyou = 1;
|
|
|
|
}
|
2021-03-19 14:12:54 +00:00
|
|
|
else
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
clipmove_tweak_pos(pos, diff.X, diff.Y, p1.X, p1.Y, p2.X, p2.Y, &v.X, &v.Y);
|
2021-12-22 09:41:47 +00:00
|
|
|
clipyou = cliptestsector(dasect, wal->nextsector, flordist, ceildist, v, pos->Z);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
2019-07-24 01:37:54 +00:00
|
|
|
// We're not interested in any sector reached by portal traversal that we're "inside" of.
|
2019-09-22 19:26:07 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_NONE && !curspr && dasect != initialsectnum
|
2021-12-22 09:40:26 +00:00
|
|
|
&& inside(pos->X, pos->Y, sec) == 1)
|
2019-07-24 01:37:54 +00:00
|
|
|
{
|
|
|
|
int k;
|
|
|
|
for (k=startwall; k<endwall; k++)
|
|
|
|
if (wall[k].nextsector == initialsectnum)
|
|
|
|
break;
|
|
|
|
if (k == endwall)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (clipyou)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-04 00:05:18 +00:00
|
|
|
CollisionBase objtype;
|
|
|
|
if (curspr) objtype.setSprite(curspr);
|
|
|
|
else objtype.setWall(j);
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
//Add 2 boxes at endpoints
|
2021-12-22 09:26:51 +00:00
|
|
|
int32_t bsz = walldist; if (diff.X < 0) bsz = -bsz;
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(p1.X-bsz, p1.Y-bsz, p1.X-bsz, p1.Y+bsz, objtype, false);
|
|
|
|
addclipline(p2.X-bsz, p2.Y-bsz, p2.X-bsz, p2.Y+bsz, objtype, false);
|
|
|
|
bsz = walldist; if (diff.Y < 0) bsz = -bsz;
|
|
|
|
addclipline(p1.X+bsz, p1.Y-bsz, p1.X-bsz, p1.Y-bsz, objtype, false);
|
|
|
|
addclipline(p2.X+bsz, p2.Y-bsz, p2.X-bsz, p2.Y-bsz, objtype, false);
|
2019-03-30 19:36:18 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
v.X = walldist; if (d.Y > 0) v.X = -v.X;
|
|
|
|
v.Y = walldist; if (d.X < 0) v.Y = -v.Y;
|
2019-07-24 05:23:43 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_NONE && d.X * (pos->Y-p1.Y-v.Y) < (pos->X-p1.X-v.X) * d.Y)
|
2021-12-22 09:28:51 +00:00
|
|
|
v.X >>= 1, v.Y >>= 1;
|
2019-07-24 05:23:43 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(p1.X+v.X, p1.Y+v.Y, p2.X+v.X, p2.Y+v.Y, objtype, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
2019-03-30 19:35:54 +00:00
|
|
|
else if (wal->nextsector>=0)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-06 19:08:32 +00:00
|
|
|
if (!clipsectormap[wal->nextsector])
|
2019-04-10 09:31:12 +00:00
|
|
|
addclipsect(wal->nextsector);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-27 06:52:30 +00:00
|
|
|
if (clipmove_warned & 1)
|
2020-04-11 21:45:45 +00:00
|
|
|
Printf("clipsectnum >= MAXCLIPSECTORS!\n");
|
2019-04-10 09:31:05 +00:00
|
|
|
|
2019-08-27 06:52:30 +00:00
|
|
|
if (clipmove_warned & 2)
|
2020-04-11 21:45:45 +00:00
|
|
|
Printf("clipnum >= MAXCLIPNUM!\n");
|
2019-04-10 09:31:05 +00:00
|
|
|
|
2016-06-21 00:33:30 +00:00
|
|
|
////////// Sprites //////////
|
|
|
|
|
|
|
|
if (dasprclipmask==0)
|
|
|
|
continue;
|
|
|
|
|
2021-12-04 00:05:18 +00:00
|
|
|
TSectIterator<DCoreActor> it(dasect);
|
|
|
|
while (auto actor = it.Next())
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-02-02 17:37:01 +00:00
|
|
|
const int32_t cstat = actor->spr.cstat;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
if (actor->spr.cstat2 & CSTAT2_SPRITE_NOFIND) continue;
|
2016-06-21 00:33:30 +00:00
|
|
|
if ((cstat&dasprclipmask) == 0)
|
|
|
|
continue;
|
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
auto p1 = actor->int_pos().vec2;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-04 00:05:18 +00:00
|
|
|
CollisionBase obj;
|
|
|
|
obj.setSprite(actor);
|
|
|
|
|
2021-12-20 19:27:12 +00:00
|
|
|
switch (cstat & (CSTAT_SPRITE_ALIGNMENT_MASK))
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2018-04-06 01:42:38 +00:00
|
|
|
case CSTAT_SPRITE_ALIGNMENT_FACING:
|
2021-12-22 09:28:51 +00:00
|
|
|
if (p1.X >= clipMin.X && p1.X <= clipMax.X && p1.Y >= clipMin.Y && p1.Y <= clipMax.Y)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-08-08 22:11:21 +00:00
|
|
|
int32_t height, daz = actor->int_pos().Z + actor->GetOffsetAndHeight(height);
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
if (pos->Z > daz-height-flordist && pos->Z < daz+ceildist)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-09-09 16:07:44 +00:00
|
|
|
int32_t bsz = (actor->int_clipdist())+walldist;
|
2021-12-22 09:26:51 +00:00
|
|
|
if (diff.X < 0) bsz = -bsz;
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(p1.X-bsz, p1.Y-bsz, p1.X-bsz, p1.Y+bsz, obj, false);
|
2022-09-09 16:07:44 +00:00
|
|
|
bsz = (actor->int_clipdist())+walldist;
|
2021-12-22 09:28:51 +00:00
|
|
|
if (diff.Y < 0) bsz = -bsz;
|
|
|
|
addclipline(p1.X+bsz, p1.Y-bsz, p1.X-bsz, p1.Y-bsz, obj, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2018-04-06 01:42:38 +00:00
|
|
|
case CSTAT_SPRITE_ALIGNMENT_WALL:
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-02-02 17:37:01 +00:00
|
|
|
int32_t height, daz = actor->int_pos().Z + actor->GetOffsetAndHeight(height);
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
if (pos->Z > daz-height-flordist && pos->Z < daz+ceildist)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-03-27 14:03:57 +00:00
|
|
|
vec2_t p2;
|
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
get_wallspr_points(actor, &p1.X, &p2.X, &p1.Y, &p2.Y);
|
2019-03-30 19:36:18 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (clipinsideboxline(cent.X, cent.Y, p1.X, p1.Y, p2.X, p2.Y, rad) != 0)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-08-16 21:15:49 +00:00
|
|
|
vec2_t v = { MulScale(bcos(actor->int_ang() + 256), walldist, 14),
|
|
|
|
MulScale(bsin(actor->int_ang() + 256), walldist, 14) };
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if ((p1.X-pos->X) * (p2.Y-pos->Y) >= (p2.X-pos->X) * (p1.Y-pos->Y)) // Front
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(p1.X+v.X, p1.Y+v.Y, p2.X+v.Y, p2.Y-v.X, obj, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
else
|
|
|
|
{
|
2021-12-18 15:39:44 +00:00
|
|
|
if ((cstat & CSTAT_SPRITE_ONE_SIDE) != 0)
|
2019-03-30 19:36:18 +00:00
|
|
|
continue;
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(p2.X-v.X, p2.Y-v.Y, p1.X-v.Y, p1.Y+v.X, obj, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//Side blocker
|
2021-12-22 09:40:26 +00:00
|
|
|
if ((p2.X-p1.X) * (pos->X-p1.X)+(p2.Y-p1.Y) * (pos->Y-p1.Y) < 0)
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(p1.X-v.Y, p1.Y+v.X, p1.X+v.X, p1.Y+v.Y, obj, true);
|
2021-12-22 09:40:26 +00:00
|
|
|
else if ((p1.X-p2.X) * (pos->X-p2.X)+(p1.Y-p2.Y) * (pos->Y-p2.Y) < 0)
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(p2.X+v.Y, p2.Y-v.X, p2.X-v.X, p2.Y-v.Y, obj, true);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-04-06 01:42:38 +00:00
|
|
|
case CSTAT_SPRITE_ALIGNMENT_FLOOR:
|
2021-12-20 19:27:12 +00:00
|
|
|
case CSTAT_SPRITE_ALIGNMENT_SLOPE:
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-08-17 19:56:14 +00:00
|
|
|
int heinum = spriteGetSlope(actor);
|
|
|
|
int sz = spriteGetZOfSlope(&actor->spr, pos->X, pos->Y, heinum);
|
2021-12-20 19:27:12 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
if (pos->Z > sz - flordist && pos->Z < sz + ceildist)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-18 15:39:44 +00:00
|
|
|
if ((cstat & CSTAT_SPRITE_ONE_SIDE) != 0)
|
2021-12-22 09:41:47 +00:00
|
|
|
if ((pos->Z > sz) == ((cstat & CSTAT_SPRITE_YFLIP)==0))
|
2016-06-21 00:33:30 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-22 09:26:51 +00:00
|
|
|
rxi[0] = p1.X;
|
2021-12-22 09:28:51 +00:00
|
|
|
ryi[0] = p1.Y;
|
2021-03-27 14:03:57 +00:00
|
|
|
|
2021-12-30 10:35:49 +00:00
|
|
|
get_floorspr_points(actor, 0, 0, &rxi[0], &rxi[1], &rxi[2], &rxi[3],
|
2022-08-17 19:37:16 +00:00
|
|
|
&ryi[0], &ryi[1], &ryi[2], &ryi[3]);
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2022-08-16 21:15:49 +00:00
|
|
|
vec2_t v = { MulScale(bcos(actor->int_ang() - 256), walldist, 14),
|
|
|
|
MulScale(bsin(actor->int_ang() - 256), walldist, 14) };
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if ((rxi[0]-pos->X) * (ryi[1]-pos->Y) < (rxi[1]-pos->X) * (ryi[0]-pos->Y))
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if (clipinsideboxline(cent.X, cent.Y, rxi[1], ryi[1], rxi[0], ryi[0], rad) != 0)
|
|
|
|
addclipline(rxi[1]-v.Y, ryi[1]+v.X, rxi[0]+v.X, ryi[0]+v.Y, obj, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
2021-12-22 09:40:26 +00:00
|
|
|
else if ((rxi[2]-pos->X) * (ryi[3]-pos->Y) < (rxi[3]-pos->X) * (ryi[2]-pos->Y))
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if (clipinsideboxline(cent.X, cent.Y, rxi[3], ryi[3], rxi[2], ryi[2], rad) != 0)
|
|
|
|
addclipline(rxi[3]+v.Y, ryi[3]-v.X, rxi[2]-v.X, ryi[2]-v.Y, obj, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if ((rxi[1]-pos->X) * (ryi[2]-pos->Y) < (rxi[2]-pos->X) * (ryi[1]-pos->Y))
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if (clipinsideboxline(cent.X, cent.Y, rxi[2], ryi[2], rxi[1], ryi[1], rad) != 0)
|
|
|
|
addclipline(rxi[2]-v.X, ryi[2]-v.Y, rxi[1]-v.Y, ryi[1]+v.X, obj, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
2021-12-22 09:40:26 +00:00
|
|
|
else if ((rxi[3]-pos->X) * (ryi[0]-pos->Y) < (rxi[0]-pos->X) * (ryi[3]-pos->Y))
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
if (clipinsideboxline(cent.X, cent.Y, rxi[0], ryi[0], rxi[3], ryi[3], rad) != 0)
|
|
|
|
addclipline(rxi[0]+v.X, ryi[0]+v.Y, rxi[3]+v.Y, ryi[3]-v.X, obj, false);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
}
|
2021-12-20 19:27:12 +00:00
|
|
|
|
|
|
|
if (heinum == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// the rest is for slope sprites only.
|
2022-02-02 17:37:01 +00:00
|
|
|
const int32_t tilenum = actor->spr.picnum;
|
2022-08-16 21:15:49 +00:00
|
|
|
const int32_t cosang = bcos(actor->int_ang());
|
|
|
|
const int32_t sinang = bsin(actor->int_ang());
|
2021-12-20 19:27:12 +00:00
|
|
|
vec2_t const span = { tileWidth(tilenum), tileHeight(tilenum) };
|
2022-02-02 17:37:01 +00:00
|
|
|
vec2_t const repeat = { actor->spr.xrepeat, actor->spr.yrepeat };
|
2021-12-20 19:27:12 +00:00
|
|
|
vec2_t adjofs = { tileLeftOffset(tilenum), tileTopOffset(tilenum) };
|
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
if (actor->spr.cstat & CSTAT_SPRITE_XFLIP)
|
2021-12-22 09:26:51 +00:00
|
|
|
adjofs.X = -adjofs.X;
|
2021-12-20 19:27:12 +00:00
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
if (actor->spr.cstat & CSTAT_SPRITE_YFLIP)
|
2021-12-22 09:28:51 +00:00
|
|
|
adjofs.Y = -adjofs.Y;
|
2021-12-20 19:27:12 +00:00
|
|
|
|
2021-12-22 09:26:51 +00:00
|
|
|
int32_t const centerx = ((span.X >> 1) + adjofs.X) * repeat.X;
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t const centery = ((span.Y >> 1) + adjofs.Y) * repeat.Y;
|
2021-12-22 09:26:51 +00:00
|
|
|
int32_t const rspanx = span.X * repeat.X;
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t const rspany = span.Y * repeat.Y;
|
2021-12-20 19:27:12 +00:00
|
|
|
int32_t const ratio = ksqrt(heinum * heinum + 4096 * 4096);
|
2021-12-22 09:41:47 +00:00
|
|
|
int32_t zz[3] = { pos->Z, pos->Z + flordist, pos->Z - ceildist };
|
2021-12-20 19:27:12 +00:00
|
|
|
for (int k = 0; k < 3; k++)
|
|
|
|
{
|
2022-02-02 17:37:01 +00:00
|
|
|
int32_t jj = DivScale(actor->int_pos().Z - zz[k], heinum, 18);
|
2021-12-20 19:27:12 +00:00
|
|
|
int32_t jj2 = MulScale(jj, ratio, 12);
|
|
|
|
if (jj2 > (centery << 8) || jj2 < ((centery - rspany) << 8))
|
|
|
|
continue;
|
2022-02-02 17:37:01 +00:00
|
|
|
int32_t x1 = actor->int_pos().X + MulScale(sinang, centerx, 16) + MulScale(jj, cosang, 24);
|
|
|
|
int32_t y1 = actor->int_pos().Y - MulScale(cosang, centerx, 16) + MulScale(jj, sinang, 24);
|
2021-12-20 19:27:12 +00:00
|
|
|
int32_t x2 = x1 - MulScale(sinang, rspanx, 16);
|
|
|
|
int32_t y2 = y1 + MulScale(cosang, rspanx, 16);
|
|
|
|
|
2022-08-16 21:15:49 +00:00
|
|
|
vec2_t const v = { MulScale(bcos(actor->int_ang() - 256), walldist, 14),
|
|
|
|
MulScale(bsin(actor->int_ang() - 256), walldist, 14) };
|
2021-12-20 19:27:12 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (clipinsideboxline(cent.X, cent.Y, x1, y1, x2, y2, rad) != 0)
|
2021-12-20 19:27:12 +00:00
|
|
|
{
|
2021-12-22 09:40:26 +00:00
|
|
|
if ((x1 - pos->X) * (y2 - pos->Y) >= (x2 - pos->X) * (y1 - pos->Y))
|
2021-12-20 19:27:12 +00:00
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(x1 + v.X, y1 + v.Y, x2 + v.Y, y2 - v.X, obj, false);
|
2021-12-20 19:27:12 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ((cstat & CSTAT_SPRITE_ONE_SIDE) != 0)
|
|
|
|
continue;
|
2021-12-22 09:28:51 +00:00
|
|
|
addclipline(x2 - v.X, y2 - v.Y, x1 - v.Y, y1 + v.X, obj, false);
|
2021-12-20 19:27:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-06-21 00:33:30 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (clipsectcnt < clipsectnum || clipspritecnt < clipspritenum);
|
|
|
|
|
2019-07-24 01:37:57 +00:00
|
|
|
int32_t hitwalls[4], hitwall;
|
2021-12-04 00:05:18 +00:00
|
|
|
CollisionBase clipReturn{};
|
2019-03-30 19:36:18 +00:00
|
|
|
|
2021-05-11 23:29:18 +00:00
|
|
|
int cnt = clipmoveboxtracenum;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
do
|
|
|
|
{
|
2020-01-29 11:36:13 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_NONE && (xvect|yvect))
|
|
|
|
{
|
2021-05-11 23:29:18 +00:00
|
|
|
for (int i=clipnum-1;i>=0;--i)
|
2019-07-24 01:38:01 +00:00
|
|
|
{
|
2021-12-22 09:40:26 +00:00
|
|
|
if (!bitmap_test(clipignore, i) && clipinsideboxline(pos->X, pos->Y, clipit[i].x1, clipit[i].y1, clipit[i].x2, clipit[i].y2, walldist))
|
2019-09-21 11:02:17 +00:00
|
|
|
{
|
|
|
|
vec2_t const vec = pos->vec2;
|
2021-12-22 09:40:26 +00:00
|
|
|
keepaway(&pos->X, &pos->Y, i);
|
|
|
|
if (inside_p(pos->X,pos->Y, *sectnum) != 1)
|
2019-09-21 11:02:17 +00:00
|
|
|
pos->vec2 = vec;
|
|
|
|
break;
|
|
|
|
}
|
2019-07-24 01:38:01 +00:00
|
|
|
}
|
2020-01-29 11:36:13 +00:00
|
|
|
}
|
2019-07-24 01:38:01 +00:00
|
|
|
|
2019-03-30 19:36:18 +00:00
|
|
|
vec2_t vec = goal;
|
2021-12-30 09:30:21 +00:00
|
|
|
|
2019-07-24 01:38:08 +00:00
|
|
|
if ((hitwall = cliptrace(pos->vec2, &vec)) >= 0)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-07-24 01:38:01 +00:00
|
|
|
vec2_t const clipr = { clipit[hitwall].x2 - clipit[hitwall].x1, clipit[hitwall].y2 - clipit[hitwall].y1 };
|
2020-02-11 09:21:28 +00:00
|
|
|
// clamp to the max value we can utilize without reworking the scaling below
|
|
|
|
// this works around the overflow issue that affects dukedc2.map
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t const templl = (int32_t)clamp<int64_t>(compat_maybe_truncate_to_int32((int64_t)clipr.X * clipr.X + (int64_t)clipr.Y * clipr.Y), INT32_MIN, INT32_MAX);
|
2019-07-29 11:12:06 +00:00
|
|
|
|
2020-02-11 17:26:58 +00:00
|
|
|
if (templl > 0)
|
2019-09-21 11:02:17 +00:00
|
|
|
{
|
2020-02-11 09:21:28 +00:00
|
|
|
// I don't know if this one actually overflows or not, but I highly doubt it hurts to check
|
|
|
|
int32_t const templl2
|
2021-12-22 09:28:51 +00:00
|
|
|
= (int32_t)clamp<int64_t>(compat_maybe_truncate_to_int32((int64_t)(goal.X - vec.X) * clipr.X + (int64_t)(goal.Y - vec.Y) * clipr.Y), INT32_MIN, INT32_MAX);
|
2021-01-04 12:02:00 +00:00
|
|
|
int32_t const i = (enginecompatibility_mode == ENGINECOMPATIBILITY_19950829 || (abs(templl2)>>11) < templl) ?
|
2021-05-11 23:29:18 +00:00
|
|
|
(int)DivScaleL(templl2, templl, 20) : 0;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
goal = { MulScale(clipr.X, i, 20)+vec.X, MulScale(clipr.Y, i, 20)+vec.Y };
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
2019-09-22 19:26:07 +00:00
|
|
|
int32_t tempint;
|
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_19950829)
|
2021-12-22 09:28:51 +00:00
|
|
|
tempint = clipr.X*(move.X>>6)+clipr.Y*(move.Y>>6);
|
2019-09-22 19:26:07 +00:00
|
|
|
else
|
2021-12-22 09:28:51 +00:00
|
|
|
tempint = DMulScale(clipr.X, move.X, clipr.Y, move.Y, 6);
|
2019-03-30 19:36:18 +00:00
|
|
|
|
2021-12-14 09:15:58 +00:00
|
|
|
for (int i=cnt+1, j; i<=clipmoveboxtracenum; ++i)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-05-27 05:45:41 +00:00
|
|
|
j = hitwalls[i];
|
2019-03-30 19:36:18 +00:00
|
|
|
|
2019-09-22 19:26:07 +00:00
|
|
|
int32_t tempint2;
|
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_19950829)
|
2021-12-22 09:28:51 +00:00
|
|
|
tempint2 = (clipit[j].x2-clipit[j].x1)*(move.X>>6)+(clipit[j].y2-clipit[j].y1)*(move.Y>>6);
|
2019-09-22 19:26:07 +00:00
|
|
|
else
|
2021-12-22 09:28:51 +00:00
|
|
|
tempint2 = DMulScale(clipit[j].x2-clipit[j].x1, move.X, clipit[j].y2-clipit[j].y1, move.Y, 6);
|
2019-09-22 19:26:07 +00:00
|
|
|
|
|
|
|
if ((tempint ^ tempint2) < 0)
|
2019-07-29 11:12:06 +00:00
|
|
|
{
|
2019-09-22 19:26:07 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_19961112)
|
2021-12-22 09:40:26 +00:00
|
|
|
updatesector(pos->X, pos->Y, sectnum);
|
2019-03-30 19:36:18 +00:00
|
|
|
return clipReturn;
|
2019-09-21 11:02:17 +00:00
|
|
|
}
|
2019-07-29 11:12:06 +00:00
|
|
|
}
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
keepaway(&goal.X, &goal.Y, hitwall);
|
2021-12-22 09:26:51 +00:00
|
|
|
xvect = (goal.X-vec.X)<<14;
|
2021-12-22 09:28:51 +00:00
|
|
|
yvect = (goal.Y-vec.Y)<<14;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
if (cnt == clipmoveboxtracenum)
|
2021-12-04 00:05:18 +00:00
|
|
|
clipReturn = clipobjectval[hitwall];
|
2016-06-21 00:33:30 +00:00
|
|
|
hitwalls[cnt] = hitwall;
|
|
|
|
}
|
|
|
|
|
2019-09-22 19:26:07 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_NONE)
|
2021-11-07 17:20:41 +00:00
|
|
|
{
|
2021-11-07 18:21:46 +00:00
|
|
|
clipupdatesector(vec, sectnum, rad);
|
2021-11-07 17:20:41 +00:00
|
|
|
}
|
2019-03-30 19:35:54 +00:00
|
|
|
|
2021-12-22 09:36:09 +00:00
|
|
|
pos->X = vec.X;
|
2021-12-22 09:40:26 +00:00
|
|
|
pos->Y = vec.Y;
|
2019-07-24 01:38:01 +00:00
|
|
|
cnt--;
|
2016-06-21 00:33:30 +00:00
|
|
|
} while ((xvect|yvect) != 0 && hitwall >= 0 && cnt > 0);
|
|
|
|
|
2019-09-22 19:26:07 +00:00
|
|
|
if (enginecompatibility_mode != ENGINECOMPATIBILITY_NONE)
|
2019-07-29 11:12:06 +00:00
|
|
|
{
|
2021-12-14 09:15:58 +00:00
|
|
|
for (int j=0; j<clipsectnum; j++)
|
2021-12-22 09:40:26 +00:00
|
|
|
if (inside_p(pos->X, pos->Y, clipsectorlist[j]) == 1)
|
2019-07-29 11:12:06 +00:00
|
|
|
{
|
|
|
|
*sectnum = clipsectorlist[j];
|
2019-09-21 11:02:17 +00:00
|
|
|
return clipReturn;
|
2019-07-29 11:12:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int32_t tempint2, tempint1 = INT32_MAX;
|
|
|
|
*sectnum = -1;
|
2021-12-21 09:51:41 +00:00
|
|
|
for (int j = (int)sector.Size() - 1; j >= 0; j--)
|
2021-12-17 20:42:57 +00:00
|
|
|
{
|
|
|
|
auto sect = §or[j];
|
2021-12-22 09:40:26 +00:00
|
|
|
if (inside(pos->X, pos->Y, sect) == 1)
|
2019-07-29 11:12:06 +00:00
|
|
|
{
|
2021-12-17 20:42:57 +00:00
|
|
|
if (enginecompatibility_mode != ENGINECOMPATIBILITY_19950829 && (sect->ceilingstat & CSTAT_SECTOR_SLOPE))
|
2021-12-22 09:41:47 +00:00
|
|
|
tempint2 = getceilzofslopeptr(sect, pos->X, pos->Y) - pos->Z;
|
2019-07-29 11:12:06 +00:00
|
|
|
else
|
2022-02-02 23:46:04 +00:00
|
|
|
tempint2 = sect->int_ceilingz() - pos->Z;
|
2019-07-29 11:12:06 +00:00
|
|
|
|
|
|
|
if (tempint2 > 0)
|
|
|
|
{
|
|
|
|
if (tempint2 < tempint1)
|
|
|
|
{
|
2021-12-17 20:42:57 +00:00
|
|
|
*sectnum = (int16_t)j;
|
2021-05-11 23:29:18 +00:00
|
|
|
tempint1 = tempint2;
|
2019-07-29 11:12:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-12-17 20:42:57 +00:00
|
|
|
if (enginecompatibility_mode != ENGINECOMPATIBILITY_19950829 && (sect->ceilingstat & CSTAT_SECTOR_SLOPE))
|
2021-12-22 09:41:47 +00:00
|
|
|
tempint2 = pos->Z - getflorzofslopeptr(sect, pos->X, pos->Y);
|
2019-07-29 11:12:06 +00:00
|
|
|
else
|
2022-02-02 23:46:04 +00:00
|
|
|
tempint2 = pos->Z - sect->int_floorz();
|
2019-07-29 11:12:06 +00:00
|
|
|
|
|
|
|
if (tempint2 <= 0)
|
|
|
|
{
|
2021-05-11 23:29:18 +00:00
|
|
|
*sectnum = (int16_t)j;
|
2019-07-29 11:12:06 +00:00
|
|
|
return clipReturn;
|
|
|
|
}
|
|
|
|
if (tempint2 < tempint1)
|
|
|
|
{
|
2021-05-11 23:29:18 +00:00
|
|
|
*sectnum = (int16_t)j;
|
|
|
|
tempint1 = tempint2;
|
2019-07-29 11:12:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-12-17 20:42:57 +00:00
|
|
|
}
|
2019-07-29 11:12:06 +00:00
|
|
|
}
|
|
|
|
|
2019-03-30 19:36:18 +00:00
|
|
|
return clipReturn;
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// pushmove
|
|
|
|
//
|
2021-11-26 19:41:03 +00:00
|
|
|
int pushmove_(vec3_t *const vect, int *const sectnum,
|
2019-07-24 01:37:47 +00:00
|
|
|
int32_t const walldist, int32_t const ceildist, int32_t const flordist, uint32_t const cliptype, bool clear /*= true*/)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-07-24 01:37:47 +00:00
|
|
|
int bad;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
const int32_t dawalclipmask = (cliptype&65535);
|
2021-07-04 11:05:33 +00:00
|
|
|
// const int32_t dasprclipmask = (cliptype >> 16);
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
if (*sectnum < 0)
|
|
|
|
return -1;
|
|
|
|
|
2019-07-24 01:37:47 +00:00
|
|
|
int32_t k = 32;
|
|
|
|
|
|
|
|
int dir = 1;
|
2016-06-21 00:33:30 +00:00
|
|
|
do
|
|
|
|
{
|
2019-07-24 01:37:47 +00:00
|
|
|
int32_t clipsectcnt = 0;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
|
|
|
bad = 0;
|
|
|
|
|
2019-07-24 01:37:47 +00:00
|
|
|
if (clear)
|
|
|
|
{
|
2020-01-26 13:26:08 +00:00
|
|
|
if (enginecompatibility_mode != ENGINECOMPATIBILITY_NONE && *sectnum < 0)
|
|
|
|
return 0;
|
2019-07-24 01:37:47 +00:00
|
|
|
clipsectorlist[0] = *sectnum;
|
|
|
|
clipsectnum = 1;
|
2019-04-10 09:31:12 +00:00
|
|
|
|
2021-12-06 19:08:32 +00:00
|
|
|
clipsectormap.Zero();
|
|
|
|
clipsectormap.Set(*sectnum);
|
2019-07-24 01:37:47 +00:00
|
|
|
}
|
2019-04-10 09:31:12 +00:00
|
|
|
|
2016-06-21 00:33:30 +00:00
|
|
|
do
|
|
|
|
{
|
2021-12-30 10:38:34 +00:00
|
|
|
const walltype* wal;
|
2019-04-18 17:23:30 +00:00
|
|
|
int32_t startwall, endwall;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2021-12-30 10:45:58 +00:00
|
|
|
auto sec = §or[clipsectorlist[clipsectcnt]];
|
2016-06-21 00:33:30 +00:00
|
|
|
if (dir > 0)
|
|
|
|
startwall = sec->wallptr, endwall = startwall + sec->wallnum;
|
|
|
|
else
|
2021-12-20 22:56:39 +00:00
|
|
|
endwall = sec->wallptr, startwall = endwall + sec->wallnum - 1;
|
2016-06-21 00:33:30 +00:00
|
|
|
|
2019-07-24 01:37:47 +00:00
|
|
|
int i;
|
|
|
|
|
2021-12-30 10:38:34 +00:00
|
|
|
for (i=startwall, wal=&wall[startwall]; i!=endwall; i+=dir, wal+=dir)
|
2022-08-17 21:24:44 +00:00
|
|
|
if (IsCloseToWall(DVector2(vect->X * inttoworld, vect->Y * inttoworld), &wall[i], (walldist-4) * inttoworld) == EClose::InFront)
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2019-07-24 01:37:47 +00:00
|
|
|
int j = 0;
|
2021-12-18 14:36:50 +00:00
|
|
|
if (wal->nextsector < 0 || wal->cstat & EWallFlags::FromInt(dawalclipmask)) j = 1;
|
2019-07-24 01:37:47 +00:00
|
|
|
else
|
2016-06-21 00:33:30 +00:00
|
|
|
{
|
2022-08-17 18:02:15 +00:00
|
|
|
auto pvect = NearestPointOnWall(vect->X * inttoworld, vect->Y * inttoworld, wal);
|
|
|
|
vec2_t closest = { int(pvect.X * worldtoint), int(pvect.Y * worldtoint) };
|
2021-12-30 09:30:21 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
j = cliptestsector(clipsectorlist[clipsectcnt], wal->nextsector, flordist, ceildist, closest, vect->Z);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (j != 0)
|
|
|
|
{
|
2022-09-10 07:34:11 +00:00
|
|
|
j = getangle(wal->delta());
|
2020-11-14 09:00:37 +00:00
|
|
|
int32_t dx = -bsin(j, -11);
|
|
|
|
int32_t dy = bcos(j, -11);
|
2019-07-24 01:37:47 +00:00
|
|
|
int bad2 = 16;
|
2016-06-21 00:33:30 +00:00
|
|
|
do
|
|
|
|
{
|
2021-12-22 09:40:26 +00:00
|
|
|
vect->X = (vect->X) + dx; vect->Y = (vect->Y) + dy;
|
2016-06-21 00:33:30 +00:00
|
|
|
bad2--; if (bad2 == 0) break;
|
2022-08-17 21:24:44 +00:00
|
|
|
} while (IsCloseToWall(DVector2(vect->X * inttoworld, vect->Y * inttoworld), &wall[i], (walldist - 4) * inttoworld) != EClose::Outside);
|
2016-06-21 00:33:30 +00:00
|
|
|
bad = -1;
|
2016-06-21 00:34:41 +00:00
|
|
|
k--; if (k <= 0) return bad;
|
2019-08-13 14:44:00 +00:00
|
|
|
clipupdatesector(vect->vec2, sectnum, walldist);
|
2021-12-20 16:32:11 +00:00
|
|
|
if (*sectnum < 0) return -1;
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
2021-12-06 19:08:32 +00:00
|
|
|
else if (!clipsectormap[wal->nextsector])
|
2019-04-10 09:31:12 +00:00
|
|
|
addclipsect(wal->nextsector);
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
clipsectcnt++;
|
|
|
|
} while (clipsectcnt < clipsectnum);
|
|
|
|
dir = -dir;
|
|
|
|
} while (bad != 0);
|
|
|
|
|
2016-06-21 00:34:41 +00:00
|
|
|
return bad;
|
2016-06-21 00:33:30 +00:00
|
|
|
}
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// getzrange
|
|
|
|
//
|
2021-12-04 14:37:08 +00:00
|
|
|
|
|
|
|
void getzrange(const vec3_t& pos, sectortype* sect, int32_t* ceilz, CollisionBase& ceilhit, int32_t* florz, CollisionBase& florhit, int32_t walldist, uint32_t cliptype)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-04 14:37:08 +00:00
|
|
|
if (sect == nullptr)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-04 00:14:57 +00:00
|
|
|
*ceilz = INT32_MIN; ceilhit.setVoid();
|
|
|
|
*florz = INT32_MAX; florhit.setVoid();
|
2019-03-19 17:08:59 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t clipsectcnt = 0;
|
|
|
|
|
2021-11-14 14:03:50 +00:00
|
|
|
int32_t clipspritecnt = 0;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
//Extra walldist for sprites on sector lines
|
|
|
|
const int32_t extradist = walldist+MAXCLIPDIST+1;
|
2021-12-22 09:40:26 +00:00
|
|
|
const int32_t xmin = pos.X-extradist, ymin = pos.Y-extradist;
|
|
|
|
const int32_t xmax = pos.X+extradist, ymax = pos.Y+extradist;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
const int32_t dawalclipmask = (cliptype&65535);
|
2021-07-04 11:05:33 +00:00
|
|
|
const int32_t dasprclipmask = (cliptype >> 16);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-04 14:37:08 +00:00
|
|
|
vec2_t closest = pos.vec2;
|
|
|
|
int sectnum = ::sectnum(sect);
|
2019-09-22 19:26:07 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_NONE)
|
2022-08-04 21:47:47 +00:00
|
|
|
{
|
|
|
|
DVector2 v;
|
|
|
|
SquareDistToSector(closest.X * inttoworld, closest.Y * inttoworld, §or[sectnum], &v);
|
|
|
|
closest = { int(v.X * worldtoint), int(v.Y * worldtoint) };
|
|
|
|
}
|
|
|
|
|
|
|
|
getzsofslopeptr(sect,closest.X,closest.Y,ceilz,florz);
|
2021-12-04 14:37:08 +00:00
|
|
|
ceilhit.setSector(sect);
|
|
|
|
florhit.setSector(sect);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
clipsectorlist[0] = sectnum;
|
|
|
|
clipsectnum = 1;
|
|
|
|
clipspritenum = 0;
|
2021-12-06 19:08:32 +00:00
|
|
|
clipsectormap.Zero();
|
|
|
|
clipsectormap.Set(sectnum);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
do //Collect sectors inside your square first
|
|
|
|
{
|
|
|
|
////////// Walls //////////
|
|
|
|
|
2021-12-30 10:45:58 +00:00
|
|
|
auto const startsec = §or[clipsectorlist[clipsectcnt]];
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-04 15:07:56 +00:00
|
|
|
for(auto&wal : wallsofsector(startsec))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-04 15:07:56 +00:00
|
|
|
if (wal.twoSided())
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-04 15:07:56 +00:00
|
|
|
auto nextsect = wal.nextSector();
|
2022-01-27 16:41:10 +00:00
|
|
|
vec2_t const v1 = wal.wall_int_pos();
|
|
|
|
vec2_t const v2 = wal.point2Wall()->wall_int_pos();
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:26:51 +00:00
|
|
|
if ((v1.X < xmin && (v2.X < xmin)) || (v1.X > xmax && v2.X > xmax) ||
|
2021-12-22 09:28:51 +00:00
|
|
|
(v1.Y < ymin && (v2.Y < ymin)) || (v1.Y > ymax && v2.Y > ymax))
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
vec2_t const d = { v2.X-v1.X, v2.Y-v1.Y };
|
2021-12-22 09:40:26 +00:00
|
|
|
if (d.X*(pos.Y-v1.Y) < (pos.X-v1.X)*d.Y) continue; //back
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
vec2_t da = { (d.X > 0) ? d.X*(ymin-v1.Y) : d.X*(ymax-v1.Y),
|
|
|
|
(d.Y > 0) ? d.Y*(xmax-v1.X) : d.Y*(xmin-v1.X) };
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
if (da.X >= da.Y)
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-18 14:36:50 +00:00
|
|
|
if (wal.cstat & EWallFlags::FromInt(dawalclipmask)) continue; // XXX?
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2022-02-02 23:46:04 +00:00
|
|
|
if (((nextsect->ceilingstat & CSTAT_SECTOR_SKY) == 0) && (pos.Z <= nextsect->int_ceilingz()+(3<<8))) continue;
|
|
|
|
if (((nextsect->floorstat & CSTAT_SECTOR_SKY) == 0) && (pos.Z >= nextsect->int_floorz()-(3<<8))) continue;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-04 15:07:56 +00:00
|
|
|
int nextsectno = ::sectnum(nextsect);
|
2021-12-06 19:08:32 +00:00
|
|
|
if (!clipsectormap[nextsectno])
|
2021-12-04 15:07:56 +00:00
|
|
|
addclipsect(nextsectno);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:26:51 +00:00
|
|
|
if (((v1.X < xmin + MAXCLIPDIST) && (v2.X < xmin + MAXCLIPDIST)) ||
|
|
|
|
((v1.X > xmax - MAXCLIPDIST) && (v2.X > xmax - MAXCLIPDIST)) ||
|
2021-12-22 09:28:51 +00:00
|
|
|
((v1.Y < ymin + MAXCLIPDIST) && (v2.Y < ymin + MAXCLIPDIST)) ||
|
|
|
|
((v1.Y > ymax - MAXCLIPDIST) && (v2.Y > ymax - MAXCLIPDIST)))
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-22 09:26:51 +00:00
|
|
|
if (d.X > 0) da.X += d.X*MAXCLIPDIST; else da.X -= d.X*MAXCLIPDIST;
|
2021-12-22 09:28:51 +00:00
|
|
|
if (d.Y > 0) da.Y -= d.Y*MAXCLIPDIST; else da.Y += d.Y*MAXCLIPDIST;
|
|
|
|
if (da.X >= da.Y)
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
//It actually got here, through all the continue's!!!
|
2021-10-08 17:06:41 +00:00
|
|
|
int32_t daz = 0, daz2 = 0;
|
2021-12-04 14:37:08 +00:00
|
|
|
closest = pos.vec2;
|
2019-09-22 19:26:07 +00:00
|
|
|
if (enginecompatibility_mode == ENGINECOMPATIBILITY_NONE)
|
2022-08-04 21:47:47 +00:00
|
|
|
{
|
|
|
|
DVector2 v;
|
|
|
|
SquareDistToSector(closest.X * inttoworld, closest.Y * inttoworld, §or[nextsectno], &v);
|
|
|
|
closest = { int(v.X * worldtoint), int(v.Y * worldtoint) };
|
|
|
|
}
|
|
|
|
|
|
|
|
getzsofslopeptr(nextsect, closest.X,closest.Y, &daz,&daz2);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
if (daz > *ceilz)
|
2021-12-04 15:07:56 +00:00
|
|
|
*ceilz = daz, ceilhit.setSector(nextsect);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
if (daz2 < *florz)
|
2021-12-04 15:07:56 +00:00
|
|
|
*florz = daz2, florhit.setSector(nextsect);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
clipsectcnt++;
|
|
|
|
}
|
|
|
|
while (clipsectcnt < clipsectnum || clipspritecnt < clipspritenum);
|
|
|
|
|
|
|
|
////////// Sprites //////////
|
|
|
|
|
2019-03-19 17:09:40 +00:00
|
|
|
if (dasprclipmask)
|
2021-12-14 09:15:58 +00:00
|
|
|
for (int i=0; i<clipsectnum; i++)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-11-11 19:20:00 +00:00
|
|
|
if (!validSectorIndex(clipsectorlist[i])) continue; // we got a deleted sprite in here somewhere. Skip this entry.
|
2021-12-04 00:14:57 +00:00
|
|
|
TSectIterator<DCoreActor> it(clipsectorlist[i]);
|
|
|
|
while (auto actor = it.Next())
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2022-02-02 17:37:01 +00:00
|
|
|
const int32_t cstat = actor->spr.cstat;
|
2021-11-14 14:03:50 +00:00
|
|
|
int32_t daz = 0, daz2 = 0;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
if (actor->spr.cstat2 & CSTAT2_SPRITE_NOFIND) continue;
|
2019-03-19 17:08:59 +00:00
|
|
|
if (cstat&dasprclipmask)
|
|
|
|
{
|
|
|
|
int32_t clipyou = 0;
|
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
vec2_t v1 = actor->int_pos().vec2;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
switch (cstat & CSTAT_SPRITE_ALIGNMENT_MASK)
|
|
|
|
{
|
|
|
|
case CSTAT_SPRITE_ALIGNMENT_FACING:
|
|
|
|
{
|
2022-09-09 16:07:44 +00:00
|
|
|
int32_t k = walldist+(actor->int_clipdist())+1;
|
2021-12-22 09:40:26 +00:00
|
|
|
if ((abs(v1.X-pos.X) <= k) && (abs(v1.Y-pos.Y) <= k))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2022-08-08 22:11:21 +00:00
|
|
|
daz = actor->int_pos().Z + actor->GetOffsetAndHeight(k);
|
2019-03-19 17:08:59 +00:00
|
|
|
daz2 = daz - k;
|
|
|
|
clipyou = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case CSTAT_SPRITE_ALIGNMENT_WALL:
|
|
|
|
{
|
2021-03-27 14:03:57 +00:00
|
|
|
vec2_t v2;
|
2022-02-02 17:37:01 +00:00
|
|
|
get_wallspr_points(actor, &v1.X, &v2.X, &v1.Y, &v2.Y);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if (clipinsideboxline(pos.X,pos.Y,v1.X,v1.Y,v2.X,v2.Y,walldist+1) != 0)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
|
|
|
int32_t k;
|
2022-02-02 17:37:01 +00:00
|
|
|
daz = actor->int_pos().Z + actor->GetOffsetAndHeight(k);
|
2021-03-27 14:03:57 +00:00
|
|
|
daz2 = daz-k;
|
2019-03-19 17:08:59 +00:00
|
|
|
clipyou = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case CSTAT_SPRITE_ALIGNMENT_FLOOR:
|
2021-12-20 19:27:12 +00:00
|
|
|
case CSTAT_SPRITE_ALIGNMENT_SLOPE:
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2022-08-17 19:56:14 +00:00
|
|
|
daz = spriteGetZOfSlope(&actor->spr, pos.X, pos.Y, spriteGetSlope(actor));
|
2021-12-23 20:58:45 +00:00
|
|
|
daz2 = daz;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
if ((cstat & CSTAT_SPRITE_ONE_SIDE) != 0 && (pos.Z > daz) == ((cstat & CSTAT_SPRITE_YFLIP)==0))
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
|
2021-03-27 14:03:57 +00:00
|
|
|
vec2_t v2, v3, v4;
|
2021-12-30 10:35:49 +00:00
|
|
|
get_floorspr_points(actor, pos.X, pos.Y, &v1.X, &v2.X, &v3.X, &v4.X,
|
2022-08-17 19:37:16 +00:00
|
|
|
&v1.Y, &v2.Y, &v3.Y, &v4.Y);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2022-08-16 21:15:49 +00:00
|
|
|
vec2_t const da = { MulScale(bcos(actor->int_ang() - 256), walldist + 4, 14),
|
|
|
|
MulScale(bsin(actor->int_ang() - 256), walldist + 4, 14) };
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:28:51 +00:00
|
|
|
v1.X += da.X; v2.X -= da.Y; v3.X -= da.X; v4.X += da.Y;
|
|
|
|
v1.Y += da.Y; v2.Y += da.X; v3.Y -= da.Y; v4.Y -= da.X;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-03-27 14:03:57 +00:00
|
|
|
clipyou = get_floorspr_clipyou(v1, v2, v3, v4);
|
2019-03-19 17:08:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (clipyou != 0)
|
|
|
|
{
|
2021-12-22 09:41:47 +00:00
|
|
|
if ((pos.Z > daz) && (daz > *ceilz))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
|
|
|
*ceilz = daz;
|
2021-12-04 00:14:57 +00:00
|
|
|
ceilhit.setSprite(actor);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
if ((pos.Z < daz2) && (daz2 < *florz))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
|
|
|
*florz = daz2;
|
2021-12-04 00:14:57 +00:00
|
|
|
florhit.setSprite(actor);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// intp: point of currently best (closest) intersection
|
2022-02-02 17:37:01 +00:00
|
|
|
int32_t try_facespr_intersect(DCoreActor* actor, vec3_t const in,
|
2019-08-09 09:28:27 +00:00
|
|
|
int32_t vx, int32_t vy, int32_t vz,
|
|
|
|
vec3_t * const intp, int32_t strictly_smaller_than_p)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2022-02-02 17:37:01 +00:00
|
|
|
vec3_t const sprpos = actor->int_pos();
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
int32_t const topt = vx * (sprpos.X - in.X) + vy * (sprpos.Y - in.Y);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2019-08-09 09:28:27 +00:00
|
|
|
if (topt <= 0) return 0;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2019-08-09 09:28:27 +00:00
|
|
|
int32_t const bot = vx * vx + vy * vy;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2019-08-09 09:28:27 +00:00
|
|
|
if (!bot) return 0;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
vec3_t newpos = { 0, 0, in.Z + Scale(vz, topt, bot) };
|
2019-08-09 09:28:27 +00:00
|
|
|
int32_t siz;
|
2022-02-02 17:37:01 +00:00
|
|
|
int32_t const z1 = sprpos.Z + actor->GetOffsetAndHeight(siz);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
if (newpos.Z < z1 - siz || newpos.Z > z1)
|
2019-08-09 09:28:27 +00:00
|
|
|
return 0;
|
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
int32_t const topu = vx * (sprpos.Y - in.Y) - vy * (sprpos.X - in.X);
|
2021-01-04 12:34:55 +00:00
|
|
|
vec2_t const off = { Scale(vx, topu, bot), Scale(vy, topu, bot) };
|
2021-12-22 09:28:51 +00:00
|
|
|
int32_t const dist = off.X * off.X + off.Y * off.Y;
|
2019-08-09 09:28:27 +00:00
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
siz = tileWidth(actor->spr.picnum) * actor->spr.xrepeat;
|
2019-08-09 09:28:27 +00:00
|
|
|
|
2021-01-04 11:36:54 +00:00
|
|
|
if (dist > MulScale(siz, siz, 7)) return 0;
|
2019-08-09 09:28:27 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
newpos.vec2 = { in.X + Scale(vx, topt, bot), in.Y + Scale(vy, topt, bot) };
|
2019-08-09 09:28:27 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if (abs(newpos.X - in.X) + abs(newpos.Y - in.Y) + strictly_smaller_than_p >
|
|
|
|
abs(intp->X - in.X) + abs(intp->Y - in.Y))
|
2019-08-09 09:28:27 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
*intp = newpos;
|
|
|
|
return 1;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
static inline void hit_set(HitInfoBase *hit, sectortype* sect, walltype* wal, DCoreActor* actor, int32_t x, int32_t y, int32_t z)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-03 23:47:08 +00:00
|
|
|
hit->hitSector = sect;
|
|
|
|
hit->hitWall = wal;
|
|
|
|
hit->hitActor = actor;
|
2022-09-03 08:47:36 +00:00
|
|
|
hit->hitpos.X = x * inttoworld;
|
|
|
|
hit->hitpos.Y = y * inttoworld;
|
|
|
|
hit->hitpos.Z = z * zinttoworld;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int32_t hitscan_hitsectcf=-1;
|
|
|
|
|
|
|
|
// stat, heinum, z: either ceiling- or floor-
|
|
|
|
// how: -1: behave like ceiling, 1: behave like floor
|
2021-12-03 23:47:08 +00:00
|
|
|
static int32_t hitscan_trysector(const vec3_t *sv, sectortype* sec, HitInfoBase *hit,
|
2019-03-19 17:08:59 +00:00
|
|
|
int32_t vx, int32_t vy, int32_t vz,
|
2021-12-03 23:47:08 +00:00
|
|
|
uint16_t stat, int16_t heinum, int32_t z, int32_t how)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-11-14 14:03:50 +00:00
|
|
|
int32_t x1 = INT32_MAX, y1 = 0, z1 = 0;
|
2019-03-19 17:08:59 +00:00
|
|
|
int32_t i;
|
|
|
|
|
|
|
|
if (stat&2)
|
|
|
|
{
|
2021-12-30 10:38:34 +00:00
|
|
|
auto const wal = sec->firstWall();
|
|
|
|
auto const wal2 = wal->point2Wall();
|
2022-01-27 16:41:10 +00:00
|
|
|
int32_t j, dax=wal2->wall_int_pos().X-wal->wall_int_pos().X, day=wal2->wall_int_pos().Y-wal->wall_int_pos().Y;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-03-14 22:38:39 +00:00
|
|
|
i = ksqrt(compat_maybe_truncate_to_int32(uhypsq(dax,day))); if (i == 0) return 1; //continue;
|
2021-01-04 11:51:41 +00:00
|
|
|
i = DivScale(heinum,i, 15);
|
2019-03-19 17:08:59 +00:00
|
|
|
dax *= i; day *= i;
|
|
|
|
|
2021-01-04 10:40:08 +00:00
|
|
|
j = (vz<<8)-DMulScale(dax,vy,-day,vx, 15);
|
2019-03-19 17:08:59 +00:00
|
|
|
if (j != 0)
|
|
|
|
{
|
2022-01-27 16:41:10 +00:00
|
|
|
i = ((z - sv->Z)<<8)+DMulScale(dax,sv->Y-wal->wall_int_pos().Y,-day,sv->X-wal->wall_int_pos().X, 15);
|
2021-01-04 12:02:00 +00:00
|
|
|
if (((i^j) >= 0) && ((abs(i)>>1) < abs(j)))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-01-04 11:51:41 +00:00
|
|
|
i = DivScale(i,j, 30);
|
2021-12-22 09:36:09 +00:00
|
|
|
x1 = sv->X + MulScale(vx,i, 30);
|
2021-12-22 09:40:26 +00:00
|
|
|
y1 = sv->Y + MulScale(vy,i, 30);
|
2021-12-22 09:41:47 +00:00
|
|
|
z1 = sv->Z + MulScale(vz,i, 30);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-12-22 09:41:47 +00:00
|
|
|
else if ((how*vz > 0) && (how*sv->Z <= how*z))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-22 09:41:47 +00:00
|
|
|
z1 = z; i = z1-sv->Z;
|
2021-01-04 12:02:00 +00:00
|
|
|
if ((abs(i)>>1) < vz*how)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-01-04 11:51:41 +00:00
|
|
|
i = DivScale(i,vz, 30);
|
2021-12-22 09:36:09 +00:00
|
|
|
x1 = sv->X + MulScale(vx,i, 30);
|
2021-12-22 09:40:26 +00:00
|
|
|
y1 = sv->Y + MulScale(vy,i, 30);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-16 22:59:32 +00:00
|
|
|
if ((x1 != INT32_MAX) && (abs(x1-sv->X)+abs(y1-sv->Y) < abs((hit->int_hitpos().X)-sv->X)+abs((hit->int_hitpos().Y)-sv->Y)))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-06 17:17:45 +00:00
|
|
|
if (inside(x1,y1,sec) == 1)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-03 23:47:08 +00:00
|
|
|
hit_set(hit, sec, nullptr, nullptr, x1, y1, z1);
|
|
|
|
hitscan_hitsectcf = (how+1)>>1;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
|
2019-03-19 17:08:59 +00:00
|
|
|
//
|
|
|
|
// hitscan
|
|
|
|
//
|
2021-12-03 23:47:08 +00:00
|
|
|
|
|
|
|
int hitscan(const vec3_t& start, const sectortype* startsect, const vec3_t& direction, HitInfoBase& hitinfo, unsigned cliptype)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-03 23:47:08 +00:00
|
|
|
auto const sv = &start;
|
2021-12-22 09:41:47 +00:00
|
|
|
int const vx = direction.X, vy = direction.Y, vz = direction.Z;
|
2019-03-19 17:08:59 +00:00
|
|
|
int32_t x1, y1=0, z1=0, x2, y2, intx, inty, intz;
|
|
|
|
int32_t i, k, daz;
|
|
|
|
|
|
|
|
const int32_t dawalclipmask = (cliptype&65535);
|
2021-07-04 11:05:33 +00:00
|
|
|
const int32_t dasprclipmask = (cliptype >> 16);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
hitinfo.clearObj(); // note that this case leaves hitpos untouched.
|
|
|
|
if (startsect == nullptr)
|
2019-03-19 17:08:59 +00:00
|
|
|
return -1;
|
|
|
|
|
2022-09-03 08:47:36 +00:00
|
|
|
hitinfo.hitpos.X = hitscangoal.X * inttoworld;
|
|
|
|
hitinfo.hitpos.Y = hitscangoal.Y * inttoworld;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
BFSSectorSearch search(startsect);
|
|
|
|
while (auto sec = search.GetNext())
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
|
|
|
i = 1;
|
2022-02-02 23:46:04 +00:00
|
|
|
if (hitscan_trysector(sv, sec, &hitinfo, vx,vy,vz, sec->ceilingstat, sec->ceilingheinum, sec->int_ceilingz(), -i))
|
2021-12-04 13:41:07 +00:00
|
|
|
continue;
|
2022-02-02 23:46:04 +00:00
|
|
|
if (hitscan_trysector(sv, sec, &hitinfo, vx,vy,vz, sec->floorstat, sec->floorheinum, sec->int_floorz(), i))
|
2021-12-04 13:41:07 +00:00
|
|
|
continue;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
////////// Walls //////////
|
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
for(auto& w : wallsofsector(sec))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-03 23:47:08 +00:00
|
|
|
auto wal = &w;
|
|
|
|
auto wal2 = wal->point2Wall();
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
auto const nextsect = wal->nextSector();
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2022-01-27 16:41:10 +00:00
|
|
|
x1 = wal->wall_int_pos().X; y1 = wal->wall_int_pos().Y; x2 = wal2->wall_int_pos().X; y2 = wal2->wall_int_pos().Y;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:40:26 +00:00
|
|
|
if (compat_maybe_truncate_to_int32((coord_t)(x1-sv->X)*(y2-sv->Y))
|
|
|
|
< compat_maybe_truncate_to_int32((coord_t)(x2-sv->X)*(y1-sv->Y))) continue;
|
2021-12-22 09:41:47 +00:00
|
|
|
if (rintersect(sv->X,sv->Y,sv->Z, vx,vy,vz, x1,y1, x2,y2, &intx,&inty,&intz) == -1) continue;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2022-08-16 22:59:32 +00:00
|
|
|
if (abs(intx-sv->X)+abs(inty-sv->Y) >= abs((hitinfo.int_hitpos().X)-sv->X)+abs((hitinfo.int_hitpos().Y)-sv->Y))
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-30 11:29:17 +00:00
|
|
|
if ((!wal->twoSided()) || (wal->cstat & EWallFlags::FromInt(dawalclipmask)))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-30 11:29:17 +00:00
|
|
|
hit_set(&hitinfo, sec, wal, nullptr, intx, inty, intz);
|
|
|
|
continue;
|
|
|
|
}
|
2019-09-22 19:26:07 +00:00
|
|
|
|
2021-12-30 11:29:17 +00:00
|
|
|
int32_t daz2;
|
|
|
|
getzsofslopeptr(nextsect,intx,inty,&daz,&daz2);
|
|
|
|
if (intz <= daz || intz >= daz2)
|
|
|
|
{
|
|
|
|
hit_set(&hitinfo, sec, wal, nullptr, intx, inty, intz);
|
|
|
|
continue;
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
2021-12-30 11:29:17 +00:00
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
search.Add(nextsect);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
////////// Sprites //////////
|
|
|
|
|
|
|
|
if (dasprclipmask==0)
|
|
|
|
continue;
|
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
TSectIterator<DCoreActor> it(sec);
|
|
|
|
while (auto actor = it.Next())
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2022-02-02 17:37:01 +00:00
|
|
|
uint32_t const cstat = actor->spr.cstat;
|
2021-04-17 07:37:38 +00:00
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
if (actor->spr.cstat2 & CSTAT2_SPRITE_NOFIND)
|
2021-04-17 07:37:38 +00:00
|
|
|
continue;
|
|
|
|
|
2022-08-17 19:37:16 +00:00
|
|
|
if ((cstat&dasprclipmask) == 0)
|
|
|
|
continue;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
x1 = actor->int_pos().X; y1 = actor->int_pos().Y; z1 = actor->int_pos().Z;
|
2021-12-18 15:39:44 +00:00
|
|
|
switch (cstat&CSTAT_SPRITE_ALIGNMENT_MASK)
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
{
|
2022-08-17 16:53:45 +00:00
|
|
|
auto v = hitinfo.int_hitpos();
|
|
|
|
if (try_facespr_intersect(actor, *sv, vx, vy, vz, &v, 0))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2022-09-03 08:47:36 +00:00
|
|
|
hit_set(&hitinfo, sec, nullptr, actor, v.X, v.Y, v.Z);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-03-30 19:36:21 +00:00
|
|
|
case CSTAT_SPRITE_ALIGNMENT_WALL:
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
|
|
|
int32_t ucoefup16;
|
2022-02-02 17:37:01 +00:00
|
|
|
int32_t tilenum = actor->spr.picnum;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
get_wallspr_points(actor, &x1, &x2, &y1, &y2);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-18 15:39:44 +00:00
|
|
|
if ((cstat & CSTAT_SPRITE_ONE_SIDE) != 0) //back side of 1-way sprite
|
2021-12-22 09:40:26 +00:00
|
|
|
if (compat_maybe_truncate_to_int32((coord_t)(x1-sv->X)*(y2-sv->Y))
|
|
|
|
< compat_maybe_truncate_to_int32((coord_t)(x2-sv->X)*(y1-sv->Y))) continue;
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
ucoefup16 = rintersect(sv->X,sv->Y,sv->Z,vx,vy,vz,x1,y1,x2,y2,&intx,&inty,&intz);
|
2019-03-19 17:08:59 +00:00
|
|
|
if (ucoefup16 == -1) continue;
|
|
|
|
|
2022-08-16 22:59:32 +00:00
|
|
|
if (abs(intx-sv->X)+abs(inty-sv->Y) > abs((hitinfo.int_hitpos().X)-sv->X)+abs((hitinfo.int_hitpos().Y)-sv->Y))
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
|
2022-02-02 17:37:01 +00:00
|
|
|
daz = actor->int_pos().Z + actor->GetOffsetAndHeight(k);
|
2019-03-19 17:08:59 +00:00
|
|
|
if (intz > daz-k && intz < daz)
|
|
|
|
{
|
|
|
|
if (picanm[tilenum].sf&PICANM_TEXHITSCAN_BIT)
|
|
|
|
{
|
2022-08-05 16:47:35 +00:00
|
|
|
tileUpdatePicnum(&tilenum);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2019-10-15 21:18:52 +00:00
|
|
|
if (tileLoad(tilenum))
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
|
|
|
// daz-intz > 0 && daz-intz < k
|
2021-01-04 11:36:54 +00:00
|
|
|
int32_t xtex = MulScale(ucoefup16, tileWidth(tilenum), 16);
|
2021-01-04 11:51:41 +00:00
|
|
|
int32_t vcoefup16 = 65536-DivScale(daz-intz, k, 16);
|
2021-01-04 11:36:54 +00:00
|
|
|
int32_t ytex = MulScale(vcoefup16, tileHeight(tilenum), 16);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2020-11-20 07:18:26 +00:00
|
|
|
auto texel = (tilePtr(tilenum) + tileHeight(tilenum)*xtex + ytex);
|
2020-04-11 22:04:02 +00:00
|
|
|
if (*texel == TRANSPARENT_INDEX)
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-03 23:47:08 +00:00
|
|
|
hit_set(&hitinfo, sec, nullptr, actor, intx, inty, intz);
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-03-30 19:36:21 +00:00
|
|
|
case CSTAT_SPRITE_ALIGNMENT_FLOOR:
|
2019-03-19 17:08:59 +00:00
|
|
|
{
|
2021-12-04 13:41:07 +00:00
|
|
|
int32_t x3, y3, x4, y4;
|
2019-03-19 17:08:59 +00:00
|
|
|
intz = z1;
|
2019-08-27 06:52:42 +00:00
|
|
|
|
2021-12-22 09:41:47 +00:00
|
|
|
if (vz == 0 || ((intz-sv->Z)^vz) < 0) continue;
|
2019-08-27 06:52:42 +00:00
|
|
|
|
2021-12-18 15:39:44 +00:00
|
|
|
if ((cstat & CSTAT_SPRITE_ONE_SIDE) != 0)
|
2021-12-22 09:41:47 +00:00
|
|
|
if ((sv->Z > intz) == ((cstat & CSTAT_SPRITE_YFLIP)==0)) continue;
|
2021-12-04 13:41:07 +00:00
|
|
|
|
|
|
|
// avoid overflow errors by using 64 bit math.
|
2021-12-22 09:41:47 +00:00
|
|
|
intx = int(sv->X + (int64_t(intz) - sv->Z) * vx / vz);
|
|
|
|
inty = int(sv->Y + (int64_t(intz) - sv->Z) * vy / vz);
|
2019-07-29 15:43:16 +00:00
|
|
|
|
2022-08-16 22:59:32 +00:00
|
|
|
if (abs(intx-sv->X)+abs(inty-sv->Y) > abs((hitinfo.int_hitpos().X)-sv->X)+abs((hitinfo.int_hitpos().Y)-sv->Y))
|
2019-03-19 17:08:59 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-30 10:35:49 +00:00
|
|
|
get_floorspr_points(actor, intx, inty, &x1, &x2, &x3, &x4,
|
2022-08-17 19:37:16 +00:00
|
|
|
&y1, &y2, &y3, &y4);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
2021-03-27 14:03:57 +00:00
|
|
|
if (get_floorspr_clipyou({x1, y1}, {x2, y2}, {x3, y3}, {x4, y4}))
|
2021-12-03 23:47:08 +00:00
|
|
|
hit_set(&hitinfo, sec, nullptr, actor, intx, inty, intz);
|
2019-03-19 17:08:59 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2021-12-20 19:27:12 +00:00
|
|
|
|
|
|
|
case CSTAT_SPRITE_ALIGNMENT_SLOPE:
|
|
|
|
{
|
2021-12-21 12:46:11 +00:00
|
|
|
int32_t x3, y3, x4, y4;
|
2021-12-30 11:21:51 +00:00
|
|
|
int32_t const heinum = spriteGetSlope(actor);
|
2022-08-16 21:15:49 +00:00
|
|
|
int32_t const dax = (heinum * sintable[(actor->int_ang() + 1024) & 2047]) << 1;
|
|
|
|
int32_t const day = (heinum * sintable[(actor->int_ang() + 512) & 2047]) << 1;
|
2021-12-20 19:27:12 +00:00
|
|
|
int32_t const j = (vz << 8) - DMulScale(dax, vy, -day, vx, 15);
|
|
|
|
if (j == 0) continue;
|
|
|
|
if ((cstat & 64) != 0)
|
|
|
|
if ((j < 0) == ((cstat & 8) == 0)) continue;
|
2022-02-02 17:37:01 +00:00
|
|
|
int32_t dist2 = ((actor->int_pos().Z - sv->Z) << 8) + DMulScale(dax, sv->Y - actor->int_pos().Y, -day, sv->X - actor->int_pos().X, 15);
|
2021-12-24 08:19:43 +00:00
|
|
|
if ((dist2 ^ j) < 0 || (abs(dist2) >> 1) >= abs(j)) continue;
|
2021-12-20 19:27:12 +00:00
|
|
|
|
2021-12-24 08:19:43 +00:00
|
|
|
dist2 = DivScale(dist2, j, 30);
|
|
|
|
intx = sv->X + MulScale(vx, dist2, 30);
|
|
|
|
inty = sv->Y + MulScale(vy, dist2, 30);
|
|
|
|
intz = sv->Z + MulScale(vz, dist2, 30);
|
2021-12-20 19:27:12 +00:00
|
|
|
|
2022-08-16 22:59:32 +00:00
|
|
|
if (abs(intx - sv->X) + abs(inty - sv->Y) > abs((hitinfo.int_hitpos().X) - sv->X) + abs((hitinfo.int_hitpos().Y) - sv->Y))
|
2021-12-20 19:27:12 +00:00
|
|
|
continue;
|
|
|
|
|
2021-12-30 10:35:49 +00:00
|
|
|
get_floorspr_points(actor, intx, inty, &x1, &x2, &x3, &x4,
|
2022-08-17 19:37:16 +00:00
|
|
|
&y1, &y2, &y3, &y4);
|
2021-12-20 19:27:12 +00:00
|
|
|
|
|
|
|
if (get_floorspr_clipyou({ x1, y1 }, { x2, y2 }, { x3, y3 }, { x4, y4 }))
|
|
|
|
hit_set(&hitinfo, sec, nullptr, actor, intx, inty, intz);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-03-19 17:08:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|