mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-07 18:20:49 +00:00
152 lines
No EOL
3 KiB
C++
152 lines
No EOL
3 KiB
C++
#ifndef __GL_CLIPPER
|
|
#define __GL_CLIPPER
|
|
|
|
#include "doomtype.h"
|
|
#include "tables.h"
|
|
#include "xs_Float.h"
|
|
#include "r_utility.h"
|
|
|
|
class ClipNode
|
|
{
|
|
friend class Clipper;
|
|
friend class ClipNodesFreer;
|
|
|
|
ClipNode *prev, *next;
|
|
angle_t start, end;
|
|
static ClipNode * freelist;
|
|
|
|
bool operator== (const ClipNode &other)
|
|
{
|
|
return other.start == start && other.end == end;
|
|
}
|
|
|
|
void Free()
|
|
{
|
|
next=freelist;
|
|
freelist=this;
|
|
}
|
|
|
|
static ClipNode * GetNew()
|
|
{
|
|
if (freelist)
|
|
{
|
|
ClipNode * p=freelist;
|
|
freelist=p->next;
|
|
return p;
|
|
}
|
|
else return new ClipNode;
|
|
}
|
|
|
|
static ClipNode * NewRange(angle_t start, angle_t end)
|
|
{
|
|
ClipNode * c=GetNew();
|
|
|
|
c->start=start;
|
|
c->end=end;
|
|
c->next=c->prev=NULL;
|
|
return c;
|
|
}
|
|
|
|
};
|
|
|
|
|
|
class Clipper
|
|
{
|
|
ClipNode * clipnodes;
|
|
ClipNode * cliphead;
|
|
ClipNode * silhouette; // will be preserved even when RemoveClipRange is called
|
|
|
|
static angle_t AngleToPseudo(angle_t ang);
|
|
bool IsRangeVisible(angle_t startangle, angle_t endangle);
|
|
void RemoveRange(ClipNode * cn);
|
|
void AddClipRange(angle_t startangle, angle_t endangle);
|
|
void RemoveClipRange(angle_t startangle, angle_t endangle);
|
|
void DoRemoveClipRange(angle_t start, angle_t end);
|
|
|
|
public:
|
|
|
|
static int anglecache;
|
|
|
|
Clipper()
|
|
{
|
|
clipnodes=cliphead=NULL;
|
|
}
|
|
|
|
~Clipper();
|
|
|
|
void Clear();
|
|
|
|
|
|
void SetSilhouette();
|
|
|
|
bool SafeCheckRange(angle_t startAngle, angle_t endAngle)
|
|
{
|
|
if(startAngle > endAngle)
|
|
{
|
|
return (IsRangeVisible(startAngle, ANGLE_MAX) || IsRangeVisible(0, endAngle));
|
|
}
|
|
|
|
return IsRangeVisible(startAngle, endAngle);
|
|
}
|
|
|
|
void SafeAddClipRange(angle_t startangle, angle_t endangle)
|
|
{
|
|
if(startangle > endangle)
|
|
{
|
|
// The range has to added in two parts.
|
|
AddClipRange(startangle, ANGLE_MAX);
|
|
AddClipRange(0, endangle);
|
|
}
|
|
else
|
|
{
|
|
// Add the range as usual.
|
|
AddClipRange(startangle, endangle);
|
|
}
|
|
}
|
|
|
|
void SafeAddClipRangeRealAngles(angle_t startangle, angle_t endangle)
|
|
{
|
|
SafeAddClipRange(AngleToPseudo(startangle), AngleToPseudo(endangle));
|
|
}
|
|
|
|
|
|
void SafeRemoveClipRange(angle_t startangle, angle_t endangle)
|
|
{
|
|
if(startangle > endangle)
|
|
{
|
|
// The range has to added in two parts.
|
|
RemoveClipRange(startangle, ANGLE_MAX);
|
|
RemoveClipRange(0, endangle);
|
|
}
|
|
else
|
|
{
|
|
// Add the range as usual.
|
|
RemoveClipRange(startangle, endangle);
|
|
}
|
|
}
|
|
|
|
void SafeRemoveClipRangeRealAngles(angle_t startangle, angle_t endangle)
|
|
{
|
|
SafeRemoveClipRange(AngleToPseudo(startangle), AngleToPseudo(endangle));
|
|
}
|
|
|
|
bool CheckBox(const fixed_t *bspcoord);
|
|
};
|
|
|
|
|
|
extern Clipper clipper;
|
|
|
|
angle_t R_PointToPseudoAngle (fixed_t viewx, fixed_t viewy, fixed_t x, fixed_t y);
|
|
|
|
inline angle_t R_PointToAnglePrecise (fixed_t viewx, fixed_t viewy, fixed_t x, fixed_t y)
|
|
{
|
|
return xs_RoundToUInt(atan2(double(y-viewy), double(x-viewx)) * (ANGLE_180/M_PI));
|
|
}
|
|
|
|
// Used to speed up angle calculations during clipping
|
|
inline angle_t vertex_t::GetClipAngle()
|
|
{
|
|
return angletime == Clipper::anglecache? viewangle : (angletime = Clipper::anglecache, viewangle = R_PointToPseudoAngle(viewx, viewy, x,y));
|
|
}
|
|
|
|
#endif |