2013-06-23 07:49:34 +00:00
|
|
|
|
/*
|
|
|
|
|
** gl_scene.cpp
|
|
|
|
|
** manages the rendering of the player's view
|
|
|
|
|
**
|
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
|
** Copyright 2004-2005 Christoph Oelckers
|
|
|
|
|
** All rights reserved.
|
|
|
|
|
**
|
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
|
** are met:
|
|
|
|
|
**
|
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
|
** 4. When not used as part of GZDoom or a GZDoom derivative, this code will be
|
|
|
|
|
** covered by the terms of the GNU Lesser General Public License as published
|
|
|
|
|
** by the Free Software Foundation; either version 2.1 of the License, or (at
|
|
|
|
|
** your option) any later version.
|
|
|
|
|
** 5. Full disclosure of the entire project's source code, except for third
|
|
|
|
|
** party libraries is mandatory. (NOTE: This clause is non-negotiable!)
|
|
|
|
|
**
|
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
|
**
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "gl/system/gl_system.h"
|
|
|
|
|
#include "gi.h"
|
|
|
|
|
#include "m_png.h"
|
|
|
|
|
#include "m_random.h"
|
|
|
|
|
#include "st_stuff.h"
|
|
|
|
|
#include "dobject.h"
|
|
|
|
|
#include "doomstat.h"
|
|
|
|
|
#include "g_level.h"
|
|
|
|
|
#include "r_data/r_interpolate.h"
|
|
|
|
|
#include "r_utility.h"
|
|
|
|
|
#include "d_player.h"
|
|
|
|
|
#include "p_effect.h"
|
|
|
|
|
#include "sbar.h"
|
|
|
|
|
#include "po_man.h"
|
|
|
|
|
#include "r_utility.h"
|
|
|
|
|
#include "a_hexenglobal.h"
|
|
|
|
|
#include "p_local.h"
|
|
|
|
|
#include "gl/gl_functions.h"
|
|
|
|
|
|
2014-08-01 18:59:39 +00:00
|
|
|
|
#include "gl/dynlights/gl_lightbuffer.h"
|
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_framebuffer.h"
|
|
|
|
|
#include "gl/system/gl_cvars.h"
|
|
|
|
|
#include "gl/renderer/gl_lightdata.h"
|
|
|
|
|
#include "gl/renderer/gl_renderstate.h"
|
|
|
|
|
#include "gl/data/gl_data.h"
|
|
|
|
|
#include "gl/data/gl_vertexbuffer.h"
|
|
|
|
|
#include "gl/dynlights/gl_dynlight.h"
|
|
|
|
|
#include "gl/models/gl_models.h"
|
|
|
|
|
#include "gl/scene/gl_clipper.h"
|
|
|
|
|
#include "gl/scene/gl_drawinfo.h"
|
|
|
|
|
#include "gl/scene/gl_portal.h"
|
|
|
|
|
#include "gl/shaders/gl_shader.h"
|
2015-10-31 00:51:35 +00:00
|
|
|
|
#include "gl/stereo3d/gl_stereo3d.h"
|
|
|
|
|
#include "gl/stereo3d/scoped_view_shifter.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
|
#include "gl/textures/gl_material.h"
|
|
|
|
|
#include "gl/utility/gl_clock.h"
|
|
|
|
|
#include "gl/utility/gl_convert.h"
|
|
|
|
|
#include "gl/utility/gl_templates.h"
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
|
//
|
|
|
|
|
// CVARs
|
|
|
|
|
//
|
|
|
|
|
//==========================================================================
|
|
|
|
|
CVAR(Bool, gl_texture, true, 0)
|
|
|
|
|
CVAR(Bool, gl_no_skyclear, false, CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
|
|
|
|
CVAR(Float, gl_mask_threshold, 0.5f,CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
|
|
|
|
CVAR(Float, gl_mask_sprite_threshold, 0.5f,CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
2014-08-30 13:34:14 +00:00
|
|
|
|
CVAR(Bool, gl_sort_textures, false, CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
EXTERN_CVAR (Int, screenblocks)
|
|
|
|
|
EXTERN_CVAR (Bool, cl_capfps)
|
|
|
|
|
EXTERN_CVAR (Bool, r_deathcamera)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
extern int viewpitch;
|
2016-03-05 11:46:47 +00:00
|
|
|
|
extern bool NoInterpolateView;
|
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
|
DWORD gl_fixedcolormap;
|
|
|
|
|
area_t in_area;
|
|
|
|
|
TArray<BYTE> currentmapsection;
|
|
|
|
|
|
|
|
|
|
void gl_ParseDefs();
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// R_FrustumAngle
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
angle_t FGLRenderer::FrustumAngle()
|
|
|
|
|
{
|
2016-03-21 01:57:02 +00:00
|
|
|
|
float tilt= fabs(mAngles.Pitch.Degrees);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// If the pitch is larger than this you can look all around at a FOV of 90<39>
|
|
|
|
|
if (tilt>46.0f) return 0xffffffff;
|
|
|
|
|
|
|
|
|
|
// ok, this is a gross hack that barely works...
|
|
|
|
|
// but at least it doesn't overestimate too much...
|
|
|
|
|
double floatangle=2.0+(45.0+((tilt/1.9)))*mCurrentFoV*48.0/BaseRatioSizes[WidescreenRatio][3]/90.0;
|
2016-03-30 18:01:44 +00:00
|
|
|
|
angle_t a1 = DAngle(floatangle).BAMs();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
if (a1>=ANGLE_180) return 0xffffffff;
|
|
|
|
|
return a1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Sets the area the camera is in
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
void FGLRenderer::SetViewArea()
|
|
|
|
|
{
|
|
|
|
|
// The render_sector is better suited to represent the current position in GL
|
2016-04-02 21:17:16 +00:00
|
|
|
|
viewsector = R_PointInSubsector(ViewPos)->render_sector;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// Get the heightsec state from the render sector, not the current one!
|
|
|
|
|
if (viewsector->heightsec && !(viewsector->heightsec->MoreFlags & SECF_IGNOREHEIGHTSEC))
|
|
|
|
|
{
|
2016-04-02 21:17:16 +00:00
|
|
|
|
in_area = ViewPos.Z <= viewsector->heightsec->floorplane.ZatPoint(ViewPos) ? area_below :
|
|
|
|
|
(ViewPos.Z > viewsector->heightsec->ceilingplane.ZatPoint(ViewPos) &&
|
|
|
|
|
!(viewsector->heightsec->MoreFlags&SECF_FAKEFLOORONLY)) ? area_above : area_normal;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2016-04-02 21:17:16 +00:00
|
|
|
|
in_area = area_default; // depends on exposed lower sectors
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// resets the 3D viewport
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::ResetViewport()
|
|
|
|
|
{
|
|
|
|
|
int trueheight = static_cast<OpenGLFrameBuffer*>(screen)->GetTrueHeight(); // ugh...
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glViewport(0, (trueheight-screen->GetHeight())/2, screen->GetWidth(), screen->GetHeight());
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// sets 3D viewport and initial state
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::SetViewport(GL_IRECT *bounds)
|
|
|
|
|
{
|
|
|
|
|
if (!bounds)
|
|
|
|
|
{
|
|
|
|
|
int height, width;
|
|
|
|
|
|
|
|
|
|
// Special handling so the view with a visible status bar displays properly
|
|
|
|
|
|
|
|
|
|
if (screenblocks >= 10)
|
|
|
|
|
{
|
|
|
|
|
height = SCREENHEIGHT;
|
|
|
|
|
width = SCREENWIDTH;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
height = (screenblocks*SCREENHEIGHT/10) & ~7;
|
|
|
|
|
width = (screenblocks*SCREENWIDTH/10);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int trueheight = static_cast<OpenGLFrameBuffer*>(screen)->GetTrueHeight(); // ugh...
|
|
|
|
|
int bars = (trueheight-screen->GetHeight())/2;
|
|
|
|
|
|
|
|
|
|
int vw = viewwidth;
|
|
|
|
|
int vh = viewheight;
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glViewport(viewwindowx, trueheight-bars-(height+viewwindowy-((height-vh)/2)), vw, height);
|
|
|
|
|
glScissor(viewwindowx, trueheight-bars-(vh+viewwindowy), vw, vh);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glViewport(bounds->left, bounds->top, bounds->width, bounds->height);
|
|
|
|
|
glScissor(bounds->left, bounds->top, bounds->width, bounds->height);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glEnable(GL_SCISSOR_TEST);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
#ifdef _DEBUG
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
|
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
#else
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glClear(GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glEnable(GL_MULTISAMPLE);
|
|
|
|
|
glEnable(GL_DEPTH_TEST);
|
|
|
|
|
glEnable(GL_STENCIL_TEST);
|
|
|
|
|
glStencilFunc(GL_ALWAYS,0,~0); // default stencil
|
|
|
|
|
glStencilOp(GL_KEEP,GL_KEEP,GL_REPLACE);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Setup the camera position
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
2016-04-02 21:17:16 +00:00
|
|
|
|
void FGLRenderer::SetViewAngle(DAngle viewangle)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
2016-04-02 21:17:16 +00:00
|
|
|
|
mAngles.Yaw = float(270.0-viewangle.Degrees);
|
|
|
|
|
DVector2 v = ViewAngle.ToVector();
|
|
|
|
|
mViewVector.X = v.X;
|
|
|
|
|
mViewVector.Y = v.Y;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
R_SetViewAngle();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// SetProjection
|
|
|
|
|
// sets projection matrix
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::SetProjection(float fov, float ratio, float fovratio)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
float fovy = 2 * RAD2DEG(atan(tan(DEG2RAD(fov) / 2) / fovratio));
|
2014-07-13 20:37:34 +00:00
|
|
|
|
gl_RenderState.mProjectionMatrix.perspective(fovy, ratio, 5.f, 65536.f);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
gl_RenderState.Set2DMode(false);
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-31 00:51:35 +00:00
|
|
|
|
// raw matrix input from stereo 3d modes
|
2015-10-31 11:46:36 +00:00
|
|
|
|
void FGLRenderer::SetProjection(VSMatrix matrix)
|
2015-10-31 00:51:35 +00:00
|
|
|
|
{
|
2015-10-31 11:46:36 +00:00
|
|
|
|
gl_RenderState.mProjectionMatrix.loadIdentity();
|
|
|
|
|
gl_RenderState.mProjectionMatrix.multMatrix(matrix);
|
2015-10-31 00:51:35 +00:00
|
|
|
|
gl_RenderState.Set2DMode(false);
|
|
|
|
|
}
|
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Setup the modelview matrix
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
2016-04-02 21:17:16 +00:00
|
|
|
|
void FGLRenderer::SetViewMatrix(float vx, float vy, float vz, bool mirror, bool planemirror)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
|
|
|
|
float mult = mirror? -1:1;
|
2014-12-29 22:52:20 +00:00
|
|
|
|
float planemult = planemirror? -glset.pixelstretch : glset.pixelstretch;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-07-13 20:37:34 +00:00
|
|
|
|
gl_RenderState.mViewMatrix.loadIdentity();
|
2016-03-21 01:57:02 +00:00
|
|
|
|
gl_RenderState.mViewMatrix.rotate(GLRenderer->mAngles.Roll.Degrees, 0.0f, 0.0f, 1.0f);
|
|
|
|
|
gl_RenderState.mViewMatrix.rotate(GLRenderer->mAngles.Pitch.Degrees, 1.0f, 0.0f, 0.0f);
|
|
|
|
|
gl_RenderState.mViewMatrix.rotate(GLRenderer->mAngles.Yaw.Degrees, 0.0f, mult, 0.0f);
|
2016-04-02 21:17:16 +00:00
|
|
|
|
gl_RenderState.mViewMatrix.translate(vx * mult, -vz * planemult , -vy);
|
2014-07-13 20:37:34 +00:00
|
|
|
|
gl_RenderState.mViewMatrix.scale(-mult, planemult, 1);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// SetupView
|
|
|
|
|
// Setup the view rotation matrix for the given viewpoint
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
2016-04-02 21:17:16 +00:00
|
|
|
|
void FGLRenderer::SetupView(float vx, float vy, float vz, DAngle va, bool mirror, bool planemirror)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
2016-04-02 21:17:16 +00:00
|
|
|
|
SetViewAngle(va);
|
|
|
|
|
SetViewMatrix(vx, vy, vz, mirror, planemirror);
|
2014-07-13 20:37:34 +00:00
|
|
|
|
gl_RenderState.ApplyMatrices();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// CreateScene
|
|
|
|
|
//
|
|
|
|
|
// creates the draw lists for the current scene
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::CreateScene()
|
|
|
|
|
{
|
|
|
|
|
// reset the portal manager
|
|
|
|
|
GLPortal::StartFrame();
|
|
|
|
|
PO_LinkToSubsectors();
|
|
|
|
|
|
|
|
|
|
ProcessAll.Clock();
|
|
|
|
|
|
|
|
|
|
// clip the scene and fill the drawlists
|
|
|
|
|
for(unsigned i=0;i<portals.Size(); i++) portals[i]->glportal = NULL;
|
|
|
|
|
gl_spriteindex=0;
|
|
|
|
|
Bsp.Clock();
|
2016-04-02 21:17:16 +00:00
|
|
|
|
R_SetView();
|
2016-04-17 08:39:29 +00:00
|
|
|
|
validcount++; // used for processing sidedefs only once by the renderer.
|
2013-06-23 07:49:34 +00:00
|
|
|
|
gl_RenderBSPNode (nodes + numnodes - 1);
|
|
|
|
|
Bsp.Unclock();
|
|
|
|
|
|
|
|
|
|
// And now the crappy hacks that have to be done to avoid rendering anomalies:
|
|
|
|
|
|
|
|
|
|
gl_drawinfo->HandleMissingTextures(); // Missing upper/lower textures
|
|
|
|
|
gl_drawinfo->HandleHackedSubsectors(); // open sector hacks for deep water
|
|
|
|
|
gl_drawinfo->ProcessSectorStacks(); // merge visplanes of sector stacks
|
|
|
|
|
|
|
|
|
|
ProcessAll.Unclock();
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// RenderScene
|
|
|
|
|
//
|
|
|
|
|
// Draws the current draw lists for the non GLSL renderer
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::RenderScene(int recursion)
|
|
|
|
|
{
|
|
|
|
|
RenderAll.Clock();
|
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthMask(true);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
if (!gl_no_skyclear) GLPortal::RenderFirstSkyPortal(recursion);
|
|
|
|
|
|
2016-04-02 21:17:16 +00:00
|
|
|
|
gl_RenderState.SetCameraPos(ViewPos.X, ViewPos.Y, ViewPos.Z);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
gl_RenderState.EnableFog(true);
|
|
|
|
|
gl_RenderState.BlendFunc(GL_ONE,GL_ZERO);
|
|
|
|
|
|
2014-08-30 13:34:14 +00:00
|
|
|
|
if (gl_sort_textures)
|
|
|
|
|
{
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_PLAINWALLS].SortWalls();
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_PLAINFLATS].SortFlats();
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDWALLS].SortWalls();
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDFLATS].SortFlats();
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDWALLSOFS].SortWalls();
|
|
|
|
|
}
|
2014-08-19 12:18:21 +00:00
|
|
|
|
|
|
|
|
|
// if we don't have a persistently mapped buffer, we have to process all the dynamic lights up front,
|
|
|
|
|
// so that we don't have to do repeated map/unmap calls on the buffer.
|
2016-04-26 11:50:05 +00:00
|
|
|
|
bool haslights = mLightCount > 0 && gl_fixedcolormap == CM_DEFAULT && gl_lights;
|
|
|
|
|
if (gl.lightmethod == LM_DEFERRED && haslights)
|
2014-08-19 12:18:21 +00:00
|
|
|
|
{
|
|
|
|
|
GLRenderer->mLights->Begin();
|
2014-08-30 12:33:06 +00:00
|
|
|
|
gl_drawinfo->drawlists[GLDL_PLAINWALLS].DrawWalls(GLPASS_LIGHTSONLY);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_PLAINFLATS].DrawFlats(GLPASS_LIGHTSONLY);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDWALLS].DrawWalls(GLPASS_LIGHTSONLY);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDFLATS].DrawFlats(GLPASS_LIGHTSONLY);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDWALLSOFS].DrawWalls(GLPASS_LIGHTSONLY);
|
2014-08-19 12:18:21 +00:00
|
|
|
|
gl_drawinfo->drawlists[GLDL_TRANSLUCENTBORDER].Draw(GLPASS_LIGHTSONLY);
|
2014-11-09 11:10:33 +00:00
|
|
|
|
gl_drawinfo->drawlists[GLDL_TRANSLUCENT].Draw(GLPASS_LIGHTSONLY, true);
|
2014-08-19 12:18:21 +00:00
|
|
|
|
GLRenderer->mLights->Finish();
|
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// Part 1: solid geometry. This is set up so that there are no transparent parts
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthFunc(GL_LESS);
|
2014-07-14 19:14:43 +00:00
|
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, 0.f);
|
2014-08-19 12:18:21 +00:00
|
|
|
|
glDisable(GL_POLYGON_OFFSET_FILL);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
int pass;
|
|
|
|
|
|
2016-04-26 11:50:05 +00:00
|
|
|
|
if (!haslights || gl.lightmethod == LM_DEFERRED)
|
|
|
|
|
{
|
|
|
|
|
pass = GLPASS_PLAIN;
|
|
|
|
|
}
|
|
|
|
|
else if (gl.lightmethod == LM_DIRECT)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
|
|
|
|
pass = GLPASS_ALL;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2016-04-26 11:50:05 +00:00
|
|
|
|
// Todo: Draw lights with multipass rendering.
|
|
|
|
|
// RenderMultpassStuff();
|
|
|
|
|
|
|
|
|
|
// The remaining stuff which is unaffected by dynamic lights is just processed as normal.
|
2014-07-15 18:49:21 +00:00
|
|
|
|
pass = GLPASS_PLAIN;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gl_RenderState.EnableTexture(gl_texture);
|
2014-07-15 18:49:21 +00:00
|
|
|
|
gl_RenderState.EnableBrightmap(true);
|
2014-08-30 12:33:06 +00:00
|
|
|
|
gl_drawinfo->drawlists[GLDL_PLAINWALLS].DrawWalls(pass);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_PLAINFLATS].DrawFlats(pass);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
|
2014-07-15 18:49:21 +00:00
|
|
|
|
// Part 2: masked geometry. This is set up so that only pixels with alpha>gl_mask_threshold will show
|
2013-06-23 07:49:34 +00:00
|
|
|
|
if (!gl_texture)
|
|
|
|
|
{
|
|
|
|
|
gl_RenderState.EnableTexture(true);
|
|
|
|
|
gl_RenderState.SetTextureMode(TM_MASK);
|
|
|
|
|
}
|
2014-07-14 19:14:43 +00:00
|
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, gl_mask_threshold);
|
2014-08-30 12:33:06 +00:00
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDWALLS].DrawWalls(pass);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDFLATS].DrawFlats(pass);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-08-19 12:18:21 +00:00
|
|
|
|
// Part 3: masked geometry with polygon offset. This list is empty most of the time so only waste time on it when in use.
|
2014-08-30 11:04:41 +00:00
|
|
|
|
if (gl_drawinfo->drawlists[GLDL_MASKEDWALLSOFS].Size() > 0)
|
2014-07-15 19:16:59 +00:00
|
|
|
|
{
|
|
|
|
|
glEnable(GL_POLYGON_OFFSET_FILL);
|
|
|
|
|
glPolygonOffset(-1.0f, -128.0f);
|
2014-08-30 12:33:06 +00:00
|
|
|
|
gl_drawinfo->drawlists[GLDL_MASKEDWALLSOFS].DrawWalls(pass);
|
2014-07-15 19:16:59 +00:00
|
|
|
|
glDisable(GL_POLYGON_OFFSET_FILL);
|
|
|
|
|
glPolygonOffset(0, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_MODELS].Draw(pass);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
gl_RenderState.BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
|
|
2014-08-19 12:18:21 +00:00
|
|
|
|
// Part 4: Draw decals (not a real pass)
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthFunc(GL_LEQUAL);
|
|
|
|
|
glEnable(GL_POLYGON_OFFSET_FILL);
|
|
|
|
|
glPolygonOffset(-1.0f, -128.0f);
|
|
|
|
|
glDepthMask(false);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-08-30 12:33:06 +00:00
|
|
|
|
// this is the only geometry type on which decals can possibly appear
|
2014-08-30 13:34:14 +00:00
|
|
|
|
gl_drawinfo->drawlists[GLDL_PLAINWALLS].DrawDecals();
|
2016-04-26 11:50:05 +00:00
|
|
|
|
if (gl.lightmethod == LM_SOFTWARE)
|
|
|
|
|
{
|
|
|
|
|
// also process the render lists with walls and dynamic lights
|
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
gl_RenderState.SetTextureMode(TM_MODULATE);
|
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthMask(true);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Push bleeding floor/ceiling textures back a little in the z-buffer
|
|
|
|
|
// so they don't interfere with overlapping mid textures.
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glPolygonOffset(1.0f, 128.0f);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-08-19 12:18:21 +00:00
|
|
|
|
// Part 5: flood all the gaps with the back sector's flat texture
|
|
|
|
|
// This will always be drawn like GLDL_PLAIN, depending on the fog settings
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthMask(false); // don't write to Z-buffer!
|
2013-08-18 13:41:52 +00:00
|
|
|
|
gl_RenderState.EnableFog(true);
|
2014-07-14 19:14:43 +00:00
|
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, 0.f);
|
2013-08-18 13:41:52 +00:00
|
|
|
|
gl_RenderState.BlendFunc(GL_ONE,GL_ZERO);
|
|
|
|
|
gl_drawinfo->DrawUnhandledMissingTextures();
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthMask(true);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glPolygonOffset(0.0f, 0.0f);
|
|
|
|
|
glDisable(GL_POLYGON_OFFSET_FILL);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
RenderAll.Unclock();
|
2014-08-30 12:33:06 +00:00
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// RenderTranslucent
|
|
|
|
|
//
|
|
|
|
|
// Draws the current draw lists for the non GLSL renderer
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::RenderTranslucent()
|
|
|
|
|
{
|
|
|
|
|
RenderAll.Clock();
|
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthMask(false);
|
2016-04-02 21:17:16 +00:00
|
|
|
|
gl_RenderState.SetCameraPos(ViewPos.X, ViewPos.Y, ViewPos.Z);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// final pass: translucent stuff
|
2014-07-14 19:14:43 +00:00
|
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, gl_mask_sprite_threshold);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
gl_RenderState.BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
|
|
|
|
|
|
gl_RenderState.EnableBrightmap(true);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_TRANSLUCENTBORDER].Draw(GLPASS_TRANSLUCENT);
|
|
|
|
|
gl_drawinfo->drawlists[GLDL_TRANSLUCENT].DrawSorted();
|
|
|
|
|
gl_RenderState.EnableBrightmap(false);
|
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDepthMask(true);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-07-14 19:14:43 +00:00
|
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, 0.5f);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
RenderAll.Unclock();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// gl_drawscene - this function renders the scene from the current
|
|
|
|
|
// viewpoint, including mirrors and skyboxes and other portals
|
|
|
|
|
// It is assumed that the GLPortal::EndFrame returns with the
|
|
|
|
|
// stencil, z-buffer and the projection matrix intact!
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
EXTERN_CVAR(Bool, gl_draw_sync)
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::DrawScene(bool toscreen)
|
|
|
|
|
{
|
|
|
|
|
static int recursion=0;
|
|
|
|
|
|
|
|
|
|
CreateScene();
|
|
|
|
|
GLRenderer->mCurrentPortal = NULL; // this must be reset before any portal recursion takes place.
|
|
|
|
|
|
|
|
|
|
// Up to this point in the main draw call no rendering is performed so we can wait
|
|
|
|
|
// with swapping the render buffer until now.
|
|
|
|
|
if (!gl_draw_sync && toscreen)
|
|
|
|
|
{
|
|
|
|
|
All.Unclock();
|
|
|
|
|
static_cast<OpenGLFrameBuffer*>(screen)->Swap();
|
|
|
|
|
All.Clock();
|
|
|
|
|
}
|
|
|
|
|
RenderScene(recursion);
|
|
|
|
|
|
|
|
|
|
// Handle all portals after rendering the opaque objects but before
|
|
|
|
|
// doing all translucent stuff
|
|
|
|
|
recursion++;
|
|
|
|
|
GLPortal::EndFrame();
|
|
|
|
|
recursion--;
|
|
|
|
|
RenderTranslucent();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-04-26 18:02:57 +00:00
|
|
|
|
void gl_FillScreen()
|
2014-05-11 12:46:37 +00:00
|
|
|
|
{
|
2014-07-14 19:14:43 +00:00
|
|
|
|
gl_RenderState.AlphaFunc(GL_GEQUAL, 0.f);
|
2014-05-11 12:46:37 +00:00
|
|
|
|
gl_RenderState.EnableTexture(false);
|
2014-05-11 20:57:42 +00:00
|
|
|
|
gl_RenderState.Apply();
|
2014-06-14 23:14:41 +00:00
|
|
|
|
FFlatVertex *ptr = GLRenderer->mVBO->GetBuffer();
|
|
|
|
|
ptr->Set(0, 0, 0, 0, 0);
|
|
|
|
|
ptr++;
|
|
|
|
|
ptr->Set(0, (float)SCREENHEIGHT, 0, 0, 0);
|
|
|
|
|
ptr++;
|
|
|
|
|
ptr->Set((float)SCREENWIDTH, 0, 0, 0, 0);
|
|
|
|
|
ptr++;
|
|
|
|
|
ptr->Set((float)SCREENWIDTH, (float)SCREENHEIGHT, 0, 0, 0);
|
|
|
|
|
ptr++;
|
|
|
|
|
GLRenderer->mVBO->RenderCurrent(ptr, GL_TRIANGLE_STRIP);
|
2014-05-11 12:46:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
|
//==========================================================================
|
|
|
|
|
//
|
|
|
|
|
// Draws a blend over the entire view
|
|
|
|
|
//
|
|
|
|
|
//==========================================================================
|
|
|
|
|
void FGLRenderer::DrawBlend(sector_t * viewsector)
|
|
|
|
|
{
|
|
|
|
|
float blend[4]={0,0,0,0};
|
|
|
|
|
PalEntry blendv=0;
|
|
|
|
|
float extra_red;
|
|
|
|
|
float extra_green;
|
|
|
|
|
float extra_blue;
|
|
|
|
|
player_t *player = NULL;
|
|
|
|
|
|
|
|
|
|
if (players[consoleplayer].camera != NULL)
|
|
|
|
|
{
|
|
|
|
|
player=players[consoleplayer].camera->player;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// don't draw sector based blends when an invulnerability colormap is active
|
|
|
|
|
if (!gl_fixedcolormap)
|
|
|
|
|
{
|
|
|
|
|
if (!viewsector->e->XFloor.ffloors.Size())
|
|
|
|
|
{
|
|
|
|
|
if (viewsector->heightsec && !(viewsector->MoreFlags&SECF_IGNOREHEIGHTSEC))
|
|
|
|
|
{
|
2014-05-11 12:46:37 +00:00
|
|
|
|
switch (in_area)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
|
|
|
|
default:
|
2014-05-11 12:46:37 +00:00
|
|
|
|
case area_normal: blendv = viewsector->heightsec->midmap; break;
|
|
|
|
|
case area_above: blendv = viewsector->heightsec->topmap; break;
|
|
|
|
|
case area_below: blendv = viewsector->heightsec->bottommap; break;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
TArray<lightlist_t> & lightlist = viewsector->e->XFloor.lightlist;
|
|
|
|
|
|
2014-05-11 12:46:37 +00:00
|
|
|
|
for (unsigned int i = 0; i < lightlist.Size(); i++)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
2016-04-02 21:17:16 +00:00
|
|
|
|
double lightbottom;
|
2014-05-11 12:46:37 +00:00
|
|
|
|
if (i < lightlist.Size() - 1)
|
2016-04-02 21:17:16 +00:00
|
|
|
|
lightbottom = lightlist[i + 1].plane.ZatPoint(ViewPos);
|
2014-05-11 12:46:37 +00:00
|
|
|
|
else
|
2016-04-02 21:17:16 +00:00
|
|
|
|
lightbottom = viewsector->floorplane.ZatPoint(ViewPos);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2016-04-02 21:17:16 +00:00
|
|
|
|
if (lightbottom < ViewPos.Z && (!lightlist[i].caster || !(lightlist[i].caster->flags&FF_FADEWALLS)))
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
|
|
|
|
// 3d floor 'fog' is rendered as a blending value
|
2014-05-11 12:46:37 +00:00
|
|
|
|
blendv = lightlist[i].blend;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
// If this is the same as the sector's it doesn't apply!
|
2014-05-11 12:46:37 +00:00
|
|
|
|
if (blendv == viewsector->ColorMap->Fade) blendv = 0;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
// a little hack to make this work for Legacy maps.
|
2014-05-11 12:46:37 +00:00
|
|
|
|
if (blendv.a == 0 && blendv != 0) blendv.a = 128;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-11 12:46:37 +00:00
|
|
|
|
if (blendv.a == 0)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
2014-05-11 12:46:37 +00:00
|
|
|
|
blendv = R_BlendForColormap(blendv);
|
|
|
|
|
if (blendv.a == 255)
|
|
|
|
|
{
|
|
|
|
|
// The calculated average is too dark so brighten it according to the palettes's overall brightness
|
|
|
|
|
int maxcol = MAX<int>(MAX<int>(framebuffer->palette_brightness, blendv.r), MAX<int>(blendv.g, blendv.b));
|
|
|
|
|
blendv.r = blendv.r * 255 / maxcol;
|
|
|
|
|
blendv.g = blendv.g * 255 / maxcol;
|
|
|
|
|
blendv.b = blendv.b * 255 / maxcol;
|
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
2014-05-11 12:46:37 +00:00
|
|
|
|
if (blendv.a == 255)
|
|
|
|
|
{
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-05-11 12:46:37 +00:00
|
|
|
|
extra_red = blendv.r / 255.0f;
|
|
|
|
|
extra_green = blendv.g / 255.0f;
|
|
|
|
|
extra_blue = blendv.b / 255.0f;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-05-11 12:46:37 +00:00
|
|
|
|
// If this is a multiplicative blend do it separately and add the additive ones on top of it.
|
|
|
|
|
blendv = 0;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-05-11 12:46:37 +00:00
|
|
|
|
// black multiplicative blends are ignored
|
|
|
|
|
if (extra_red || extra_green || extra_blue)
|
|
|
|
|
{
|
|
|
|
|
gl_RenderState.BlendFunc(GL_DST_COLOR, GL_ZERO);
|
2014-05-11 20:57:42 +00:00
|
|
|
|
gl_RenderState.SetColor(extra_red, extra_green, extra_blue, 1.0f);
|
2016-04-26 18:02:57 +00:00
|
|
|
|
gl_FillScreen();
|
2014-05-11 12:46:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (blendv.a)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
2014-05-11 12:46:37 +00:00
|
|
|
|
V_AddBlend(blendv.r / 255.f, blendv.g / 255.f, blendv.b / 255.f, blendv.a / 255.0f, blend);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-05-11 12:46:37 +00:00
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
|
// This mostly duplicates the code in shared_sbar.cpp
|
|
|
|
|
// When I was writing this the original was called too late so that I
|
|
|
|
|
// couldn't get the blend in time. However, since then I made some changes
|
|
|
|
|
// here that would get lost if I switched back so I won't do it.
|
|
|
|
|
|
|
|
|
|
if (player)
|
|
|
|
|
{
|
|
|
|
|
V_AddPlayerBlend(player, blend, 0.5, 175);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (players[consoleplayer].camera != NULL)
|
|
|
|
|
{
|
|
|
|
|
// except for fadeto effects
|
|
|
|
|
player_t *player = (players[consoleplayer].camera->player != NULL) ? players[consoleplayer].camera->player : &players[consoleplayer];
|
|
|
|
|
V_AddBlend (player->BlendR, player->BlendG, player->BlendB, player->BlendA, blend);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (blend[3]>0.0f)
|
|
|
|
|
{
|
|
|
|
|
gl_RenderState.BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
2014-05-11 20:57:42 +00:00
|
|
|
|
gl_RenderState.SetColor(blend[0], blend[1], blend[2], blend[3]);
|
2016-04-26 18:02:57 +00:00
|
|
|
|
gl_FillScreen();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Draws player sprites and color blend
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::EndDrawScene(sector_t * viewsector)
|
|
|
|
|
{
|
2014-07-26 20:26:17 +00:00
|
|
|
|
gl_RenderState.EnableFog(false);
|
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
|
// [BB] HUD models need to be rendered here. Make sure that
|
|
|
|
|
// DrawPlayerSprites is only called once. Either to draw
|
|
|
|
|
// HUD models or to draw the weapon sprites.
|
|
|
|
|
const bool renderHUDModel = gl_IsHUDModelForPlayerAvailable( players[consoleplayer].camera->player );
|
|
|
|
|
if ( renderHUDModel )
|
|
|
|
|
{
|
|
|
|
|
// [BB] The HUD model should be drawn over everything else already drawn.
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glClear(GL_DEPTH_BUFFER_BIT);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
DrawPlayerSprites (viewsector, true);
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDisable(GL_STENCIL_TEST);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
framebuffer->Begin2D(false);
|
|
|
|
|
|
|
|
|
|
ResetViewport();
|
|
|
|
|
// [BB] Only draw the sprites if we didn't render a HUD model before.
|
|
|
|
|
if ( renderHUDModel == false )
|
|
|
|
|
{
|
|
|
|
|
DrawPlayerSprites (viewsector, false);
|
|
|
|
|
}
|
2016-04-26 18:02:57 +00:00
|
|
|
|
int cm = gl_RenderState.GetFixedColormap();
|
2014-05-11 19:47:54 +00:00
|
|
|
|
gl_RenderState.SetFixedColormap(CM_DEFAULT);
|
2014-05-12 12:45:41 +00:00
|
|
|
|
gl_RenderState.SetSoftLightLevel(-1);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
DrawTargeterSprites();
|
|
|
|
|
DrawBlend(viewsector);
|
2016-04-26 18:02:57 +00:00
|
|
|
|
if (gl.glslversion == 0.0)
|
|
|
|
|
{
|
|
|
|
|
gl_RenderState.SetFixedColormap(cm);
|
|
|
|
|
gl_RenderState.DrawColormapOverlay();
|
|
|
|
|
gl_RenderState.SetFixedColormap(CM_DEFAULT);
|
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// Restore standard rendering state
|
|
|
|
|
gl_RenderState.BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
2014-05-11 20:57:42 +00:00
|
|
|
|
gl_RenderState.ResetColor();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
gl_RenderState.EnableTexture(true);
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDisable(GL_SCISSOR_TEST);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// R_RenderView - renders one view - either the screen or a camera texture
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::ProcessScene(bool toscreen)
|
|
|
|
|
{
|
|
|
|
|
FDrawInfo::StartDrawInfo();
|
|
|
|
|
iter_dlightf = iter_dlight = draw_dlight = draw_dlightf = 0;
|
|
|
|
|
GLPortal::BeginScene();
|
|
|
|
|
|
2016-04-02 21:17:16 +00:00
|
|
|
|
int mapsection = R_PointInSubsector(ViewPos)->mapsection;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
memset(¤tmapsection[0], 0, currentmapsection.Size());
|
|
|
|
|
currentmapsection[mapsection>>3] |= 1 << (mapsection & 7);
|
|
|
|
|
DrawScene(toscreen);
|
|
|
|
|
FDrawInfo::EndDrawInfo();
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// gl_SetFixedColormap
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::SetFixedColormap (player_t *player)
|
|
|
|
|
{
|
|
|
|
|
gl_fixedcolormap=CM_DEFAULT;
|
|
|
|
|
|
|
|
|
|
// check for special colormaps
|
|
|
|
|
player_t * cplayer = player->camera->player;
|
|
|
|
|
if (cplayer)
|
|
|
|
|
{
|
2013-07-22 15:30:08 +00:00
|
|
|
|
if (cplayer->extralight == INT_MIN)
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
|
|
|
|
gl_fixedcolormap=CM_FIRSTSPECIALCOLORMAP + INVERSECOLORMAP;
|
|
|
|
|
extralight=0;
|
|
|
|
|
}
|
|
|
|
|
else if (cplayer->fixedcolormap != NOFIXEDCOLORMAP)
|
|
|
|
|
{
|
|
|
|
|
gl_fixedcolormap = CM_FIRSTSPECIALCOLORMAP + cplayer->fixedcolormap;
|
|
|
|
|
}
|
|
|
|
|
else if (cplayer->fixedlightlevel != -1)
|
|
|
|
|
{
|
|
|
|
|
for(AInventory * in = cplayer->mo->Inventory; in; in = in->Inventory)
|
|
|
|
|
{
|
|
|
|
|
PalEntry color = in->GetBlend ();
|
|
|
|
|
|
|
|
|
|
// Need special handling for light amplifiers
|
|
|
|
|
if (in->IsKindOf(RUNTIME_CLASS(APowerTorch)))
|
|
|
|
|
{
|
|
|
|
|
gl_fixedcolormap = cplayer->fixedlightlevel + CM_TORCH;
|
|
|
|
|
}
|
|
|
|
|
else if (in->IsKindOf(RUNTIME_CLASS(APowerLightAmp)))
|
|
|
|
|
{
|
|
|
|
|
gl_fixedcolormap = CM_LITE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-05-11 19:47:54 +00:00
|
|
|
|
gl_RenderState.SetFixedColormap(gl_fixedcolormap);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Renders one viewpoint in a scene
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
sector_t * FGLRenderer::RenderViewpoint (AActor * camera, GL_IRECT * bounds, float fov, float ratio, float fovratio, bool mainview, bool toscreen)
|
|
|
|
|
{
|
|
|
|
|
sector_t * retval;
|
|
|
|
|
R_SetupFrame (camera);
|
|
|
|
|
SetViewArea();
|
2014-12-31 11:04:55 +00:00
|
|
|
|
|
|
|
|
|
// We have to scale the pitch to account for the pixel stretching, because the playsim doesn't know about this and treats it as 1:1.
|
2016-04-02 21:17:16 +00:00
|
|
|
|
double radPitch = clamp(ViewPitch.Normalized180().Radians(), -PI / 2, PI / 2);
|
2014-12-31 11:04:55 +00:00
|
|
|
|
double angx = cos(radPitch);
|
|
|
|
|
double angy = sin(radPitch) * glset.pixelstretch;
|
|
|
|
|
double alen = sqrt(angx*angx + angy*angy);
|
|
|
|
|
|
|
|
|
|
mAngles.Pitch = (float)RAD2DEG(asin(angy / alen));
|
2016-03-21 01:57:02 +00:00
|
|
|
|
mAngles.Roll.Degrees = camera->Angles.Roll.Degrees;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// Scroll the sky
|
|
|
|
|
mSky1Pos = (float)fmod(gl_frameMS * level.skyspeed1, 1024.f) * 90.f/256.f;
|
|
|
|
|
mSky2Pos = (float)fmod(gl_frameMS * level.skyspeed2, 1024.f) * 90.f/256.f;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (camera->player && camera->player-players==consoleplayer &&
|
|
|
|
|
((camera->player->cheats & CF_CHASECAM) || (r_deathcamera && camera->health <= 0)) && camera==camera->player->mo)
|
|
|
|
|
{
|
|
|
|
|
mViewActor=NULL;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
mViewActor=camera;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-03 23:45:44 +00:00
|
|
|
|
// 'viewsector' will not survive the rendering so it cannot be used anymore below.
|
2013-06-23 07:49:34 +00:00
|
|
|
|
retval = viewsector;
|
|
|
|
|
|
2015-10-31 00:51:35 +00:00
|
|
|
|
// Render (potentially) multiple views for stereo 3d
|
|
|
|
|
float viewShift[3];
|
|
|
|
|
const s3d::Stereo3DMode& stereo3dMode = s3d::Stereo3DMode::getCurrentMode();
|
|
|
|
|
stereo3dMode.SetUp();
|
2015-12-31 14:36:37 +00:00
|
|
|
|
for (int eye_ix = 0; eye_ix < stereo3dMode.eye_count(); ++eye_ix)
|
2015-10-31 00:51:35 +00:00
|
|
|
|
{
|
2015-12-31 14:36:37 +00:00
|
|
|
|
const s3d::EyePose * eye = stereo3dMode.getEyePose(eye_ix);
|
|
|
|
|
eye->SetUp();
|
2015-10-31 00:51:35 +00:00
|
|
|
|
// TODO: stereo specific viewport - needed when implementing side-by-side modes etc.
|
|
|
|
|
SetViewport(bounds);
|
|
|
|
|
mCurrentFoV = fov;
|
|
|
|
|
// Stereo mode specific perspective projection
|
2015-12-31 14:36:37 +00:00
|
|
|
|
SetProjection( eye->GetProjection(fov, ratio, fovratio) );
|
2015-10-31 01:29:36 +00:00
|
|
|
|
// SetProjection(fov, ratio, fovratio); // switch to perspective mode and set up clipper
|
2016-04-02 21:17:16 +00:00
|
|
|
|
SetViewAngle(ViewAngle);
|
2015-10-31 00:51:35 +00:00
|
|
|
|
// Stereo mode specific viewpoint adjustment - temporarily shifts global viewx, viewy, viewz
|
2016-03-21 01:57:02 +00:00
|
|
|
|
eye->GetViewShift(GLRenderer->mAngles.Yaw.Degrees, viewShift);
|
2015-10-31 00:51:35 +00:00
|
|
|
|
s3d::ScopedViewShifter viewShifter(viewShift);
|
2016-04-02 21:17:16 +00:00
|
|
|
|
SetViewMatrix(ViewPos.X, ViewPos.Y, ViewPos.Z, false, false);
|
2015-10-31 00:51:35 +00:00
|
|
|
|
gl_RenderState.ApplyMatrices();
|
|
|
|
|
|
|
|
|
|
clipper.Clear();
|
|
|
|
|
angle_t a1 = FrustumAngle();
|
2016-04-02 21:17:16 +00:00
|
|
|
|
clipper.SafeAddClipRangeRealAngles(ViewAngle.BAMs() + a1, ViewAngle.BAMs() - a1);
|
2015-10-31 00:51:35 +00:00
|
|
|
|
|
|
|
|
|
ProcessScene(toscreen);
|
2016-01-03 23:45:44 +00:00
|
|
|
|
if (mainview) EndDrawScene(retval); // do not call this for camera textures.
|
2015-12-31 14:36:37 +00:00
|
|
|
|
eye->TearDown();
|
2015-10-31 00:51:35 +00:00
|
|
|
|
}
|
|
|
|
|
stereo3dMode.TearDown();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
gl_frameCount++; // This counter must be increased right before the interpolations are restored.
|
|
|
|
|
interpolator.RestoreInterpolations ();
|
|
|
|
|
return retval;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// renders the view
|
|
|
|
|
//
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::RenderView (player_t* player)
|
|
|
|
|
{
|
|
|
|
|
OpenGLFrameBuffer* GLTarget = static_cast<OpenGLFrameBuffer*>(screen);
|
|
|
|
|
AActor *&LastCamera = GLTarget->LastCamera;
|
|
|
|
|
|
2014-08-30 12:33:06 +00:00
|
|
|
|
checkBenchActive();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
if (player->camera != LastCamera)
|
|
|
|
|
{
|
|
|
|
|
// If the camera changed don't interpolate
|
|
|
|
|
// Otherwise there will be some not so nice effects.
|
|
|
|
|
R_ResetViewInterpolation();
|
|
|
|
|
LastCamera=player->camera;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-10 19:47:07 +00:00
|
|
|
|
gl_RenderState.SetVertexBuffer(mVBO);
|
2014-05-10 23:23:27 +00:00
|
|
|
|
GLRenderer->mVBO->Reset();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// reset statistics counters
|
|
|
|
|
ResetProfilingData();
|
|
|
|
|
|
|
|
|
|
// Get this before everything else
|
2016-03-30 18:01:44 +00:00
|
|
|
|
if (cl_capfps || r_NoInterpolate) r_TicFracF = 1.;
|
|
|
|
|
else r_TicFracF = I_GetTimeFrac (&r_FrameTime);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
gl_frameMS = I_MSTime();
|
|
|
|
|
|
|
|
|
|
P_FindParticleSubsectors ();
|
|
|
|
|
|
2016-04-26 11:50:05 +00:00
|
|
|
|
if (gl.lightmethod != LM_SOFTWARE) GLRenderer->mLights->Clear();
|
2014-09-21 19:01:11 +00:00
|
|
|
|
|
2016-03-05 11:46:47 +00:00
|
|
|
|
// NoInterpolateView should have no bearing on camera textures, but needs to be preserved for the main view below.
|
|
|
|
|
bool saved_niv = NoInterpolateView;
|
|
|
|
|
NoInterpolateView = false;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
// prepare all camera textures that have been used in the last frame
|
|
|
|
|
FCanvasTextureInfo::UpdateAll();
|
2016-03-05 12:09:49 +00:00
|
|
|
|
NoInterpolateView = saved_niv;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// I stopped using BaseRatioSizes here because the information there wasn't well presented.
|
|
|
|
|
// 4:3 16:9 16:10 17:10 5:4
|
2016-03-04 10:47:15 +00:00
|
|
|
|
static float ratios[]={1.333333f, 1.777777f, 1.6f, 1.7f, 1.25f, 1.7f, 2.333333f};
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// now render the main view
|
|
|
|
|
float fovratio;
|
|
|
|
|
float ratio = ratios[WidescreenRatio];
|
2016-03-04 10:47:15 +00:00
|
|
|
|
if (! Is54Aspect(WidescreenRatio))
|
2013-06-23 07:49:34 +00:00
|
|
|
|
{
|
2014-12-29 21:42:19 +00:00
|
|
|
|
fovratio = 1.333333f;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
fovratio = ratio;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SetFixedColormap (player);
|
|
|
|
|
|
|
|
|
|
// Check if there's some lights. If not some code can be skipped.
|
|
|
|
|
TThinkerIterator<ADynamicLight> it(STAT_DLIGHT);
|
|
|
|
|
GLRenderer->mLightCount = ((it.Next()) != NULL);
|
|
|
|
|
|
|
|
|
|
sector_t * viewsector = RenderViewpoint(player->camera, NULL, FieldOfView * 360.0f / FINEANGLES, ratio, fovratio, true, true);
|
|
|
|
|
|
|
|
|
|
All.Unclock();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
// Render the view to a savegame picture
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLRenderer::WriteSavePic (player_t *player, FILE *file, int width, int height)
|
|
|
|
|
{
|
|
|
|
|
GL_IRECT bounds;
|
|
|
|
|
|
|
|
|
|
bounds.left=0;
|
|
|
|
|
bounds.top=0;
|
|
|
|
|
bounds.width=width;
|
|
|
|
|
bounds.height=height;
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glFlush();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
SetFixedColormap(player);
|
2014-05-10 23:23:27 +00:00
|
|
|
|
gl_RenderState.SetVertexBuffer(mVBO);
|
|
|
|
|
GLRenderer->mVBO->Reset();
|
2016-04-26 11:50:05 +00:00
|
|
|
|
if (gl.lightmethod != LM_SOFTWARE) GLRenderer->mLights->Clear();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
// Check if there's some lights. If not some code can be skipped.
|
|
|
|
|
TThinkerIterator<ADynamicLight> it(STAT_DLIGHT);
|
|
|
|
|
GLRenderer->mLightCount = ((it.Next()) != NULL);
|
|
|
|
|
|
|
|
|
|
sector_t *viewsector = RenderViewpoint(players[consoleplayer].camera, &bounds,
|
|
|
|
|
FieldOfView * 360.0f / FINEANGLES, 1.6f, 1.6f, true, false);
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glDisable(GL_STENCIL_TEST);
|
2014-05-11 19:47:54 +00:00
|
|
|
|
gl_RenderState.SetFixedColormap(CM_DEFAULT);
|
2014-05-12 12:45:41 +00:00
|
|
|
|
gl_RenderState.SetSoftLightLevel(-1);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
screen->Begin2D(false);
|
|
|
|
|
DrawBlend(viewsector);
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glFlush();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
byte * scr = (byte *)M_Malloc(width * height * 3);
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glReadPixels(0,0,width, height,GL_RGB,GL_UNSIGNED_BYTE,scr);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
M_CreatePNG (file, scr + ((height-1) * width * 3), NULL, SS_RGB, width, height, -width*3);
|
|
|
|
|
M_Free(scr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
struct FGLInterface : public FRenderer
|
|
|
|
|
{
|
|
|
|
|
bool UsesColormap() const;
|
|
|
|
|
void PrecacheTexture(FTexture *tex, int cache);
|
|
|
|
|
void RenderView(player_t *player);
|
|
|
|
|
void WriteSavePic (player_t *player, FILE *file, int width, int height);
|
|
|
|
|
void StateChanged(AActor *actor);
|
|
|
|
|
void StartSerialize(FArchive &arc);
|
|
|
|
|
void EndSerialize(FArchive &arc);
|
|
|
|
|
void RenderTextureView (FCanvasTexture *self, AActor *viewpoint, int fov);
|
|
|
|
|
sector_t *FakeFlat(sector_t *sec, sector_t *tempsec, int *floorlightlevel, int *ceilinglightlevel, bool back);
|
|
|
|
|
void SetFogParams(int _fogdensity, PalEntry _outsidefogcolor, int _outsidefogdensity, int _skyfog);
|
|
|
|
|
void PreprocessLevel();
|
|
|
|
|
void CleanLevelData();
|
|
|
|
|
bool RequireGLNodes();
|
|
|
|
|
|
|
|
|
|
int GetMaxViewPitch(bool down);
|
|
|
|
|
void ClearBuffer(int color);
|
|
|
|
|
void Init();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
// The GL renderer has no use for colormaps so let's
|
|
|
|
|
// not create them and save us some time.
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
bool FGLInterface::UsesColormap() const
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
|
//
|
|
|
|
|
// DFrameBuffer :: PrecacheTexture
|
|
|
|
|
//
|
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::PrecacheTexture(FTexture *tex, int cache)
|
|
|
|
|
{
|
|
|
|
|
if (tex != NULL)
|
|
|
|
|
{
|
|
|
|
|
if (cache)
|
|
|
|
|
{
|
2014-08-22 21:50:38 +00:00
|
|
|
|
tex->PrecacheGL(cache);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tex->UncacheGL();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
|
//
|
|
|
|
|
// DFrameBuffer :: StateChanged
|
|
|
|
|
//
|
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::StateChanged(AActor *actor)
|
|
|
|
|
{
|
|
|
|
|
gl_SetActorLights(actor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
// notify the renderer that serialization of the curent level is about to start/end
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::StartSerialize(FArchive &arc)
|
|
|
|
|
{
|
|
|
|
|
gl_DeleteAllAttachedLights();
|
2014-11-16 08:08:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void gl_SerializeGlobals(FArchive &arc)
|
|
|
|
|
{
|
2013-06-23 07:49:34 +00:00
|
|
|
|
arc << fogdensity << outsidefogdensity << skyfog;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FGLInterface::EndSerialize(FArchive &arc)
|
|
|
|
|
{
|
|
|
|
|
gl_RecreateAllAttachedLights();
|
|
|
|
|
if (arc.IsLoading()) gl_InitPortals();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
// Get max. view angle (renderer specific information so it goes here now)
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
EXTERN_CVAR(Float, maxviewpitch)
|
|
|
|
|
|
|
|
|
|
int FGLInterface::GetMaxViewPitch(bool down)
|
|
|
|
|
{
|
|
|
|
|
return int(maxviewpitch);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::ClearBuffer(int color)
|
|
|
|
|
{
|
|
|
|
|
PalEntry pe = GPalette.BaseColors[color];
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glClearColor(pe.r/255.f, pe.g/255.f, pe.b/255.f, 1.f);
|
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
// Render the view to a savegame picture
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::WriteSavePic (player_t *player, FILE *file, int width, int height)
|
|
|
|
|
{
|
|
|
|
|
GLRenderer->WriteSavePic(player, file, width, height);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::RenderView(player_t *player)
|
|
|
|
|
{
|
|
|
|
|
GLRenderer->RenderView(player);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::Init()
|
|
|
|
|
{
|
|
|
|
|
gl_ParseDefs();
|
2014-06-19 13:22:00 +00:00
|
|
|
|
gl_InitData();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
// Camera texture rendering
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
CVAR(Bool, gl_usefb, false , CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
|
|
|
|
extern TexFilter_s TexFilter[];
|
|
|
|
|
|
|
|
|
|
void FGLInterface::RenderTextureView (FCanvasTexture *tex, AActor *Viewpoint, int FOV)
|
|
|
|
|
{
|
2014-08-22 21:50:38 +00:00
|
|
|
|
FMaterial * gltex = FMaterial::ValidateTexture(tex, false);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2014-08-22 21:50:38 +00:00
|
|
|
|
int width = gltex->TextureWidth();
|
|
|
|
|
int height = gltex->TextureHeight();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
gl_fixedcolormap=CM_DEFAULT;
|
2014-05-11 19:47:54 +00:00
|
|
|
|
gl_RenderState.SetFixedColormap(CM_DEFAULT);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
2016-04-27 00:56:55 +00:00
|
|
|
|
bool usefb = gl_usefb || gltex->GetWidth() > screen->GetWidth() || gltex->GetHeight() > screen->GetHeight();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
if (!usefb)
|
|
|
|
|
{
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glFlush();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
#if defined(_WIN32) && (defined(_MSC_VER) || defined(__INTEL_COMPILER))
|
|
|
|
|
__try
|
|
|
|
|
#endif
|
|
|
|
|
{
|
|
|
|
|
GLRenderer->StartOffscreen();
|
|
|
|
|
gltex->BindToFrameBuffer();
|
|
|
|
|
}
|
|
|
|
|
#if defined(_WIN32) && (defined(_MSC_VER) || defined(__INTEL_COMPILER))
|
|
|
|
|
__except(1)
|
|
|
|
|
{
|
|
|
|
|
usefb = false;
|
|
|
|
|
gl_usefb = false;
|
|
|
|
|
GLRenderer->EndOffscreen();
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glFlush();
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GL_IRECT bounds;
|
|
|
|
|
bounds.left=bounds.top=0;
|
2014-08-22 21:50:38 +00:00
|
|
|
|
bounds.width=FHardwareTexture::GetTexDimension(gltex->GetWidth());
|
|
|
|
|
bounds.height=FHardwareTexture::GetTexDimension(gltex->GetHeight());
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
GLRenderer->RenderViewpoint(Viewpoint, &bounds, FOV, (float)width/height, (float)width/height, false, false);
|
|
|
|
|
|
|
|
|
|
if (!usefb)
|
|
|
|
|
{
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glFlush();
|
2014-09-09 10:00:42 +00:00
|
|
|
|
gl_RenderState.SetMaterial(gltex, 0, 0, -1, false);
|
2013-09-03 12:05:41 +00:00
|
|
|
|
glCopyTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, 0, 0, bounds.width, bounds.height);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
GLRenderer->EndOffscreen();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tex->SetUpdated();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
|
|
sector_t *FGLInterface::FakeFlat(sector_t *sec, sector_t *tempsec, int *floorlightlevel, int *ceilinglightlevel, bool back)
|
|
|
|
|
{
|
|
|
|
|
if (floorlightlevel != NULL)
|
|
|
|
|
{
|
|
|
|
|
*floorlightlevel = sec->GetFloorLight ();
|
|
|
|
|
}
|
|
|
|
|
if (ceilinglightlevel != NULL)
|
|
|
|
|
{
|
|
|
|
|
*ceilinglightlevel = sec->GetCeilingLight ();
|
|
|
|
|
}
|
|
|
|
|
return gl_FakeFlat(sec, tempsec, back);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
void FGLInterface::SetFogParams(int _fogdensity, PalEntry _outsidefogcolor, int _outsidefogdensity, int _skyfog)
|
|
|
|
|
{
|
|
|
|
|
gl_SetFogParams(_fogdensity, _outsidefogcolor, _outsidefogdensity, _skyfog);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FGLInterface::PreprocessLevel()
|
|
|
|
|
{
|
|
|
|
|
gl_PreprocessLevel();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FGLInterface::CleanLevelData()
|
|
|
|
|
{
|
|
|
|
|
gl_CleanLevelData();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool FGLInterface::RequireGLNodes()
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
|
|
FRenderer *gl_CreateInterface()
|
|
|
|
|
{
|
|
|
|
|
return new FGLInterface;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|