2016-09-14 18:01:13 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// Copyright(C) 2000-2016 Christoph Oelckers
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 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 Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with this program. If not, see http://www.gnu.org/licenses/
|
|
|
|
//
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
//
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
#include "gl/system/gl_system.h"
|
|
|
|
#include "p_local.h"
|
|
|
|
#include "p_lnspec.h"
|
|
|
|
#include "a_sharedglobal.h"
|
2017-01-28 19:44:46 +00:00
|
|
|
#include "g_levellocals.h"
|
2017-03-10 01:22:42 +00:00
|
|
|
#include "actor.h"
|
|
|
|
#include "actorinlines.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2013-09-03 16:29:39 +00:00
|
|
|
#include "gl/system/gl_interface.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
#include "gl/system/gl_cvars.h"
|
|
|
|
#include "gl/renderer/gl_lightdata.h"
|
|
|
|
#include "gl/renderer/gl_renderstate.h"
|
2014-05-10 23:23:27 +00:00
|
|
|
#include "gl/renderer/gl_renderer.h"
|
|
|
|
#include "gl/data/gl_vertexbuffer.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
#include "gl/dynlights/gl_dynlight.h"
|
2014-08-01 18:59:39 +00:00
|
|
|
#include "gl/dynlights/gl_lightbuffer.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
#include "gl/scene/gl_drawinfo.h"
|
|
|
|
#include "gl/scene/gl_portal.h"
|
2017-03-12 20:57:39 +00:00
|
|
|
#include "gl/scene/gl_scenedrawer.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
#include "gl/shaders/gl_shader.h"
|
|
|
|
#include "gl/textures/gl_material.h"
|
|
|
|
#include "gl/utility/gl_clock.h"
|
2016-08-25 21:20:23 +00:00
|
|
|
#include "gl/renderer/gl_quaddrawer.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
EXTERN_CVAR(Bool, gl_seamless)
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
2014-07-15 18:49:21 +00:00
|
|
|
// Collect lights for shader
|
2013-06-23 07:49:34 +00:00
|
|
|
//
|
|
|
|
//==========================================================================
|
2014-07-15 18:49:21 +00:00
|
|
|
FDynLightData lightdata;
|
|
|
|
|
2014-08-19 12:18:21 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
void GLWall::SetupLights()
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2018-04-01 16:45:27 +00:00
|
|
|
if (RenderStyle == STYLE_Add && !level.lightadditivesurfaces) return; // no lights on additively blended surfaces.
|
2016-05-18 20:20:18 +00:00
|
|
|
|
2014-08-19 12:18:21 +00:00
|
|
|
// check for wall types which cannot have dynamic lights on them (portal types never get here so they don't need to be checked.)
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case RENDERWALL_FOGBOUNDARY:
|
|
|
|
case RENDERWALL_MIRRORSURFACE:
|
|
|
|
case RENDERWALL_COLOR:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
float vtx[]={glseg.x1,zbottom[0],glseg.y1, glseg.x1,ztop[0],glseg.y1, glseg.x2,ztop[1],glseg.y2, glseg.x2,zbottom[1],glseg.y2};
|
|
|
|
Plane p;
|
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
lightdata.Clear();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2018-04-01 10:43:23 +00:00
|
|
|
auto normal = glseg.Normal();
|
2018-04-15 21:51:41 +00:00
|
|
|
p.Set(normal, -normal.X * glseg.x1 - normal.Z * glseg.y1);
|
2018-04-01 10:43:23 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
FLightNode *node;
|
|
|
|
if (seg->sidedef == NULL)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2014-07-15 18:49:21 +00:00
|
|
|
node = NULL;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2014-07-15 18:49:21 +00:00
|
|
|
else if (!(seg->sidedef->Flags & WALLF_POLYOBJ))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2014-07-15 18:49:21 +00:00
|
|
|
node = seg->sidedef->lighthead;
|
|
|
|
}
|
|
|
|
else if (sub)
|
|
|
|
{
|
|
|
|
// Polobject segs cannot be checked per sidedef so use the subsector instead.
|
|
|
|
node = sub->lighthead;
|
|
|
|
}
|
|
|
|
else node = NULL;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
// Iterate through all dynamic lights which touch this wall and render them
|
|
|
|
while (node)
|
|
|
|
{
|
|
|
|
if (!(node->lightsource->flags2&MF2_DORMANT))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2014-07-15 18:49:21 +00:00
|
|
|
iter_dlight++;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2016-12-08 11:49:40 +00:00
|
|
|
DVector3 posrel = node->lightsource->PosRelative(seg->frontsector);
|
|
|
|
float x = posrel.X;
|
|
|
|
float y = posrel.Y;
|
|
|
|
float z = posrel.Z;
|
2014-07-15 18:49:21 +00:00
|
|
|
float dist = fabsf(p.DistToPoint(x, z, y));
|
2016-09-04 10:45:09 +00:00
|
|
|
float radius = node->lightsource->GetRadius();
|
2014-07-15 18:49:21 +00:00
|
|
|
float scale = 1.0f / ((2.f * radius) - dist);
|
2017-03-12 18:44:00 +00:00
|
|
|
FVector3 fn, pos;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
if (radius > 0.f && dist < radius)
|
|
|
|
{
|
2017-03-12 18:44:00 +00:00
|
|
|
FVector3 nearPt, up, right;
|
|
|
|
|
|
|
|
pos = { x, z, y };
|
|
|
|
fn = p.Normal();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
fn.GetRightUp(right, up);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2017-03-12 18:44:00 +00:00
|
|
|
FVector3 tmpVec = fn * dist;
|
2014-07-15 18:49:21 +00:00
|
|
|
nearPt = pos + tmpVec;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2017-03-12 18:44:00 +00:00
|
|
|
FVector3 t1;
|
2014-07-15 18:49:21 +00:00
|
|
|
int outcnt[4]={0,0,0,0};
|
|
|
|
texcoord tcs[4];
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
// do a quick check whether the light touches this polygon
|
|
|
|
for(int i=0;i<4;i++)
|
|
|
|
{
|
2017-03-12 18:44:00 +00:00
|
|
|
t1 = FVector3(&vtx[i*3]);
|
|
|
|
FVector3 nearToVert = t1 - nearPt;
|
|
|
|
tcs[i].u = ((nearToVert | right) * scale) + 0.5f;
|
|
|
|
tcs[i].v = ((nearToVert | up) * scale) + 0.5f;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
if (tcs[i].u<0) outcnt[0]++;
|
|
|
|
if (tcs[i].u>1) outcnt[1]++;
|
|
|
|
if (tcs[i].v<0) outcnt[2]++;
|
|
|
|
if (tcs[i].v>1) outcnt[3]++;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
}
|
|
|
|
if (outcnt[0]!=4 && outcnt[1]!=4 && outcnt[2]!=4 && outcnt[3]!=4)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2018-04-16 06:55:13 +00:00
|
|
|
lightdata.GetLight(seg->frontsector->PortalGroup, p, node->lightsource, true);
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-15 18:49:21 +00:00
|
|
|
node = node->nextLight;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
2014-08-01 18:59:39 +00:00
|
|
|
dynlightindex = GLRenderer->mLights->UploadLights(lightdata);
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
2016-08-25 20:54:08 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// build the vertices for this wall
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void GLWall::MakeVertices(bool nosplit)
|
|
|
|
{
|
|
|
|
if (vertcount == 0)
|
|
|
|
{
|
|
|
|
bool split = (gl_seamless && !nosplit && seg->sidedef != NULL && !(seg->sidedef->Flags & WALLF_POLYOBJ) && !(flags & GLWF_NOSPLIT));
|
|
|
|
|
|
|
|
FFlatVertex *ptr = GLRenderer->mVBO->GetBuffer();
|
|
|
|
|
|
|
|
ptr->Set(glseg.x1, zbottom[0], glseg.y1, tcs[LOLFT].u, tcs[LOLFT].v);
|
|
|
|
ptr++;
|
|
|
|
if (split && glseg.fracleft == 0) SplitLeftEdge(ptr);
|
|
|
|
ptr->Set(glseg.x1, ztop[0], glseg.y1, tcs[UPLFT].u, tcs[UPLFT].v);
|
|
|
|
ptr++;
|
|
|
|
if (split && !(flags & GLWF_NOSPLITUPPER)) SplitUpperEdge(ptr);
|
|
|
|
ptr->Set(glseg.x2, ztop[1], glseg.y2, tcs[UPRGT].u, tcs[UPRGT].v);
|
|
|
|
ptr++;
|
|
|
|
if (split && glseg.fracright == 1) SplitRightEdge(ptr);
|
|
|
|
ptr->Set(glseg.x2, zbottom[1], glseg.y2, tcs[LORGT].u, tcs[LORGT].v);
|
|
|
|
ptr++;
|
|
|
|
if (split && !(flags & GLWF_NOSPLITLOWER)) SplitLowerEdge(ptr);
|
|
|
|
vertcount = GLRenderer->mVBO->GetCount(ptr, &vertindex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-14 23:14:41 +00:00
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// General purpose wall rendering function
|
|
|
|
// everything goes through here
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-08-25 21:02:43 +00:00
|
|
|
void GLWall::RenderWall(int textured)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2016-08-25 21:02:43 +00:00
|
|
|
gl_RenderState.Apply();
|
|
|
|
gl_RenderState.ApplyLightIndex(dynlightindex);
|
2016-08-25 20:54:08 +00:00
|
|
|
if (gl.buffermethod != BM_DEFERRED)
|
|
|
|
{
|
2017-01-02 22:21:28 +00:00
|
|
|
MakeVertices(!!(textured&RWF_NOSPLIT));
|
2016-08-25 20:54:08 +00:00
|
|
|
}
|
2016-08-25 21:20:23 +00:00
|
|
|
else if (vertcount == 0)
|
|
|
|
{
|
2016-09-01 09:52:52 +00:00
|
|
|
// This should never happen but in case it actually does, use the quad drawer as fallback (without edge splitting.)
|
|
|
|
// This way it at least gets drawn.
|
2016-08-25 21:20:23 +00:00
|
|
|
FQuadDrawer qd;
|
|
|
|
qd.Set(0, glseg.x1, zbottom[0], glseg.y1, tcs[LOLFT].u, tcs[LOLFT].v);
|
|
|
|
qd.Set(1, glseg.x1, ztop[0], glseg.y1, tcs[UPLFT].u, tcs[UPLFT].v);
|
|
|
|
qd.Set(2, glseg.x2, ztop[1], glseg.y2, tcs[UPRGT].u, tcs[UPRGT].v);
|
|
|
|
qd.Set(3, glseg.x2, zbottom[1], glseg.y2, tcs[LORGT].u, tcs[LORGT].v);
|
|
|
|
qd.Render(GL_TRIANGLE_FAN);
|
|
|
|
vertexcount += 4;
|
|
|
|
return;
|
|
|
|
}
|
2016-08-25 21:02:43 +00:00
|
|
|
GLRenderer->mVBO->RenderArray(GL_TRIANGLE_FAN, vertindex, vertcount);
|
|
|
|
vertexcount += vertcount;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void GLWall::RenderFogBoundary()
|
|
|
|
{
|
2017-03-12 20:57:39 +00:00
|
|
|
if (gl_fogmode && mDrawer->FixedColormap == 0)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2016-09-01 09:52:52 +00:00
|
|
|
if (!gl.legacyMode)
|
2016-05-03 21:28:42 +00:00
|
|
|
{
|
|
|
|
int rel = rellight + getExtraLight();
|
2017-03-12 20:57:39 +00:00
|
|
|
mDrawer->SetFog(lightlevel, rel, &Colormap, false);
|
2016-10-21 22:09:06 +00:00
|
|
|
gl_RenderState.EnableDrawBuffers(1);
|
2016-05-03 21:28:42 +00:00
|
|
|
gl_RenderState.SetEffect(EFF_FOGBOUNDARY);
|
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, 0.f);
|
|
|
|
glEnable(GL_POLYGON_OFFSET_FILL);
|
|
|
|
glPolygonOffset(-1.0f, -128.0f);
|
|
|
|
RenderWall(RWF_BLANK);
|
|
|
|
glPolygonOffset(0.0f, 0.0f);
|
|
|
|
glDisable(GL_POLYGON_OFFSET_FILL);
|
|
|
|
gl_RenderState.SetEffect(EFF_NONE);
|
2016-10-21 22:09:06 +00:00
|
|
|
gl_RenderState.EnableDrawBuffers(gl_RenderState.GetPassDrawBufferCount());
|
2016-05-03 21:28:42 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
RenderFogBoundaryCompat();
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
void GLWall::RenderMirrorSurface()
|
|
|
|
{
|
2018-02-15 16:56:04 +00:00
|
|
|
if (!GLRenderer->mirrorTexture.isValid()) return;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
// For the sphere map effect we need a normal of the mirror surface,
|
2016-10-03 14:09:32 +00:00
|
|
|
FVector3 v = glseg.Normal();
|
2014-07-14 17:54:07 +00:00
|
|
|
|
2016-09-01 09:52:52 +00:00
|
|
|
if (!gl.legacyMode)
|
2016-05-03 21:28:42 +00:00
|
|
|
{
|
|
|
|
// we use texture coordinates and texture matrix to pass the normal stuff to the shader so that the default vertex buffer format can be used as is.
|
2016-10-03 14:09:32 +00:00
|
|
|
tcs[LOLFT].u = tcs[LORGT].u = tcs[UPLFT].u = tcs[UPRGT].u = v.X;
|
|
|
|
tcs[LOLFT].v = tcs[LORGT].v = tcs[UPLFT].v = tcs[UPRGT].v = v.Z;
|
2014-07-14 17:54:07 +00:00
|
|
|
|
2016-05-03 21:28:42 +00:00
|
|
|
gl_RenderState.EnableTextureMatrix(true);
|
|
|
|
gl_RenderState.mTextureMatrix.computeNormalMatrix(gl_RenderState.mViewMatrix);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
glNormal3fv(&v[0]);
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
// Use sphere mapping for this
|
|
|
|
gl_RenderState.SetEffect(EFF_SPHEREMAP);
|
|
|
|
|
2017-03-12 20:57:39 +00:00
|
|
|
mDrawer->SetColor(lightlevel, 0, Colormap ,0.1f);
|
|
|
|
mDrawer->SetFog(lightlevel, 0, &Colormap, true);
|
2013-06-23 07:49:34 +00:00
|
|
|
gl_RenderState.BlendFunc(GL_SRC_ALPHA,GL_ONE);
|
|
|
|
gl_RenderState.AlphaFunc(GL_GREATER,0);
|
2013-09-03 12:05:41 +00:00
|
|
|
glDepthFunc(GL_LEQUAL);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2018-02-15 16:56:04 +00:00
|
|
|
FMaterial * pat=FMaterial::ValidateTexture(GLRenderer->mirrorTexture, false, false);
|
2014-09-09 10:00:42 +00:00
|
|
|
gl_RenderState.SetMaterial(pat, CLAMP_NONE, 0, -1, false);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
flags &= ~GLWF_GLOW;
|
2014-06-15 08:15:44 +00:00
|
|
|
RenderWall(RWF_BLANK);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2014-07-14 17:54:07 +00:00
|
|
|
gl_RenderState.EnableTextureMatrix(false);
|
2013-06-23 07:49:34 +00:00
|
|
|
gl_RenderState.SetEffect(EFF_NONE);
|
|
|
|
|
|
|
|
// Restore the defaults for the translucent pass
|
|
|
|
gl_RenderState.BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
2014-07-14 19:14:43 +00:00
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, gl_mask_sprite_threshold);
|
2013-09-03 12:05:41 +00:00
|
|
|
glDepthFunc(GL_LESS);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
// This is drawn in the translucent pass which is done after the decal pass
|
|
|
|
// As a result the decals have to be drawn here.
|
|
|
|
if (seg->sidedef->AttachedDecals)
|
|
|
|
{
|
2013-09-03 12:05:41 +00:00
|
|
|
glEnable(GL_POLYGON_OFFSET_FILL);
|
|
|
|
glPolygonOffset(-1.0f, -128.0f);
|
|
|
|
glDepthMask(false);
|
2013-06-23 07:49:34 +00:00
|
|
|
DoDrawDecals();
|
2013-09-03 12:05:41 +00:00
|
|
|
glDepthMask(true);
|
|
|
|
glPolygonOffset(0.0f, 0.0f);
|
|
|
|
glDisable(GL_POLYGON_OFFSET_FILL);
|
2013-06-23 07:49:34 +00:00
|
|
|
gl_RenderState.SetTextureMode(TM_MODULATE);
|
|
|
|
gl_RenderState.BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-01-31 20:10:59 +00:00
|
|
|
void GLWall::RenderTextured(int rflags)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2016-01-31 20:10:59 +00:00
|
|
|
int tmode = gl_RenderState.GetTextureMode();
|
2016-02-19 16:40:08 +00:00
|
|
|
int rel = rellight + getExtraLight();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2016-01-31 20:10:59 +00:00
|
|
|
if (flags & GLWF_GLOW)
|
2015-12-11 21:26:10 +00:00
|
|
|
{
|
2016-01-31 20:10:59 +00:00
|
|
|
gl_RenderState.EnableGlow(true);
|
|
|
|
gl_RenderState.SetGlowParams(topglowcolor, bottomglowcolor);
|
2015-12-11 21:26:10 +00:00
|
|
|
}
|
2017-01-28 19:44:46 +00:00
|
|
|
gl_RenderState.SetGlowPlanes(topplane, bottomplane);
|
2016-01-31 20:10:59 +00:00
|
|
|
gl_RenderState.SetMaterial(gltexture, flags & 3, 0, -1, false);
|
2015-12-11 21:26:10 +00:00
|
|
|
|
2016-01-31 20:10:59 +00:00
|
|
|
if (type == RENDERWALL_M2SNF)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2016-01-31 20:10:59 +00:00
|
|
|
if (flags & GLT_CLAMPY)
|
|
|
|
{
|
|
|
|
if (tmode == TM_MODULATE) gl_RenderState.SetTextureMode(TM_CLAMPY);
|
|
|
|
}
|
2017-03-12 20:57:39 +00:00
|
|
|
mDrawer->SetFog(255, 0, NULL, false);
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2017-01-28 19:44:46 +00:00
|
|
|
gl_RenderState.SetObjectColor(seg->frontsector->SpecialColors[sector_t::walltop] | 0xff000000);
|
|
|
|
gl_RenderState.SetObjectColor2(seg->frontsector->SpecialColors[sector_t::wallbottom] | 0xff000000);
|
2016-01-31 20:10:59 +00:00
|
|
|
|
|
|
|
float absalpha = fabsf(alpha);
|
2016-02-01 00:03:34 +00:00
|
|
|
if (lightlist == NULL)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2017-03-12 20:57:39 +00:00
|
|
|
if (type != RENDERWALL_M2SNF) mDrawer->SetFog(lightlevel, rel, &Colormap, RenderStyle == STYLE_Add);
|
|
|
|
mDrawer->SetColor(lightlevel, rel, Colormap, absalpha);
|
2016-01-31 20:10:59 +00:00
|
|
|
RenderWall(rflags);
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2014-09-21 09:08:17 +00:00
|
|
|
else
|
|
|
|
{
|
2016-01-31 20:10:59 +00:00
|
|
|
gl_RenderState.EnableSplit(true);
|
2016-02-01 00:03:34 +00:00
|
|
|
|
|
|
|
for (unsigned i = 0; i < lightlist->Size(); i++)
|
2014-09-21 09:08:17 +00:00
|
|
|
{
|
2016-02-01 15:13:07 +00:00
|
|
|
secplane_t &lowplane = i == (*lightlist).Size() - 1 ? bottomplane : (*lightlist)[i + 1].plane;
|
|
|
|
// this must use the exact same calculation method as GLWall::Process etc.
|
2016-03-30 18:01:44 +00:00
|
|
|
float low1 = lowplane.ZatPoint(vertexes[0]);
|
|
|
|
float low2 = lowplane.ZatPoint(vertexes[1]);
|
2016-02-01 15:13:07 +00:00
|
|
|
|
2016-02-04 00:12:18 +00:00
|
|
|
if (low1 < ztop[0] || low2 < ztop[1])
|
2016-02-01 15:13:07 +00:00
|
|
|
{
|
|
|
|
int thisll = (*lightlist)[i].caster != NULL ? gl_ClampLight(*(*lightlist)[i].p_lightlevel) : lightlevel;
|
|
|
|
FColormap thiscm;
|
|
|
|
thiscm.FadeColor = Colormap.FadeColor;
|
2017-03-28 21:36:26 +00:00
|
|
|
thiscm.FogDensity = Colormap.FogDensity;
|
2016-02-01 15:13:07 +00:00
|
|
|
thiscm.CopyFrom3DLight(&(*lightlist)[i]);
|
2017-03-12 20:57:39 +00:00
|
|
|
mDrawer->SetColor(thisll, rel, thiscm, absalpha);
|
|
|
|
if (type != RENDERWALL_M2SNF) mDrawer->SetFog(thisll, rel, &thiscm, RenderStyle == STYLE_Add);
|
2016-02-01 15:13:07 +00:00
|
|
|
gl_RenderState.SetSplitPlanes((*lightlist)[i].plane, lowplane);
|
|
|
|
RenderWall(rflags);
|
|
|
|
}
|
|
|
|
if (low1 <= zbottom[0] && low2 <= zbottom[1]) break;
|
2014-09-21 09:08:17 +00:00
|
|
|
}
|
2016-02-01 00:03:34 +00:00
|
|
|
|
2016-01-31 20:10:59 +00:00
|
|
|
gl_RenderState.EnableSplit(false);
|
2014-09-21 09:08:17 +00:00
|
|
|
}
|
2017-01-28 19:44:46 +00:00
|
|
|
gl_RenderState.SetObjectColor(0xffffffff);
|
|
|
|
gl_RenderState.SetObjectColor2(0);
|
2016-01-31 20:10:59 +00:00
|
|
|
gl_RenderState.SetTextureMode(tmode);
|
|
|
|
gl_RenderState.EnableGlow(false);
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2016-01-31 20:10:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2016-01-31 20:10:59 +00:00
|
|
|
void GLWall::RenderTranslucentWall()
|
|
|
|
{
|
|
|
|
if (gltexture)
|
|
|
|
{
|
2017-03-12 20:57:39 +00:00
|
|
|
if (mDrawer->FixedColormap == CM_DEFAULT && gl_lights && gl.lightmethod == LM_DIRECT)
|
2016-01-31 21:24:48 +00:00
|
|
|
{
|
|
|
|
SetupLights();
|
|
|
|
}
|
2018-04-01 12:38:48 +00:00
|
|
|
if (!gltexture->tex->GetTranslucency()) gl_RenderState.AlphaFunc(GL_GEQUAL, gl_mask_threshold);
|
2016-01-31 20:10:59 +00:00
|
|
|
else gl_RenderState.AlphaFunc(GL_GEQUAL, 0.f);
|
|
|
|
if (RenderStyle == STYLE_Add) gl_RenderState.BlendFunc(GL_SRC_ALPHA,GL_ONE);
|
|
|
|
RenderTextured(RWF_TEXTURED | RWF_NOSPLIT);
|
|
|
|
if (RenderStyle == STYLE_Add) gl_RenderState.BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
}
|
|
|
|
else
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2016-01-31 20:10:59 +00:00
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, 0.f);
|
2017-03-12 20:57:39 +00:00
|
|
|
mDrawer->SetColor(lightlevel, 0, Colormap, fabsf(alpha));
|
|
|
|
mDrawer->SetFog(lightlevel, 0, &Colormap, RenderStyle == STYLE_Add);
|
2016-01-31 21:24:48 +00:00
|
|
|
gl_RenderState.EnableTexture(false);
|
2016-01-31 20:10:59 +00:00
|
|
|
RenderWall(RWF_NOSPLIT);
|
2013-06-23 07:49:34 +00:00
|
|
|
gl_RenderState.EnableTexture(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
void GLWall::Draw(int pass)
|
|
|
|
{
|
2016-10-03 14:09:32 +00:00
|
|
|
gl_RenderState.SetNormal(glseg.Normal());
|
2013-06-23 07:49:34 +00:00
|
|
|
switch (pass)
|
|
|
|
{
|
2014-08-19 12:18:21 +00:00
|
|
|
case GLPASS_LIGHTSONLY:
|
|
|
|
SetupLights();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GLPASS_ALL:
|
2013-06-23 07:49:34 +00:00
|
|
|
SetupLights();
|
|
|
|
// fall through
|
2014-08-19 12:18:21 +00:00
|
|
|
case GLPASS_PLAIN:
|
2016-01-31 20:10:59 +00:00
|
|
|
RenderTextured(RWF_TEXTURED);
|
2013-06-23 07:49:34 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case GLPASS_TRANSLUCENT:
|
2014-09-20 07:04:36 +00:00
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case RENDERWALL_MIRRORSURFACE:
|
|
|
|
RenderMirrorSurface();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RENDERWALL_FOGBOUNDARY:
|
|
|
|
RenderFogBoundary();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
RenderTranslucentWall();
|
|
|
|
break;
|
|
|
|
}
|
2016-05-08 22:58:55 +00:00
|
|
|
break;
|
2016-05-05 10:18:09 +00:00
|
|
|
|
|
|
|
case GLPASS_LIGHTTEX:
|
|
|
|
case GLPASS_LIGHTTEX_ADDITIVE:
|
2016-08-24 07:26:11 +00:00
|
|
|
case GLPASS_LIGHTTEX_FOGGY:
|
2016-05-05 10:18:09 +00:00
|
|
|
RenderLightsCompat(pass);
|
|
|
|
break;
|
2016-05-08 20:07:18 +00:00
|
|
|
|
|
|
|
case GLPASS_TEXONLY:
|
|
|
|
gl_RenderState.SetMaterial(gltexture, flags & 3, 0, -1, false);
|
|
|
|
RenderWall(RWF_TEXTURED);
|
|
|
|
break;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
}
|