#ifndef __GL_CLIPPER #define __GL_CLIPPER #include "doomtype.h" #include "xs_Float.h" #include "r_utility.h" #include "memarena.h" angle_t R_PointToPseudoAngle(double x, double y); // Used to speed up angle calculations during clipping inline angle_t vertex_t::GetClipAngle() { return R_PointToPseudoAngle(p.X, p.Y); } class ClipNode { friend class Clipper; ClipNode *prev, *next; angle_t start, end; bool operator== (const ClipNode &other) { return other.start == start && other.end == end; } }; class Clipper { unsigned starttime; TStaticArray anglecache; FMemArena nodearena; ClipNode * freelist = nullptr; ClipNode * clipnodes = nullptr; ClipNode * cliphead = nullptr; ClipNode * silhouette = nullptr; // will be preserved even when RemoveClipRange is called bool blocked = false; 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: Clipper(); void Clear(); void Free(ClipNode *node) { node->next = freelist; freelist = node; } ClipNode * GetNew() { if (freelist) { ClipNode * p = freelist; freelist = p->next; return p; } else return (ClipNode*)nodearena.Alloc(sizeof(ClipNode)); } ClipNode * NewRange(angle_t start, angle_t end) { ClipNode * c = GetNew(); c->start = start; c->end = end; c->next = c->prev = NULL; return c; } 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)); } void SetBlocked(bool on) { blocked = on; } bool IsBlocked() const { return blocked; } bool CheckBox(const float *bspcoord); // Used to speed up angle calculations during clipping inline angle_t GetClipAngle(vertex_t *v) { return v->angletime == starttime ? v->viewangle : (v->angletime = starttime, v->viewangle = R_PointToPseudoAngle(v->p.X, v->p.Y)); } }; #endif