2016-03-01 15:47:10 +00:00
|
|
|
/*
|
|
|
|
** texture.cpp
|
|
|
|
** The base texture class
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 2004-2007 Randy Heit
|
2018-03-31 17:20:59 +00:00
|
|
|
** Copyright 2006-2018 Christoph Oelckers
|
2016-03-01 15:47:10 +00:00
|
|
|
** 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.
|
|
|
|
**
|
|
|
|
** 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 "doomtype.h"
|
|
|
|
#include "files.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "templates.h"
|
|
|
|
#include "i_system.h"
|
|
|
|
#include "r_data/r_translate.h"
|
|
|
|
#include "bitmap.h"
|
|
|
|
#include "colormatcher.h"
|
|
|
|
#include "c_dispatch.h"
|
|
|
|
#include "v_video.h"
|
|
|
|
#include "m_fixed.h"
|
2018-04-24 21:51:19 +00:00
|
|
|
#include "hwrenderer/textures/hw_material.h"
|
2018-04-25 12:58:13 +00:00
|
|
|
#include "hwrenderer/textures/hw_ihwtexture.h"
|
2018-12-08 13:06:16 +00:00
|
|
|
#include "swrenderer/textures/r_swtexture.h"
|
2018-12-08 19:44:28 +00:00
|
|
|
#include "imagehelpers.h"
|
2018-12-09 06:39:05 +00:00
|
|
|
#include "image.h"
|
2018-12-15 15:05:48 +00:00
|
|
|
#include "formats/multipatchtexture.h"
|
2018-12-15 15:29:37 +00:00
|
|
|
#include "g_levellocals.h"
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2018-12-09 06:39:05 +00:00
|
|
|
FTexture *CreateBrightmapTexture(FImageSource*);
|
2018-03-31 17:20:59 +00:00
|
|
|
|
2018-03-31 08:37:46 +00:00
|
|
|
// Make sprite offset adjustment user-configurable per renderer.
|
|
|
|
int r_spriteadjustSW, r_spriteadjustHW;
|
|
|
|
CUSTOM_CVAR(Int, r_spriteadjust, 2, CVAR_ARCHIVE | CVAR_GLOBALCONFIG)
|
|
|
|
{
|
|
|
|
r_spriteadjustHW = !!(self & 2);
|
|
|
|
r_spriteadjustSW = !!(self & 1);
|
|
|
|
TexMan.SpriteAdjustChanged();
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2018-12-08 19:44:28 +00:00
|
|
|
uint8_t ImageHelpers::GrayMap[256];
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
void FTexture::InitGrayMap()
|
|
|
|
{
|
|
|
|
for (int i = 0; i < 256; ++i)
|
|
|
|
{
|
2018-12-08 19:44:28 +00:00
|
|
|
ImageHelpers::GrayMap[i] = ColorMatcher.Pick(i, i, i);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Examines the lump contents to decide what type of texture to create,
|
|
|
|
// and creates the texture.
|
2018-12-09 16:10:51 +00:00
|
|
|
FTexture * FTexture::CreateTexture(const char *name, int lumpnum, ETextureType usetype)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-12-09 16:10:51 +00:00
|
|
|
if (lumpnum == -1) return nullptr;
|
|
|
|
|
|
|
|
auto image = FImageSource::GetImage(lumpnum, usetype);
|
|
|
|
if (image != nullptr)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-12-09 16:10:51 +00:00
|
|
|
FTexture *tex = new FImageTexture(image);
|
|
|
|
if (tex != nullptr)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-12-09 16:10:51 +00:00
|
|
|
tex->UseType = usetype;
|
|
|
|
if (usetype == ETextureType::Flat)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-12-09 16:10:51 +00:00
|
|
|
int w = tex->GetWidth();
|
|
|
|
int h = tex->GetHeight();
|
|
|
|
|
|
|
|
// Auto-scale flats with dimensions 128x128 and 256x256.
|
|
|
|
// In hindsight, a bad idea, but RandomLag made it sound better than it really is.
|
|
|
|
// Now we're stuck with this stupid behaviour.
|
|
|
|
if (w==128 && h==128)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-12-09 16:10:51 +00:00
|
|
|
tex->Scale.X = tex->Scale.Y = 2;
|
|
|
|
tex->bWorldPanning = true;
|
|
|
|
}
|
|
|
|
else if (w==256 && h==256)
|
|
|
|
{
|
|
|
|
tex->Scale.X = tex->Scale.Y = 4;
|
|
|
|
tex->bWorldPanning = true;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
2018-12-09 16:10:51 +00:00
|
|
|
tex->Name = name;
|
|
|
|
tex->Name.ToUpper();
|
|
|
|
return tex;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
2018-12-09 16:10:51 +00:00
|
|
|
return nullptr;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
FTexture::FTexture (const char *name, int lumpnum)
|
2018-03-31 08:37:46 +00:00
|
|
|
:
|
2018-12-07 01:13:11 +00:00
|
|
|
Scale(1,1), SourceLump(lumpnum),
|
2018-03-25 18:26:16 +00:00
|
|
|
UseType(ETextureType::Any), bNoDecals(false), bNoRemap0(false), bWorldPanning(false),
|
2018-03-31 17:20:59 +00:00
|
|
|
bMasked(true), bAlphaTexture(false), bHasCanvas(false), bWarped(0), bComplex(false), bMultiPatch(false), bKeepAround(false), bFullNameTexture(false),
|
2018-12-07 01:13:11 +00:00
|
|
|
Rotations(0xFFFF), SkyOffset(0), Width(0), Height(0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-03-31 17:20:59 +00:00
|
|
|
bBrightmapChecked = false;
|
|
|
|
bGlowing = false;
|
2018-04-02 11:00:33 +00:00
|
|
|
bAutoGlowing = false;
|
|
|
|
bFullbright = false;
|
|
|
|
bDisableFullbright = false;
|
|
|
|
bSkybox = false;
|
|
|
|
bNoCompress = false;
|
2018-04-24 21:51:19 +00:00
|
|
|
bNoExpand = false;
|
2018-03-31 17:20:59 +00:00
|
|
|
bTranslucent = -1;
|
|
|
|
|
|
|
|
|
2018-03-31 08:37:46 +00:00
|
|
|
_LeftOffset[0] = _LeftOffset[1] = _TopOffset[0] = _TopOffset[1] = 0;
|
2016-03-01 15:47:10 +00:00
|
|
|
id.SetInvalid();
|
|
|
|
if (name != NULL)
|
|
|
|
{
|
|
|
|
Name = name;
|
|
|
|
Name.ToUpper();
|
|
|
|
}
|
|
|
|
else if (lumpnum < 0)
|
|
|
|
{
|
|
|
|
Name = FString();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Wads.GetLumpName (Name, lumpnum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FTexture::~FTexture ()
|
|
|
|
{
|
|
|
|
FTexture *link = Wads.GetLinkedTexture(SourceLump);
|
2018-03-31 17:20:59 +00:00
|
|
|
if (link == this) Wads.SetLinkedTexture(SourceLump, nullptr);
|
|
|
|
if (areas != nullptr) delete[] areas;
|
|
|
|
areas = nullptr;
|
2018-04-24 21:51:19 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
|
|
{
|
|
|
|
if (Material[i] != nullptr) delete Material[i];
|
|
|
|
Material[i] = nullptr;
|
|
|
|
}
|
2018-12-08 11:42:35 +00:00
|
|
|
if (SoftwareTexture != nullptr)
|
|
|
|
{
|
|
|
|
delete SoftwareTexture;
|
|
|
|
SoftwareTexture = nullptr;
|
|
|
|
}
|
2016-06-10 11:50:34 +00:00
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
void FTexture::SetFrontSkyLayer ()
|
|
|
|
{
|
|
|
|
bNoRemap0 = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
2018-12-10 01:35:10 +00:00
|
|
|
// FTexture::GetBgraBitmap
|
2016-03-01 15:47:10 +00:00
|
|
|
//
|
2018-12-10 01:35:10 +00:00
|
|
|
// Default returns just an empty bitmap. This needs to be overridden by
|
|
|
|
// any subclass that actually does return a software pixel buffer.
|
2016-03-01 15:47:10 +00:00
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2018-12-08 16:23:15 +00:00
|
|
|
FBitmap FTexture::GetBgraBitmap(PalEntry *remap, int *ptrans)
|
2018-12-08 13:06:16 +00:00
|
|
|
{
|
|
|
|
FBitmap bmp;
|
2018-12-10 01:35:10 +00:00
|
|
|
bmp.Create(Width, Height);
|
2018-12-08 13:06:16 +00:00
|
|
|
return bmp;
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
FTexture *FTexture::GetRawTexture()
|
|
|
|
{
|
2018-12-15 15:05:48 +00:00
|
|
|
if (OffsetLess) return OffsetLess;
|
|
|
|
// Reject anything that cannot have been a single-patch multipatch texture in vanilla.
|
|
|
|
auto image = static_cast<FMultiPatchTexture *>(GetImage());
|
|
|
|
if (bMultiPatch != 1 || UseType != ETextureType::Wall || Scale.X != 1 || Scale.Y != 1 || bWorldPanning || image == nullptr || image->NumParts != 1)
|
|
|
|
{
|
|
|
|
OffsetLess = this;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
// Set up a new texture that directly references the underlying patch.
|
|
|
|
// From here we cannot retrieve the original texture made for it, so just create a new one.
|
|
|
|
FImageSource *source = image->Parts[0].Image;
|
|
|
|
OffsetLess = new FImageTexture(source, "");
|
|
|
|
TexMan.AddTexture(OffsetLess);
|
|
|
|
return OffsetLess;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FTexture::SetScaledSize(int fitwidth, int fitheight)
|
|
|
|
{
|
2016-03-26 12:37:44 +00:00
|
|
|
Scale.X = double(Width) / fitwidth;
|
|
|
|
Scale.Y =double(Height) / fitheight;
|
2016-03-01 15:47:10 +00:00
|
|
|
// compensate for roundoff errors
|
2016-03-26 12:37:44 +00:00
|
|
|
if (int(Scale.X * fitwidth) != Width) Scale.X += (1 / 65536.);
|
|
|
|
if (int(Scale.Y * fitheight) != Height) Scale.Y += (1 / 65536.);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-10-19 21:52:09 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Gets the average color of a texture for use as a sky cap color
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2017-06-07 20:42:19 +00:00
|
|
|
PalEntry FTexture::averageColor(const uint32_t *data, int size, int maxout)
|
2016-10-19 21:52:09 +00:00
|
|
|
{
|
2017-06-07 20:42:19 +00:00
|
|
|
int i;
|
|
|
|
unsigned int r, g, b;
|
2016-10-19 21:52:09 +00:00
|
|
|
|
2017-06-07 20:42:19 +00:00
|
|
|
// First clear them.
|
|
|
|
r = g = b = 0;
|
|
|
|
if (size == 0)
|
|
|
|
{
|
|
|
|
return PalEntry(255, 255, 255);
|
|
|
|
}
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
{
|
|
|
|
b += BPART(data[i]);
|
|
|
|
g += GPART(data[i]);
|
|
|
|
r += RPART(data[i]);
|
|
|
|
}
|
2016-10-19 21:52:09 +00:00
|
|
|
|
2017-06-07 20:42:19 +00:00
|
|
|
r = r / size;
|
|
|
|
g = g / size;
|
|
|
|
b = b / size;
|
2016-10-19 21:52:09 +00:00
|
|
|
|
2017-06-07 20:42:19 +00:00
|
|
|
int maxv = MAX(MAX(r, g), b);
|
2016-10-19 21:52:09 +00:00
|
|
|
|
2017-06-07 20:42:19 +00:00
|
|
|
if (maxv && maxout)
|
|
|
|
{
|
|
|
|
r = ::Scale(r, maxout, maxv);
|
|
|
|
g = ::Scale(g, maxout, maxv);
|
|
|
|
b = ::Scale(b, maxout, maxv);
|
2016-10-19 21:52:09 +00:00
|
|
|
}
|
2017-06-07 20:42:19 +00:00
|
|
|
return PalEntry(255, r, g, b);
|
2016-10-19 21:52:09 +00:00
|
|
|
}
|
|
|
|
|
2016-10-20 07:08:07 +00:00
|
|
|
PalEntry FTexture::GetSkyCapColor(bool bottom)
|
2016-10-19 21:52:09 +00:00
|
|
|
{
|
|
|
|
if (!bSWSkyColorDone)
|
|
|
|
{
|
|
|
|
bSWSkyColorDone = true;
|
|
|
|
|
2018-12-08 16:23:15 +00:00
|
|
|
FBitmap bitmap = GetBgraBitmap(nullptr);
|
|
|
|
int w = bitmap.GetWidth();
|
|
|
|
int h = bitmap.GetHeight();
|
2016-10-19 21:52:09 +00:00
|
|
|
|
|
|
|
const uint32_t *buffer = (const uint32_t *)bitmap.GetPixels();
|
|
|
|
if (buffer)
|
|
|
|
{
|
2017-03-09 19:19:55 +00:00
|
|
|
CeilingSkyColor = averageColor((uint32_t *)buffer, w * MIN(30, h), 0);
|
2016-10-19 21:52:09 +00:00
|
|
|
if (h>30)
|
|
|
|
{
|
2017-03-09 19:19:55 +00:00
|
|
|
FloorSkyColor = averageColor(((uint32_t *)buffer) + (h - 30)*w, w * 30, 0);
|
2016-10-19 21:52:09 +00:00
|
|
|
}
|
2016-10-20 07:08:07 +00:00
|
|
|
else FloorSkyColor = CeilingSkyColor;
|
2016-10-19 21:52:09 +00:00
|
|
|
}
|
|
|
|
}
|
2016-10-20 07:08:07 +00:00
|
|
|
return bottom ? FloorSkyColor : CeilingSkyColor;
|
2016-10-19 21:52:09 +00:00
|
|
|
}
|
|
|
|
|
2017-03-17 22:08:22 +00:00
|
|
|
//====================================================================
|
|
|
|
//
|
|
|
|
// CheckRealHeight
|
|
|
|
//
|
|
|
|
// Checks the posts in a texture and returns the lowest row (plus one)
|
|
|
|
// of the texture that is actually used.
|
|
|
|
//
|
|
|
|
//====================================================================
|
|
|
|
|
|
|
|
int FTexture::CheckRealHeight()
|
|
|
|
{
|
2018-12-08 11:42:35 +00:00
|
|
|
auto pixels = Get8BitPixels(false);
|
2018-12-06 00:11:04 +00:00
|
|
|
|
|
|
|
for(int h = GetHeight()-1; h>= 0; h--)
|
2017-03-17 22:08:22 +00:00
|
|
|
{
|
2018-12-06 00:11:04 +00:00
|
|
|
for(int w = 0; w < GetWidth(); w++)
|
2017-03-17 22:08:22 +00:00
|
|
|
{
|
2018-12-06 00:11:04 +00:00
|
|
|
if (pixels[h + w * GetHeight()] != 0)
|
2017-03-17 22:08:22 +00:00
|
|
|
{
|
2018-12-06 00:11:04 +00:00
|
|
|
// Scale maxy before returning it
|
|
|
|
h = int((h * 2) / Scale.Y);
|
|
|
|
h = (h >> 1) + (h & 1);
|
|
|
|
return h;
|
2017-03-17 22:08:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-12-06 00:11:04 +00:00
|
|
|
return 0;
|
2017-03-17 22:08:22 +00:00
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Search auto paths for extra material textures
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FTexture::AddAutoMaterials()
|
|
|
|
{
|
|
|
|
struct AutoTextureSearchPath
|
|
|
|
{
|
|
|
|
const char *path;
|
|
|
|
FTexture *FTexture::*pointer;
|
|
|
|
};
|
|
|
|
|
|
|
|
static AutoTextureSearchPath autosearchpaths[] =
|
|
|
|
{
|
|
|
|
{ "brightmaps/", &FTexture::Brightmap }, // For backwards compatibility, only for short names
|
|
|
|
{ "materials/brightmaps/", &FTexture::Brightmap },
|
|
|
|
{ "materials/normalmaps/", &FTexture::Normal },
|
|
|
|
{ "materials/specular/", &FTexture::Specular },
|
|
|
|
{ "materials/metallic/", &FTexture::Metallic },
|
|
|
|
{ "materials/roughness/", &FTexture::Roughness },
|
|
|
|
{ "materials/ao/", &FTexture::AmbientOcclusion }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
int startindex = bFullNameTexture ? 1 : 0;
|
|
|
|
FString searchname = Name;
|
|
|
|
|
|
|
|
if (bFullNameTexture)
|
|
|
|
{
|
|
|
|
auto dot = searchname.LastIndexOf('.');
|
|
|
|
auto slash = searchname.LastIndexOf('/');
|
|
|
|
if (dot > slash) searchname.Truncate(dot);
|
|
|
|
}
|
2017-03-17 22:08:22 +00:00
|
|
|
|
2018-04-08 09:53:41 +00:00
|
|
|
for (size_t i = 0; i < countof(autosearchpaths); i++)
|
2018-03-31 17:20:59 +00:00
|
|
|
{
|
|
|
|
auto &layer = autosearchpaths[i];
|
|
|
|
if (this->*(layer.pointer) == nullptr) // only if no explicit assignment had been done.
|
|
|
|
{
|
|
|
|
FStringf lookup("%s%s%s", layer.path, bFullNameTexture ? "" : "auto/", searchname.GetChars());
|
|
|
|
auto lump = Wads.CheckNumForFullName(lookup, false, ns_global, true);
|
|
|
|
if (lump != -1)
|
|
|
|
{
|
|
|
|
auto bmtex = TexMan.FindTexture(Wads.GetLumpFullName(lump), ETextureType::Any, FTextureManager::TEXMAN_TryAny);
|
|
|
|
if (bmtex != nullptr)
|
|
|
|
{
|
|
|
|
bmtex->bMasked = false;
|
|
|
|
this->*(layer.pointer) = bmtex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-03-17 22:08:22 +00:00
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Checks if the texture has a default brightmap and creates it if so
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
void FTexture::CreateDefaultBrightmap()
|
|
|
|
{
|
|
|
|
if (!bBrightmapChecked)
|
|
|
|
{
|
|
|
|
// Check for brightmaps
|
2018-12-09 14:25:56 +00:00
|
|
|
if (GetImage() && GetImage()->UseGamePalette() && TexMan.HasGlobalBrightmap &&
|
2018-03-31 17:20:59 +00:00
|
|
|
UseType != ETextureType::Decal && UseType != ETextureType::MiscPatch && UseType != ETextureType::FontChar &&
|
2018-12-08 11:42:35 +00:00
|
|
|
Brightmap == NULL && bWarped == 0)
|
2018-03-31 17:20:59 +00:00
|
|
|
{
|
|
|
|
// May have one - let's check when we use this texture
|
2018-12-08 11:42:35 +00:00
|
|
|
auto texbuf = Get8BitPixels(false);
|
2018-03-31 17:20:59 +00:00
|
|
|
const int white = ColorMatcher.Pick(255, 255, 255);
|
|
|
|
|
|
|
|
int size = GetWidth() * GetHeight();
|
|
|
|
for (int i = 0; i<size; i++)
|
|
|
|
{
|
|
|
|
if (TexMan.GlobalBrightmap.Remap[texbuf[i]] == white)
|
|
|
|
{
|
|
|
|
// Create a brightmap
|
|
|
|
DPrintf(DMSG_NOTIFY, "brightmap created for texture '%s'\n", Name.GetChars());
|
2018-12-09 06:39:05 +00:00
|
|
|
Brightmap = CreateBrightmapTexture(static_cast<FImageTexture*>(this)->GetImage());
|
2018-03-31 17:20:59 +00:00
|
|
|
bBrightmapChecked = true;
|
|
|
|
TexMan.AddTexture(Brightmap);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// No bright pixels found
|
|
|
|
DPrintf(DMSG_SPAMMY, "No bright pixels found in texture '%s'\n", Name.GetChars());
|
|
|
|
bBrightmapChecked = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// does not have one so set the flag to 'done'
|
|
|
|
bBrightmapChecked = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Calculates glow color for a texture
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void FTexture::GetGlowColor(float *data)
|
|
|
|
{
|
|
|
|
if (bGlowing && GlowColor == 0)
|
|
|
|
{
|
2018-12-08 13:06:16 +00:00
|
|
|
auto buffer = GetBgraBitmap(nullptr);
|
|
|
|
GlowColor = averageColor((uint32_t*)buffer.GetPixels(), buffer.GetWidth() * buffer.GetHeight(), 153);
|
2018-03-31 17:20:59 +00:00
|
|
|
|
|
|
|
// Black glow equals nothing so switch glowing off
|
|
|
|
if (GlowColor == 0) bGlowing = false;
|
|
|
|
}
|
|
|
|
data[0] = GlowColor.r / 255.0f;
|
|
|
|
data[1] = GlowColor.g / 255.0f;
|
|
|
|
data[2] = GlowColor.b / 255.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Finds gaps in the texture which can be skipped by the renderer
|
|
|
|
// This was mainly added to speed up one area in E4M6 of 007LTSD
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
bool FTexture::FindHoles(const unsigned char * buffer, int w, int h)
|
|
|
|
{
|
|
|
|
const unsigned char * li;
|
|
|
|
int y, x;
|
|
|
|
int startdraw, lendraw;
|
|
|
|
int gaps[5][2];
|
|
|
|
int gapc = 0;
|
|
|
|
|
|
|
|
|
|
|
|
// already done!
|
|
|
|
if (areacount) return false;
|
|
|
|
if (UseType == ETextureType::Flat) return false; // flats don't have transparent parts
|
|
|
|
areacount = -1; //whatever happens next, it shouldn't be done twice!
|
|
|
|
|
|
|
|
// large textures are excluded for performance reasons
|
|
|
|
if (h>512) return false;
|
|
|
|
|
|
|
|
startdraw = -1;
|
|
|
|
lendraw = 0;
|
|
|
|
for (y = 0; y<h; y++)
|
|
|
|
{
|
|
|
|
li = buffer + w * y * 4 + 3;
|
|
|
|
|
|
|
|
for (x = 0; x<w; x++, li += 4)
|
|
|
|
{
|
|
|
|
if (*li != 0) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (x != w)
|
|
|
|
{
|
|
|
|
// non - transparent
|
|
|
|
if (startdraw == -1)
|
|
|
|
{
|
|
|
|
startdraw = y;
|
|
|
|
// merge transparent gaps of less than 16 pixels into the last drawing block
|
|
|
|
if (gapc && y <= gaps[gapc - 1][0] + gaps[gapc - 1][1] + 16)
|
|
|
|
{
|
|
|
|
gapc--;
|
|
|
|
startdraw = gaps[gapc][0];
|
|
|
|
lendraw = y - startdraw;
|
|
|
|
}
|
|
|
|
if (gapc == 4) return false; // too many splits - this isn't worth it
|
|
|
|
}
|
|
|
|
lendraw++;
|
|
|
|
}
|
|
|
|
else if (startdraw != -1)
|
|
|
|
{
|
|
|
|
if (lendraw == 1) lendraw = 2;
|
|
|
|
gaps[gapc][0] = startdraw;
|
|
|
|
gaps[gapc][1] = lendraw;
|
|
|
|
gapc++;
|
|
|
|
|
|
|
|
startdraw = -1;
|
|
|
|
lendraw = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (startdraw != -1)
|
|
|
|
{
|
|
|
|
gaps[gapc][0] = startdraw;
|
|
|
|
gaps[gapc][1] = lendraw;
|
|
|
|
gapc++;
|
|
|
|
}
|
|
|
|
if (startdraw == 0 && lendraw == h) return false; // nothing saved so don't create a split list
|
|
|
|
|
|
|
|
if (gapc > 0)
|
|
|
|
{
|
|
|
|
FloatRect * rcs = new FloatRect[gapc];
|
|
|
|
|
|
|
|
for (x = 0; x < gapc; x++)
|
|
|
|
{
|
|
|
|
// gaps are stored as texture (u/v) coordinates
|
|
|
|
rcs[x].width = rcs[x].left = -1.0f;
|
|
|
|
rcs[x].top = (float)gaps[x][0] / (float)h;
|
|
|
|
rcs[x].height = (float)gaps[x][1] / (float)h;
|
|
|
|
}
|
|
|
|
areas = rcs;
|
|
|
|
}
|
|
|
|
else areas = nullptr;
|
|
|
|
areacount = gapc;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void FTexture::CheckTrans(unsigned char * buffer, int size, int trans)
|
|
|
|
{
|
|
|
|
if (bTranslucent == -1)
|
|
|
|
{
|
|
|
|
bTranslucent = trans;
|
|
|
|
if (trans == -1)
|
|
|
|
{
|
|
|
|
uint32_t * dwbuf = (uint32_t*)buffer;
|
|
|
|
for (int i = 0; i<size; i++)
|
|
|
|
{
|
|
|
|
uint32_t alpha = dwbuf[i] >> 24;
|
|
|
|
|
|
|
|
if (alpha != 0xff && alpha != 0)
|
|
|
|
{
|
|
|
|
bTranslucent = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bTranslucent = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// smooth the edges of transparent fields in the texture
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
#define MSB 0
|
|
|
|
#define SOME_MASK 0xffffff00
|
|
|
|
#else
|
|
|
|
#define MSB 3
|
|
|
|
#define SOME_MASK 0x00ffffff
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define CHKPIX(ofs) (l1[(ofs)*4+MSB]==255 ? (( ((uint32_t*)l1)[0] = ((uint32_t*)l1)[ofs]&SOME_MASK), trans=true ) : false)
|
|
|
|
|
|
|
|
bool FTexture::SmoothEdges(unsigned char * buffer, int w, int h)
|
|
|
|
{
|
|
|
|
int x, y;
|
|
|
|
bool trans = buffer[MSB] == 0; // If I set this to false here the code won't detect textures
|
|
|
|
// that only contain transparent pixels.
|
|
|
|
bool semitrans = false;
|
|
|
|
unsigned char * l1;
|
|
|
|
|
|
|
|
if (h <= 1 || w <= 1) return false; // makes (a) no sense and (b) doesn't work with this code!
|
|
|
|
|
|
|
|
l1 = buffer;
|
|
|
|
|
|
|
|
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(1)) CHKPIX(w);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
l1 += 4;
|
|
|
|
for (x = 1; x<w - 1; x++, l1 += 4)
|
|
|
|
{
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-1) && !CHKPIX(1)) CHKPIX(w);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
}
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-1)) CHKPIX(w);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
l1 += 4;
|
|
|
|
|
|
|
|
for (y = 1; y<h - 1; y++)
|
|
|
|
{
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-w) && !CHKPIX(1)) CHKPIX(w);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
l1 += 4;
|
|
|
|
for (x = 1; x<w - 1; x++, l1 += 4)
|
|
|
|
{
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-w) && !CHKPIX(-1) && !CHKPIX(1) && !CHKPIX(-w - 1) && !CHKPIX(-w + 1) && !CHKPIX(w - 1) && !CHKPIX(w + 1)) CHKPIX(w);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
}
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-w) && !CHKPIX(-1)) CHKPIX(w);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
l1 += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-w)) CHKPIX(1);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
l1 += 4;
|
|
|
|
for (x = 1; x<w - 1; x++, l1 += 4)
|
|
|
|
{
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-w) && !CHKPIX(-1)) CHKPIX(1);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
}
|
|
|
|
if (l1[MSB] == 0 && !CHKPIX(-w)) CHKPIX(-1);
|
|
|
|
else if (l1[MSB]<255) semitrans = true;
|
|
|
|
|
|
|
|
return trans || semitrans;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Post-process the texture data after the buffer has been created
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
bool FTexture::ProcessData(unsigned char * buffer, int w, int h, bool ispatch)
|
|
|
|
{
|
|
|
|
if (bMasked)
|
|
|
|
{
|
|
|
|
bMasked = SmoothEdges(buffer, w, h);
|
|
|
|
if (bMasked && !ispatch) FindHoles(buffer, w, h);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-01 12:38:48 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Initializes the buffer for the texture data
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2018-12-11 19:26:33 +00:00
|
|
|
FTextureBuffer FTexture::CreateTexBuffer(int translation, int flags)
|
2018-04-01 12:38:48 +00:00
|
|
|
{
|
2018-12-11 19:26:33 +00:00
|
|
|
FTextureBuffer result;
|
|
|
|
|
2018-04-24 18:41:52 +00:00
|
|
|
unsigned char * buffer = nullptr;
|
2018-04-01 12:38:48 +00:00
|
|
|
int W, H;
|
|
|
|
int isTransparent = -1;
|
2018-12-11 23:46:58 +00:00
|
|
|
bool checkonly = !!(flags & CTF_CheckOnly);
|
2018-04-01 12:38:48 +00:00
|
|
|
|
2018-06-08 17:12:06 +00:00
|
|
|
if (flags & CTF_CheckHires)
|
2018-04-01 12:38:48 +00:00
|
|
|
{
|
2018-12-11 20:06:34 +00:00
|
|
|
// No image means that this cannot be checked,
|
2018-12-12 00:27:04 +00:00
|
|
|
if (GetImage() && LoadHiresTexture(result, checkonly)) return result;
|
2018-04-01 12:38:48 +00:00
|
|
|
}
|
2018-04-25 19:02:50 +00:00
|
|
|
int exx = !!(flags & CTF_Expand);
|
2018-04-01 12:38:48 +00:00
|
|
|
|
2018-12-11 19:26:33 +00:00
|
|
|
W = GetWidth() + 2 * exx;
|
|
|
|
H = GetHeight() + 2 * exx;
|
2018-04-01 12:38:48 +00:00
|
|
|
|
2018-12-11 23:46:58 +00:00
|
|
|
if (!checkonly)
|
|
|
|
{
|
|
|
|
buffer = new unsigned char[W*(H + 1) * 4];
|
|
|
|
memset(buffer, 0, W * (H + 1) * 4);
|
2018-04-01 12:38:48 +00:00
|
|
|
|
2018-12-11 23:46:58 +00:00
|
|
|
auto remap = translation <= 0 ? nullptr : FUniquePalette::GetPalette(translation);
|
|
|
|
FBitmap bmp(buffer, W * 4, W, H);
|
2018-04-24 18:41:52 +00:00
|
|
|
|
2018-12-11 23:46:58 +00:00
|
|
|
int trans;
|
|
|
|
auto Pixels = GetBgraBitmap(remap, &trans);
|
|
|
|
bmp.Blit(exx, exx, Pixels);
|
2018-12-08 16:23:15 +00:00
|
|
|
|
2018-12-11 23:46:58 +00:00
|
|
|
if (remap == nullptr)
|
|
|
|
{
|
|
|
|
CheckTrans(buffer, W*H, trans);
|
|
|
|
isTransparent = bTranslucent;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
isTransparent = 0;
|
|
|
|
// A translated image is not conclusive for setting the texture's transparency info.
|
|
|
|
}
|
2018-04-25 19:02:50 +00:00
|
|
|
}
|
|
|
|
|
2018-12-11 20:06:34 +00:00
|
|
|
if (GetImage())
|
|
|
|
{
|
|
|
|
FContentIdBuilder builder;
|
|
|
|
builder.id = 0;
|
|
|
|
builder.imageID = GetImage()->GetId();
|
|
|
|
builder.translation = MAX(0, translation);
|
|
|
|
builder.expand = exx;
|
|
|
|
result.mContentId = builder.id;
|
|
|
|
}
|
|
|
|
else result.mContentId = 0; // for non-image backed textures this has no meaning so leave it at 0.
|
2018-12-11 19:26:33 +00:00
|
|
|
|
|
|
|
result.mBuffer = buffer;
|
|
|
|
result.mWidth = W;
|
|
|
|
result.mHeight = H;
|
|
|
|
|
2018-12-11 20:06:34 +00:00
|
|
|
// Only do postprocessing for image-backed textures. (i.e. not for the burn texture which can also pass through here.)
|
|
|
|
if (GetImage() && flags & CTF_ProcessData)
|
2018-04-25 19:02:50 +00:00
|
|
|
{
|
2018-12-11 23:46:58 +00:00
|
|
|
CreateUpsampledTextureBuffer(result, !!isTransparent, checkonly);
|
|
|
|
if (!checkonly) ProcessData(result.mBuffer, result.mWidth, result.mHeight, false);
|
2018-06-08 17:12:06 +00:00
|
|
|
}
|
2018-04-01 12:38:48 +00:00
|
|
|
|
2018-12-11 19:26:33 +00:00
|
|
|
return result;
|
2018-04-01 12:38:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Dummy texture for the 0-entry.
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
bool FTexture::GetTranslucency()
|
|
|
|
{
|
|
|
|
if (bTranslucent == -1)
|
|
|
|
{
|
|
|
|
if (!bHasCanvas)
|
|
|
|
{
|
2018-12-11 20:06:34 +00:00
|
|
|
// This will calculate all we need, so just discard the result.
|
|
|
|
CreateTexBuffer(0);
|
2018-04-01 12:38:48 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bTranslucent = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return !!bTranslucent;
|
|
|
|
}
|
|
|
|
|
2018-04-24 21:51:19 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Sprite adjust has changed.
|
|
|
|
// This needs to alter the material's sprite rect.
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void FTexture::SetSpriteAdjust()
|
|
|
|
{
|
|
|
|
for (auto mat : Material)
|
|
|
|
{
|
|
|
|
if (mat != nullptr) mat->SetSpriteRect();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-08 19:19:57 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
2018-12-08 11:42:35 +00:00
|
|
|
// the default just returns an empty texture.
|
2018-04-08 19:19:57 +00:00
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2018-12-08 11:42:35 +00:00
|
|
|
TArray<uint8_t> FTexture::Get8BitPixels(bool alphatex)
|
2018-04-08 19:19:57 +00:00
|
|
|
{
|
2018-12-08 11:42:35 +00:00
|
|
|
TArray<uint8_t> Pixels(Width * Height, true);
|
|
|
|
memset(Pixels.Data(), 0, Width * Height);
|
|
|
|
return Pixels;
|
2018-04-08 19:19:57 +00:00
|
|
|
}
|
|
|
|
|
2018-10-29 06:39:33 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
FWrapperTexture::FWrapperTexture(int w, int h, int bits)
|
|
|
|
{
|
|
|
|
Width = w;
|
|
|
|
Height = h;
|
2018-12-07 01:13:11 +00:00
|
|
|
Format = bits;
|
2018-10-29 06:39:33 +00:00
|
|
|
UseType = ETextureType::SWCanvas;
|
|
|
|
bNoCompress = true;
|
2018-12-12 17:39:38 +00:00
|
|
|
auto hwtex = screen->CreateHardwareTexture();
|
|
|
|
// todo: Initialize here.
|
|
|
|
SystemTextures.AddHardwareTexture(0, false, hwtex);
|
2018-10-29 06:39:33 +00:00
|
|
|
}
|
|
|
|
|
2018-11-17 17:24:14 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Coordinate helper.
|
|
|
|
// The only reason this is even needed is that many years ago someone
|
|
|
|
// was convinced that having per-texel panning on walls was a good idea.
|
|
|
|
// If it wasn't for this relatively useless feature the entire positioning
|
|
|
|
// code for wall textures could be a lot simpler.
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
float FTexCoordInfo::RowOffset(float rowoffset) const
|
|
|
|
{
|
|
|
|
float scale = fabs(mScale.Y);
|
2018-12-14 21:34:28 +00:00
|
|
|
if (scale == 1.f || mWorldPanning) return rowoffset;
|
|
|
|
else return rowoffset / scale;
|
2018-11-17 17:24:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
float FTexCoordInfo::TextureOffset(float textureoffset) const
|
|
|
|
{
|
|
|
|
float scale = fabs(mScale.X);
|
2018-12-14 21:34:28 +00:00
|
|
|
if (scale == 1.f || mWorldPanning) return textureoffset;
|
|
|
|
else return textureoffset / scale;
|
2018-11-17 17:24:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Returns the size for which texture offset coordinates are used.
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
float FTexCoordInfo::TextureAdjustWidth() const
|
|
|
|
{
|
|
|
|
if (mWorldPanning)
|
|
|
|
{
|
|
|
|
float tscale = fabs(mTempScale.X);
|
|
|
|
if (tscale == 1.f) return (float)mRenderWidth;
|
|
|
|
else return mWidth / fabs(tscale);
|
|
|
|
}
|
|
|
|
else return (float)mWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Retrieve texture coordinate info for per-wall scaling
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void FTexCoordInfo::GetFromTexture(FTexture *tex, float x, float y)
|
|
|
|
{
|
|
|
|
if (x == 1.f)
|
|
|
|
{
|
|
|
|
mRenderWidth = tex->GetScaledWidth();
|
|
|
|
mScale.X = (float)tex->Scale.X;
|
|
|
|
mTempScale.X = 1.f;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
float scale_x = x * (float)tex->Scale.X;
|
|
|
|
mRenderWidth = xs_CeilToInt(tex->GetWidth() / scale_x);
|
|
|
|
mScale.X = scale_x;
|
|
|
|
mTempScale.X = x;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (y == 1.f)
|
|
|
|
{
|
|
|
|
mRenderHeight = tex->GetScaledHeight();
|
|
|
|
mScale.Y = (float)tex->Scale.Y;
|
|
|
|
mTempScale.Y = 1.f;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
float scale_y = y * (float)tex->Scale.Y;
|
|
|
|
mRenderHeight = xs_CeilToInt(tex->GetHeight() / scale_y);
|
|
|
|
mScale.Y = scale_y;
|
|
|
|
mTempScale.Y = y;
|
|
|
|
}
|
|
|
|
if (tex->bHasCanvas)
|
|
|
|
{
|
|
|
|
mScale.Y = -mScale.Y;
|
|
|
|
mRenderHeight = -mRenderHeight;
|
|
|
|
}
|
2018-12-15 15:29:37 +00:00
|
|
|
mWorldPanning = tex->bWorldPanning || (level.flags3 & LEVEL3_FORCEWORLDPANNING);
|
2018-11-17 17:24:14 +00:00
|
|
|
mWidth = tex->GetWidth();
|
|
|
|
}
|
2018-12-07 01:31:30 +00:00
|
|
|
|
2018-12-07 02:35:10 +00:00
|
|
|
|