- fixed player visibility determination by tracking and translating the actual actor position in addition to the camera position through portals. I hope this fixes the randomly appearing player sprites for good.

This commit is contained in:
Christoph Oelckers 2016-12-29 22:19:09 +01:00
parent f52744e8a4
commit a000b57204
6 changed files with 35 additions and 36 deletions

View file

@ -283,11 +283,11 @@ bool GLPortal::Start(bool usestencil, bool doquery)
// save viewpoint // save viewpoint
savedViewPos = ViewPos; savedViewPos = ViewPos;
savedViewActorPos = ViewActorPos;
savedshowviewer = r_showviewer;
savedAngle = ViewAngle; savedAngle = ViewAngle;
savedviewactor=GLRenderer->mViewActor; savedviewactor=GLRenderer->mViewActor;
savedviewarea=in_area; savedviewarea=in_area;
savedviewpath[0] = ViewPath[0];
savedviewpath[1] = ViewPath[1];
savedvisibility = camera ? camera->renderflags & RF_INVISIBLE : ActorRenderFlags::FromInt(0); savedvisibility = camera ? camera->renderflags & RF_INVISIBLE : ActorRenderFlags::FromInt(0);
@ -350,9 +350,9 @@ void GLPortal::End(bool usestencil)
if (needdepth) FDrawInfo::EndDrawInfo(); if (needdepth) FDrawInfo::EndDrawInfo();
// Restore the old view // Restore the old view
ViewPath[0] = savedviewpath[0];
ViewPath[1] = savedviewpath[1];
ViewPos = savedViewPos; ViewPos = savedViewPos;
r_showviewer = savedshowviewer;
ViewActorPos = savedViewActorPos;
ViewAngle = savedAngle; ViewAngle = savedAngle;
GLRenderer->mViewActor=savedviewactor; GLRenderer->mViewActor=savedviewactor;
in_area=savedviewarea; in_area=savedviewarea;
@ -409,6 +409,8 @@ void GLPortal::End(bool usestencil)
glDepthMask(true); glDepthMask(true);
} }
// Restore the old view // Restore the old view
r_showviewer = savedshowviewer;
ViewActorPos = savedViewActorPos;
ViewPos = savedViewPos; ViewPos = savedViewPos;
ViewAngle = savedAngle; ViewAngle = savedAngle;
GLRenderer->mViewActor=savedviewactor; GLRenderer->mViewActor=savedviewactor;
@ -628,6 +630,7 @@ void GLSkyboxPortal::DrawContents()
bool oldclamp = gl_RenderState.SetDepthClamp(false); bool oldclamp = gl_RenderState.SetDepthClamp(false);
ViewPos = origin->InterpolatedPosition(r_TicFracF); ViewPos = origin->InterpolatedPosition(r_TicFracF);
ViewActorPos = origin->Pos();
ViewAngle += (origin->PrevAngles.Yaw + deltaangle(origin->PrevAngles.Yaw, origin->Angles.Yaw) * r_TicFracF); ViewAngle += (origin->PrevAngles.Yaw + deltaangle(origin->PrevAngles.Yaw, origin->Angles.Yaw) * r_TicFracF);
// Don't let the viewpoint be too close to a floor or ceiling // Don't let the viewpoint be too close to a floor or ceiling
@ -729,6 +732,7 @@ void GLSectorStackPortal::DrawContents()
FPortal *portal = origin; FPortal *portal = origin;
ViewPos += origin->mDisplacement; ViewPos += origin->mDisplacement;
ViewActorPos += origin->mDisplacement;
GLRenderer->mViewActor = NULL; GLRenderer->mViewActor = NULL;
// avoid recursions! // avoid recursions!
@ -772,7 +776,7 @@ void GLSectorStackPortal::DrawContents()
void GLPlaneMirrorPortal::DrawContents() void GLPlaneMirrorPortal::DrawContents()
{ {
if (renderdepth>r_mirror_recursions) if (renderdepth > r_mirror_recursions)
{ {
ClearScreen(); ClearScreen();
return; return;
@ -780,7 +784,10 @@ void GLPlaneMirrorPortal::DrawContents()
// A plane mirror needs to flip the portal exclusion logic because inside the mirror, up is down and down is up. // A plane mirror needs to flip the portal exclusion logic because inside the mirror, up is down and down is up.
std::swap(instack[sector_t::floor], instack[sector_t::ceiling]); std::swap(instack[sector_t::floor], instack[sector_t::ceiling]);
int old_pm=PlaneMirrorMode; int old_pm = PlaneMirrorMode;
// the player is always visible in a mirror.
r_showviewer = true;
double planez = origin->ZatPoint(ViewPos); double planez = origin->ZatPoint(ViewPos);
ViewPos.Z = 2 * planez - ViewPos.Z; ViewPos.Z = 2 * planez - ViewPos.Z;
@ -788,14 +795,14 @@ void GLPlaneMirrorPortal::DrawContents()
PlaneMirrorMode = origin->fC() < 0 ? -1 : 1; PlaneMirrorMode = origin->fC() < 0 ? -1 : 1;
PlaneMirrorFlag++; PlaneMirrorFlag++;
GLRenderer->SetupView(ViewPos.X, ViewPos.Y, ViewPos.Z, ViewAngle, !!(MirrorFlag&1), !!(PlaneMirrorFlag&1)); GLRenderer->SetupView(ViewPos.X, ViewPos.Y, ViewPos.Z, ViewAngle, !!(MirrorFlag & 1), !!(PlaneMirrorFlag & 1));
ClearClipper(); ClearClipper();
gl_RenderState.SetClipHeight(planez, PlaneMirrorMode < 0? -1.f : 1.f); gl_RenderState.SetClipHeight(planez, PlaneMirrorMode < 0 ? -1.f : 1.f);
GLRenderer->DrawScene(DM_PORTAL); GLRenderer->DrawScene(DM_PORTAL);
gl_RenderState.SetClipHeight(0.f, 0.f); gl_RenderState.SetClipHeight(0.f, 0.f);
PlaneMirrorFlag--; PlaneMirrorFlag--;
PlaneMirrorMode=old_pm; PlaneMirrorMode = old_pm;
std::swap(instack[sector_t::floor], instack[sector_t::ceiling]); std::swap(instack[sector_t::floor], instack[sector_t::ceiling]);
} }
@ -903,6 +910,8 @@ void GLMirrorPortal::DrawContents()
vertex_t *v1 = linedef->v1; vertex_t *v1 = linedef->v1;
vertex_t *v2 = linedef->v2; vertex_t *v2 = linedef->v2;
// the player is always visible in a mirror.
r_showviewer = true;
// Reflect the current view behind the mirror. // Reflect the current view behind the mirror.
if (linedef->Delta().X == 0) if (linedef->Delta().X == 0)
{ {
@ -999,25 +1008,9 @@ void GLLineToLinePortal::DrawContents()
line_t *origin = glport->lines[0]->mOrigin; line_t *origin = glport->lines[0]->mOrigin;
P_TranslatePortalXY(origin, ViewPos.X, ViewPos.Y); P_TranslatePortalXY(origin, ViewPos.X, ViewPos.Y);
P_TranslatePortalXY(origin, ViewActorPos.X, ViewActorPos.Y);
P_TranslatePortalAngle(origin, ViewAngle); P_TranslatePortalAngle(origin, ViewAngle);
P_TranslatePortalZ(origin, ViewPos.Z); P_TranslatePortalZ(origin, ViewPos.Z);
P_TranslatePortalXY(origin, ViewPath[0].X, ViewPath[0].Y);
P_TranslatePortalXY(origin, ViewPath[1].X, ViewPath[1].Y);
if (!r_showviewer && camera != nullptr && P_PointOnLineSidePrecise(ViewPath[0], glport->lines[0]->mDestination) != P_PointOnLineSidePrecise(ViewPath[1], glport->lines[0]->mDestination))
{
double distp = (ViewPath[0] - ViewPath[1]).Length();
if (distp > EQUAL_EPSILON)
{
double dist1 = (ViewPos - ViewPath[0]).Length();
double dist2 = (ViewPos - ViewPath[1]).Length();
if (dist1 + dist2 < distp + 1)
{
camera->renderflags |= RF_INVISIBLE;
}
}
}
SaveMapSection(); SaveMapSection();

View file

@ -102,11 +102,12 @@ private:
void DrawPortalStencil(); void DrawPortalStencil();
DVector3 savedViewPos; DVector3 savedViewPos;
DVector3 savedViewActorPos;
DAngle savedAngle; DAngle savedAngle;
bool savedshowviewer;
AActor * savedviewactor; AActor * savedviewactor;
area_t savedviewarea; area_t savedviewarea;
ActorRenderFlags savedvisibility; ActorRenderFlags savedvisibility;
DVector3 savedviewpath[2];
GLPortal *PrevPortal; GLPortal *PrevPortal;
GLPortal *PrevClipPortal; GLPortal *PrevClipPortal;
TArray<BYTE> savedmapsection; TArray<BYTE> savedmapsection;

View file

@ -498,10 +498,6 @@ void FGLRenderer::DrawScene(int drawmode)
if (camera != nullptr) if (camera != nullptr)
{ {
ActorRenderFlags savedflags = camera->renderflags; ActorRenderFlags savedflags = camera->renderflags;
if (drawmode != DM_PORTAL && !r_showviewer)
{
camera->renderflags |= RF_INVISIBLE;
}
CreateScene(); CreateScene();
camera->renderflags = savedflags; camera->renderflags = savedflags;
} }

View file

@ -75,6 +75,7 @@ EXTERN_CVAR (Float, transsouls)
extern TArray<spritedef_t> sprites; extern TArray<spritedef_t> sprites;
extern TArray<spriteframe_t> SpriteFrames; extern TArray<spriteframe_t> SpriteFrames;
extern TArray<PalEntry> BloodTranslationColors; extern TArray<PalEntry> BloodTranslationColors;
extern bool r_showviewer;
enum HWRenderStyle enum HWRenderStyle
{ {
@ -655,7 +656,6 @@ void GLSprite::Process(AActor* thing, sector_t * sector, int thruportal)
if (!(thing->flags & MF_STEALTH) || !gl_fixedcolormap || !gl_enhanced_nightvision || thing == camera) if (!(thing->flags & MF_STEALTH) || !gl_fixedcolormap || !gl_enhanced_nightvision || thing == camera)
return; return;
} }
int spritenum = thing->sprite; int spritenum = thing->sprite;
DVector2 sprscale = thing->Scale; DVector2 sprscale = thing->Scale;
if (thing->player != NULL) if (thing->player != NULL)
@ -670,6 +670,14 @@ void GLSprite::Process(AActor* thing, sector_t * sector, int thruportal)
DVector3 thingpos = thing->InterpolatedPosition(r_TicFracF); DVector3 thingpos = thing->InterpolatedPosition(r_TicFracF);
if (thruportal == 1) thingpos += Displacements.getOffset(thing->Sector->PortalGroup, sector->PortalGroup); if (thruportal == 1) thingpos += Displacements.getOffset(thing->Sector->PortalGroup, sector->PortalGroup);
// Some added checks if the camera actor is not supposed to be seen. It can happen that some portal setup has this actor in view in which case it may not be skipped here
if (thing == camera && !r_showviewer)
{
DVector3 thingorigin = thing->Pos();
if (thruportal == 1) thingorigin += Displacements.getOffset(thing->Sector->PortalGroup, sector->PortalGroup);
if (fabs(thingorigin.X - ViewActorPos.X) < 2 && fabs(thingorigin.Y - ViewActorPos.Y) < 2) return;
}
// Too close to the camera. This doesn't look good if it is a sprite. // Too close to the camera. This doesn't look good if it is a sprite.
if (fabs(thingpos.X - ViewPos.X) < 2 && fabs(thingpos.Y - ViewPos.Y) < 2) if (fabs(thingpos.X - ViewPos.X) < 2 && fabs(thingpos.Y - ViewPos.Y) < 2)
{ {
@ -1202,9 +1210,7 @@ void gl_RenderActorsInPortal(FGLLinePortal *glport)
th->Prev += newpos - savedpos; th->Prev += newpos - savedpos;
GLSprite spr; GLSprite spr;
th->fillcolor = 0xff0000ff;
spr.Process(th, gl_FakeFlat(th->Sector, &fakesector, false), 2); spr.Process(th, gl_FakeFlat(th->Sector, &fakesector, false), 2);
th->fillcolor = 0xffffffff;
th->Angles.Yaw = savedangle; th->Angles.Yaw = savedangle;
th->SetXYZ(savedpos); th->SetXYZ(savedpos);
th->Prev -= newpos - savedpos; th->Prev -= newpos - savedpos;

View file

@ -106,6 +106,7 @@ int viewwindowx;
int viewwindowy; int viewwindowy;
DVector3 ViewPos; DVector3 ViewPos;
DVector3 ViewActorPos; // the actual position of the viewing actor, without interpolation and quake offsets.
DAngle ViewAngle; DAngle ViewAngle;
DAngle ViewPitch; DAngle ViewPitch;
DAngle ViewRoll; DAngle ViewRoll;
@ -743,10 +744,11 @@ void R_SetupFrame (AActor *actor)
iview->Old = iview->New; iview->Old = iview->New;
r_NoInterpolate = true; r_NoInterpolate = true;
} }
ViewActorPos = campos;
} }
else else
{ {
iview->New.Pos = { camera->Pos().XY(), camera->player ? camera->player->viewz : camera->Z() + camera->GetCameraHeight() }; ViewActorPos = iview->New.Pos = { camera->Pos().XY(), camera->player ? camera->player->viewz : camera->Z() + camera->GetCameraHeight() };
viewsector = camera->Sector; viewsector = camera->Sector;
r_showviewer = false; r_showviewer = false;
} }

View file

@ -15,6 +15,7 @@ class FSerializer;
extern DCanvas *RenderTarget; extern DCanvas *RenderTarget;
extern DVector3 ViewPos; extern DVector3 ViewPos;
extern DVector3 ViewActorPos;
extern DAngle ViewAngle; extern DAngle ViewAngle;
extern DAngle ViewPitch; extern DAngle ViewPitch;
extern DAngle ViewRoll; extern DAngle ViewRoll;