2017-01-12 15:21:46 +00:00
|
|
|
//
|
|
|
|
// Copyright (C) 1993-1996 by id Software, Inc.
|
|
|
|
//
|
|
|
|
// This source is available for distribution and/or modification
|
|
|
|
// only under the terms of the DOOM Source Code License as
|
|
|
|
// published by id Software. All rights reserved.
|
|
|
|
//
|
|
|
|
// The source is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the DOOM Source Code License
|
|
|
|
// for more details.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <float.h>
|
|
|
|
|
|
|
|
#include "templates.h"
|
|
|
|
#include "i_system.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "doomdef.h"
|
|
|
|
#include "doomstat.h"
|
|
|
|
#include "r_sky.h"
|
|
|
|
#include "stats.h"
|
|
|
|
#include "v_video.h"
|
|
|
|
#include "a_sharedglobal.h"
|
|
|
|
#include "c_console.h"
|
|
|
|
#include "cmdlib.h"
|
|
|
|
#include "d_net.h"
|
|
|
|
#include "g_level.h"
|
|
|
|
#include "r_utility.h"
|
2017-02-02 14:10:06 +00:00
|
|
|
#include "swrenderer/viewport/r_viewport.h"
|
2017-01-12 15:21:46 +00:00
|
|
|
#include "swrenderer/scene/r_light.h"
|
|
|
|
#include "swrenderer/drawers/r_draw.h"
|
|
|
|
#include "swrenderer/things/r_playersprite.h"
|
|
|
|
#include "swrenderer/plane/r_flatplane.h"
|
2017-02-03 07:06:47 +00:00
|
|
|
#include "swrenderer/drawers/r_draw_pal.h"
|
|
|
|
#include "swrenderer/drawers/r_draw_rgba.h"
|
2017-01-12 15:21:46 +00:00
|
|
|
|
|
|
|
CVAR(String, r_viewsize, "", CVAR_NOSET)
|
|
|
|
|
|
|
|
namespace swrenderer
|
|
|
|
{
|
2017-01-15 03:06:52 +00:00
|
|
|
RenderViewport *RenderViewport::Instance()
|
|
|
|
{
|
|
|
|
static RenderViewport instance;
|
|
|
|
return &instance;
|
|
|
|
}
|
|
|
|
|
2017-02-03 07:06:47 +00:00
|
|
|
RenderViewport::RenderViewport()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
RenderViewport::~RenderViewport()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-01-15 03:06:52 +00:00
|
|
|
void RenderViewport::SetViewport(int fullWidth, int fullHeight, float trueratio)
|
2017-01-12 15:21:46 +00:00
|
|
|
{
|
|
|
|
int virtheight, virtwidth, virtwidth2, virtheight2;
|
|
|
|
|
2017-02-01 20:59:48 +00:00
|
|
|
if (!RenderingToCanvas())
|
2017-01-12 15:21:46 +00:00
|
|
|
{ // Set r_viewsize cvar to reflect the current view size
|
|
|
|
UCVarValue value;
|
|
|
|
char temp[16];
|
|
|
|
|
|
|
|
mysnprintf(temp, countof(temp), "%d x %d", viewwidth, viewheight);
|
|
|
|
value.String = temp;
|
|
|
|
r_viewsize.ForceSet(value, CVAR_String);
|
|
|
|
}
|
|
|
|
|
|
|
|
fuzzviewheight = viewheight - 2; // Maximum row the fuzzer can draw to
|
|
|
|
|
|
|
|
CenterX = centerx;
|
|
|
|
CenterY = centery;
|
|
|
|
|
|
|
|
virtwidth = virtwidth2 = fullWidth;
|
|
|
|
virtheight = virtheight2 = fullHeight;
|
|
|
|
|
|
|
|
if (AspectTallerThanWide(trueratio))
|
|
|
|
{
|
|
|
|
virtheight2 = virtheight2 * AspectMultiplier(trueratio) / 48;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
virtwidth2 = virtwidth2 * AspectMultiplier(trueratio) / 48;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (AspectTallerThanWide(WidescreenRatio))
|
|
|
|
{
|
|
|
|
virtheight = virtheight * AspectMultiplier(WidescreenRatio) / 48;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
virtwidth = virtwidth * AspectMultiplier(WidescreenRatio) / 48;
|
|
|
|
}
|
|
|
|
|
|
|
|
BaseYaspectMul = 320.0 * virtheight2 / (r_Yaspect * virtwidth2);
|
|
|
|
YaspectMul = 320.0 * virtheight / (r_Yaspect * virtwidth);
|
|
|
|
IYaspectMul = (double)virtwidth * r_Yaspect / 320.0 / virtheight;
|
|
|
|
InvZtoScale = YaspectMul * CenterX;
|
|
|
|
|
|
|
|
WallTMapScale2 = IYaspectMul / CenterX;
|
|
|
|
|
|
|
|
// thing clipping
|
|
|
|
fillshort(screenheightarray, viewwidth, (short)viewheight);
|
|
|
|
|
2017-01-15 03:06:52 +00:00
|
|
|
InitTextureMapping();
|
2017-01-12 15:21:46 +00:00
|
|
|
|
|
|
|
// Reset r_*Visibility vars
|
2017-01-26 08:49:07 +00:00
|
|
|
LightVisibility *visibility = LightVisibility::Instance();
|
|
|
|
visibility->SetVisibility(visibility->GetVisibility());
|
2017-01-15 03:06:52 +00:00
|
|
|
|
|
|
|
SetupBuffer();
|
2017-01-12 15:21:46 +00:00
|
|
|
}
|
|
|
|
|
2017-01-15 03:06:52 +00:00
|
|
|
void RenderViewport::SetupFreelook()
|
2017-01-12 15:21:46 +00:00
|
|
|
{
|
|
|
|
double dy;
|
|
|
|
|
|
|
|
if (camera != NULL)
|
|
|
|
{
|
|
|
|
dy = FocalLengthY * (-ViewPitch).Tan();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dy = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
CenterY = (viewheight / 2.0) + dy;
|
|
|
|
centery = xs_ToInt(CenterY);
|
|
|
|
globaluclip = -CenterY / InvZtoScale;
|
|
|
|
globaldclip = (viewheight - CenterY) / InvZtoScale;
|
|
|
|
}
|
|
|
|
|
2017-01-15 03:06:52 +00:00
|
|
|
void RenderViewport::SetupBuffer()
|
2017-01-12 15:21:46 +00:00
|
|
|
{
|
2017-02-01 21:24:34 +00:00
|
|
|
R_InitFuzzTable(RenderTarget->GetPitch());
|
|
|
|
R_InitParticleTexture();
|
|
|
|
}
|
2017-02-01 20:52:29 +00:00
|
|
|
|
2017-02-01 21:24:34 +00:00
|
|
|
uint8_t *RenderViewport::GetDest(int x, int y)
|
|
|
|
{
|
|
|
|
x += viewwindowx;
|
|
|
|
y += viewwindowy;
|
2017-01-13 14:43:06 +00:00
|
|
|
|
2017-02-01 21:24:34 +00:00
|
|
|
int pitch = RenderTarget->GetPitch();
|
|
|
|
int pixelsize = RenderTarget->IsBgra() ? 4 : 1;
|
|
|
|
return RenderTarget->GetBuffer() + (x + y * pitch) * pixelsize;
|
2017-01-12 15:21:46 +00:00
|
|
|
}
|
|
|
|
|
2017-01-15 03:06:52 +00:00
|
|
|
void RenderViewport::InitTextureMapping()
|
2017-01-12 15:21:46 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
// Calc focallength so FieldOfView angles cover viewwidth.
|
|
|
|
FocalLengthX = CenterX / FocalTangent;
|
|
|
|
FocalLengthY = FocalLengthX * YaspectMul;
|
|
|
|
|
2017-02-11 21:10:52 +00:00
|
|
|
// This is 1/FocalTangent before the widescreen extension of FOV.
|
|
|
|
viewingrangerecip = FLOAT2FIXED(1. / tan(FieldOfView.Radians() / 2));
|
|
|
|
|
2017-01-12 15:21:46 +00:00
|
|
|
// Now generate xtoviewangle for sky texture mapping.
|
|
|
|
// [RH] Do not generate viewangletox, because texture mapping is no
|
|
|
|
// longer done with trig, so it's not needed.
|
|
|
|
const double slopestep = FocalTangent / centerx;
|
|
|
|
double slope;
|
|
|
|
|
|
|
|
for (i = centerx, slope = 0; i <= viewwidth; i++, slope += slopestep)
|
|
|
|
{
|
|
|
|
xtoviewangle[i] = angle_t((2 * M_PI - atan(slope)) * (ANGLE_180 / M_PI));
|
|
|
|
}
|
|
|
|
for (i = 0; i < centerx; i++)
|
|
|
|
{
|
|
|
|
xtoviewangle[i] = 0 - xtoviewangle[viewwidth - i - 1];
|
|
|
|
}
|
|
|
|
}
|
2017-02-01 22:23:10 +00:00
|
|
|
|
|
|
|
DVector2 RenderViewport::PointWorldToView(const DVector2 &worldPos) const
|
|
|
|
{
|
|
|
|
double translatedX = worldPos.X - ViewPos.X;
|
|
|
|
double translatedY = worldPos.Y - ViewPos.Y;
|
|
|
|
return {
|
|
|
|
translatedX * ViewSin - translatedY * ViewCos,
|
|
|
|
translatedX * ViewTanCos + translatedY * ViewTanSin
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
DVector3 RenderViewport::PointWorldToView(const DVector3 &worldPos) const
|
|
|
|
{
|
|
|
|
double translatedX = worldPos.X - ViewPos.X;
|
|
|
|
double translatedY = worldPos.Y - ViewPos.Y;
|
|
|
|
double translatedZ = worldPos.Z - ViewPos.Z;
|
|
|
|
return {
|
|
|
|
translatedX * ViewSin - translatedY * ViewCos,
|
|
|
|
translatedZ,
|
|
|
|
translatedX * ViewTanCos + translatedY * ViewTanSin
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
DVector3 RenderViewport::PointWorldToScreen(const DVector3 &worldPos) const
|
|
|
|
{
|
|
|
|
return PointViewToScreen(PointWorldToView(worldPos));
|
|
|
|
}
|
|
|
|
|
|
|
|
DVector3 RenderViewport::PointViewToScreen(const DVector3 &viewPos) const
|
|
|
|
{
|
|
|
|
double screenX = CenterX + viewPos.X / viewPos.Z * CenterX;
|
|
|
|
double screenY = CenterY - viewPos.Y / viewPos.Z * InvZtoScale;
|
|
|
|
return { screenX, screenY, viewPos.Z };
|
|
|
|
}
|
|
|
|
|
|
|
|
DVector2 RenderViewport::ScaleViewToScreen(const DVector2 &scale, double viewZ, bool pixelstretch) const
|
|
|
|
{
|
|
|
|
double screenScaleX = scale.X / viewZ * CenterX;
|
|
|
|
double screenScaleY = scale.Y / viewZ * InvZtoScale;
|
|
|
|
if (!pixelstretch) screenScaleY /= YaspectMul;
|
|
|
|
return { screenScaleX, screenScaleY };
|
|
|
|
}
|
2017-01-12 15:21:46 +00:00
|
|
|
}
|